Skip to content

Clerk provider

ClerkState

Bases: State

Source code in custom_components/reflex_clerk_api/clerk_provider.py
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
class ClerkState(rx.State):
    is_signed_in: bool = False
    """Whether the user is logged in."""

    auth_checked: bool = False
    """Whether the auth state of the user has been checked yet.
    I.e., has Clerk sent a response to the frontend yet."""

    claims: JWTClaims | None = None
    """The JWT claims of the user, if they are logged in."""
    user_id: str | None = None
    """The clerk user ID of the user, if they are logged in."""

    # NOTE: ClassVar tells reflex it doesn't need to include these in the persisted state per instance.
    _auth_wait_timeout_seconds: ClassVar[float] = 1.0
    _secret_key: ClassVar[str | None] = None
    """The Clerk secret_key set during clerk_provider creation."""
    _on_load_events: ClassVar[dict[uuid.UUID, EventType[()]]] = {}
    _dependent_handlers: ClassVar[dict[int, EventCallback]] = {}
    _client: ClassVar[clerk_backend_api.Clerk | None] = None
    _jwk_keys: ClassVar[dict[str, Any] | None] = None
    _last_jwk_reset: ClassVar[float] = 0.0
    _claims_options: ClassVar[dict[str, Any]] = {
        # "iss": {"value": "https://<your-iss>.clerk.accounts.dev"},
        "exp": {"essential": True},
        "nbf": {"essential": True},
        # "azp": {"essential": False, "values": ["http://localhost:3000", "https://example.com"]},
    }

    @classmethod
    def register_dependent_handler(cls, handler: EventCallback) -> None:
        """Register a handler to be called any time this state updates.

        I.e. Any events that should be triggered on login/logout.
        """
        assert isinstance(handler, rx.EventHandler)
        hash_id = hash((handler.state_full_name, handler.fn))
        logging.debug(f"Dependent hash_id: {hash_id}")
        cls._dependent_handlers[hash_id] = handler

    @classmethod
    def set_auth_wait_timeout_seconds(cls, seconds: float) -> None:
        """Sets the max time to wait for initial auth check before running other on_load events.

        Note: on_load events will still be run after a timed out auth check.
        Check ClerkState.auth_checked to see if auth check is complete.
        """
        cls._auth_wait_timeout_seconds = seconds

    @classmethod
    def set_claims_options(cls, claims_options: dict[str, Any]) -> None:
        """Set the claims options for the JWT claims validation."""
        cls._claims_options = claims_options

    @property
    def client(self) -> clerk_backend_api.Clerk:
        if self._client is None:
            self._set_client()
        assert self._client is not None
        return self._client

    @rx.event
    async def set_clerk_session(self, token: str) -> EventType:
        """Manually obtain user session information via the Clerk JWT.

        This event is triggered by the frontend via the ClerkSessionSynchronizer/ClerkProvider component.
        """
        logging.debug("Setting Clerk session")
        jwks = await self._get_jwk_keys()
        try:
            decoded: JWTClaims = jwt.decode(
                token, {"keys": jwks}, claims_options=self._claims_options
            )
        except jose_errors.DecodeError as e:
            # E.g. DecodeError -- Something went wrong just getting the JWT
            # On next attempt, new JWKs will be fetched
            self.auth_error = e
            self._request_jwk_reset()
            logging.warning(f"JWT decode error: {e}")
            return ClerkState.clear_clerk_session
        try:
            # Validate the token according to the claim options (e.g. iss, exp, nbf, azp.)
            decoded.validate()
        except (jose_errors.InvalidClaimError, jose_errors.MissingClaimError) as e:
            logging.warning(f"JWT token is invalid: {e}")
            return ClerkState.clear_clerk_session

        self.is_signed_in = True
        self.claims = decoded
        self.user_id = str(decoded.get("sub"))
        self.auth_checked = True
        return list(self._dependent_handlers.values())

    @rx.event
    def clear_clerk_session(self) -> EventType:
        """Clear the Clerk session information.

        This event is triggered by the frontend via the ClerkSessionSynchronizer/ClerkProvider component.
        """
        logging.debug("Clearing Clerk session")
        self.reset()
        self.auth_checked = True
        return list(self._dependent_handlers.values())

    @rx.event(background=True)
    async def wait_for_auth_check(self, uid: uuid.UUID | str) -> EventType:
        """Wait for the Clerk authentication to complete (event sent from frontend).

        Can't just use a blocking wait_for_auth_check because we are really waiting for the frontend event trigger to run, so we need to not block that while we wait.

        This can then return on_load events once auth_checked is True.
        """
        uid = uuid.UUID(uid) if isinstance(uid, str) else uid
        logging.debug(f"Waiting for auth check: {uid} ({type(uid)})")

        on_loads = self._on_load_events.get(uid, None)
        if on_loads is None:
            logging.warning("Waited for auth, but no on_load events registered.")
            on_loads = []

        start_time = time.time()
        while time.time() - start_time < self._auth_wait_timeout_seconds:
            if self.auth_checked:
                logging.debug("Auth check complete")
                return on_loads
            logging.debug("...waiting for auth...")
            # TODO: Ideally, wait on some event instead of sleeping
            await asyncio.sleep(0.05)
        logging.warning("Auth check timed out")
        return on_loads

    @classmethod
    def _set_secret_key(cls, secret_key: str) -> None:
        if not secret_key:
            raise MissingSecretKeyError("secret_key must be set (and not empty)")
        cls._secret_key = secret_key

    @classmethod
    def _set_on_load_events(cls, uid: uuid.UUID, on_load_events: EventType[()]) -> None:
        logging.debug(f"Registing on_load events: {uid}")
        cls._on_load_events[uid] = on_load_events

    @classmethod
    def _set_client(cls) -> None:
        if cls._secret_key:
            secret_key = cls._secret_key
        else:
            if "CLERK_SECRET_KEY" not in os.environ:
                raise MissingSecretKeyError(
                    "CLERK_SECRET_KEY either needs to be passed into clerk_provider(...) or set as an environment variable."
                )
            secret_key = os.environ["CLERK_SECRET_KEY"]
        client = clerk_backend_api.Clerk(bearer_auth=secret_key)
        cls._client = client

    @classmethod
    def _set_jwk_keys(cls, keys: dict[str, Any] | None) -> None:
        cls._jwk_keys = keys

    @classmethod
    def _request_jwk_reset(cls) -> None:
        """Reset the JWK keys so they will be re-fetched on next attempt.

        Only do so if it has been a while since last reset (to prevent malicious tokens from forcing
        constant re-fetching).
        """
        now = time.time()
        if now - cls._last_jwk_reset < 10:
            logging.warning("JWK reset requested too soon")
            return
        cls._last_jwk_reset = time.time()
        cls._jwk_keys = None

    async def _get_jwk_keys(self) -> dict[str, Any]:
        """Get the JWK keys from the Clerk API.

        Note: Cannot be a property because it requires async call to populate.
        Only needs to be done once (will be refreshed on errors).
        """
        if self._jwk_keys:
            return self._jwk_keys
        jwks = await self.client.jwks.get_jwks_async()
        assert jwks is not None
        assert jwks.keys is not None
        keys = jwks.model_dump()["keys"]
        self._set_jwk_keys(keys)
        return keys

is_signed_in = False class-attribute instance-attribute

Whether the user is logged in.

auth_checked = False class-attribute instance-attribute

Whether the auth state of the user has been checked yet. I.e., has Clerk sent a response to the frontend yet.

claims = None class-attribute instance-attribute

The JWT claims of the user, if they are logged in.

user_id = None class-attribute instance-attribute

The clerk user ID of the user, if they are logged in.

register_dependent_handler(handler) classmethod

Register a handler to be called any time this state updates.

I.e. Any events that should be triggered on login/logout.

Source code in custom_components/reflex_clerk_api/clerk_provider.py
61
62
63
64
65
66
67
68
69
70
@classmethod
def register_dependent_handler(cls, handler: EventCallback) -> None:
    """Register a handler to be called any time this state updates.

    I.e. Any events that should be triggered on login/logout.
    """
    assert isinstance(handler, rx.EventHandler)
    hash_id = hash((handler.state_full_name, handler.fn))
    logging.debug(f"Dependent hash_id: {hash_id}")
    cls._dependent_handlers[hash_id] = handler

set_auth_wait_timeout_seconds(seconds) classmethod

Sets the max time to wait for initial auth check before running other on_load events.

Note: on_load events will still be run after a timed out auth check. Check ClerkState.auth_checked to see if auth check is complete.

Source code in custom_components/reflex_clerk_api/clerk_provider.py
72
73
74
75
76
77
78
79
@classmethod
def set_auth_wait_timeout_seconds(cls, seconds: float) -> None:
    """Sets the max time to wait for initial auth check before running other on_load events.

    Note: on_load events will still be run after a timed out auth check.
    Check ClerkState.auth_checked to see if auth check is complete.
    """
    cls._auth_wait_timeout_seconds = seconds

set_claims_options(claims_options) classmethod

Set the claims options for the JWT claims validation.

Source code in custom_components/reflex_clerk_api/clerk_provider.py
81
82
83
84
@classmethod
def set_claims_options(cls, claims_options: dict[str, Any]) -> None:
    """Set the claims options for the JWT claims validation."""
    cls._claims_options = claims_options

set_clerk_session(token) async

Manually obtain user session information via the Clerk JWT.

This event is triggered by the frontend via the ClerkSessionSynchronizer/ClerkProvider component.

Source code in custom_components/reflex_clerk_api/clerk_provider.py
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
@rx.event
async def set_clerk_session(self, token: str) -> EventType:
    """Manually obtain user session information via the Clerk JWT.

    This event is triggered by the frontend via the ClerkSessionSynchronizer/ClerkProvider component.
    """
    logging.debug("Setting Clerk session")
    jwks = await self._get_jwk_keys()
    try:
        decoded: JWTClaims = jwt.decode(
            token, {"keys": jwks}, claims_options=self._claims_options
        )
    except jose_errors.DecodeError as e:
        # E.g. DecodeError -- Something went wrong just getting the JWT
        # On next attempt, new JWKs will be fetched
        self.auth_error = e
        self._request_jwk_reset()
        logging.warning(f"JWT decode error: {e}")
        return ClerkState.clear_clerk_session
    try:
        # Validate the token according to the claim options (e.g. iss, exp, nbf, azp.)
        decoded.validate()
    except (jose_errors.InvalidClaimError, jose_errors.MissingClaimError) as e:
        logging.warning(f"JWT token is invalid: {e}")
        return ClerkState.clear_clerk_session

    self.is_signed_in = True
    self.claims = decoded
    self.user_id = str(decoded.get("sub"))
    self.auth_checked = True
    return list(self._dependent_handlers.values())

clear_clerk_session()

Clear the Clerk session information.

This event is triggered by the frontend via the ClerkSessionSynchronizer/ClerkProvider component.

Source code in custom_components/reflex_clerk_api/clerk_provider.py
125
126
127
128
129
130
131
132
133
134
@rx.event
def clear_clerk_session(self) -> EventType:
    """Clear the Clerk session information.

    This event is triggered by the frontend via the ClerkSessionSynchronizer/ClerkProvider component.
    """
    logging.debug("Clearing Clerk session")
    self.reset()
    self.auth_checked = True
    return list(self._dependent_handlers.values())

wait_for_auth_check(uid) async

Wait for the Clerk authentication to complete (event sent from frontend).

Can't just use a blocking wait_for_auth_check because we are really waiting for the frontend event trigger to run, so we need to not block that while we wait.

This can then return on_load events once auth_checked is True.

Source code in custom_components/reflex_clerk_api/clerk_provider.py
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
@rx.event(background=True)
async def wait_for_auth_check(self, uid: uuid.UUID | str) -> EventType:
    """Wait for the Clerk authentication to complete (event sent from frontend).

    Can't just use a blocking wait_for_auth_check because we are really waiting for the frontend event trigger to run, so we need to not block that while we wait.

    This can then return on_load events once auth_checked is True.
    """
    uid = uuid.UUID(uid) if isinstance(uid, str) else uid
    logging.debug(f"Waiting for auth check: {uid} ({type(uid)})")

    on_loads = self._on_load_events.get(uid, None)
    if on_loads is None:
        logging.warning("Waited for auth, but no on_load events registered.")
        on_loads = []

    start_time = time.time()
    while time.time() - start_time < self._auth_wait_timeout_seconds:
        if self.auth_checked:
            logging.debug("Auth check complete")
            return on_loads
        logging.debug("...waiting for auth...")
        # TODO: Ideally, wait on some event instead of sleeping
        await asyncio.sleep(0.05)
    logging.warning("Auth check timed out")
    return on_loads

ClerkUser

Bases: State

Convenience class for using Clerk User information.

This only contains a subset of the information available. Create your own state if you need more.

Note: For this to be updated on login/logout events, it must be registered on the ClerkState.

Source code in custom_components/reflex_clerk_api/clerk_provider.py
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
class ClerkUser(rx.State):
    """Convenience class for using Clerk User information.

    This only contains a subset of the information available. Create your own state if you need more.

    Note: For this to be updated on login/logout events, it must be registered on the ClerkState.
    """

    first_name: str = ""
    last_name: str = ""
    username: str = ""
    email_address: str = ""
    has_image: bool = False
    image_url: str = ""

    # Set to True when the state is registered on the ClerkState to avoid registering it multiple times.
    _is_registered: ClassVar[bool] = False

    @rx.event
    async def load_user(self) -> None:
        try:
            user: clerk_backend_api.models.User = await get_user(self)
        except MissingUserError:
            logging.debug("Clearing user state")
            self.reset()
            return

        logging.debug("Updating user state")
        self.first_name = (
            user.first_name
            if user.first_name and user.first_name != clerk_backend_api.UNSET
            else ""
        )
        self.last_name = (
            user.last_name
            if user.last_name and user.last_name != clerk_backend_api.UNSET
            else ""
        )
        self.username = (
            user.username
            if user.username and user.username != clerk_backend_api.UNSET
            else ""
        )
        self.email_address = (
            user.email_addresses[0].email_address if user.email_addresses else ""
        )
        self.has_image = True if user.has_image is True else False
        self.image_url = user.image_url or ""

ClerkSessionSynchronizer

Bases: Component

ClerkSessionSynchronizer component.

This is slightly adapted from Elliot Kroo's reflex-clerk.

Source code in custom_components/reflex_clerk_api/clerk_provider.py
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
class ClerkSessionSynchronizer(rx.Component):
    """ClerkSessionSynchronizer component.

    This is slightly adapted from Elliot Kroo's reflex-clerk.
    """

    tag = "ClerkSessionSynchronizer"

    def add_imports(
        self,
    ) -> rx.ImportDict:
        addl_imports: rx.ImportDict = {
            "@clerk/clerk-react": ["useAuth"],
            "react": ["useContext", "useEffect"],
            "/utils/context": ["EventLoopContext"],
            "/utils/state": ["Event"],
        }
        return addl_imports

    def add_custom_code(self) -> list[str]:
        clerk_state_name = ClerkState.get_full_name()

        return [
            """
function ClerkSessionSynchronizer({ children }) {
  const { getToken, isLoaded, isSignedIn } = useAuth()
  const [ addEvents, connectErrors ] = useContext(EventLoopContext)

  useEffect(() => {
      if (isLoaded && !!addEvents) {
        if (isSignedIn) {
          getToken().then(token => {
            addEvents([Event("%s.set_clerk_session", {token})])
          })
        } else {
          addEvents([Event("%s.clear_clerk_session")])
        }
      }
  }, [isSignedIn])

  return (
      <>{children}</>
  )
}
"""
            % (clerk_state_name, clerk_state_name)
        ]

ClerkProvider

Bases: ClerkBase

ClerkProvider component.

Source code in custom_components/reflex_clerk_api/clerk_provider.py
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
class ClerkProvider(ClerkBase):
    """ClerkProvider component."""

    # The React component tag.
    tag = "ClerkProvider"

    # NOTE: This might be relevant to getting apperance.base_theme to work.
    # lib_dependencies: list[str] = ["@clerk/themes"]
    # def add_imports(self) -> rx.ImportDict:
    #     return {
    #         "@clerk/themes": ["dark", "neobrutalism", "shadesOfPurple"],
    #     }

    # The props of the React component.
    # Note: when Reflex compiles the component to Javascript,
    # `snake_case` property names are automatically formatted as `camelCase`.
    # The prop names may be defined in `camelCase` as well.
    # some_prop: rx.Var[str] = "some default value"
    # some_other_prop: rx.Var[int] = 1

    # Event triggers declaration if any.
    # Below is equivalent to merging `{ "on_change": lambda e: [e] }`
    # onto the default event triggers of parent/base Component.
    # The function defined for the `on_change` trigger maps event for the javascript
    # trigger to what will be passed to the backend event handler function.
    # on_change: rx.EventHandler[lambda e: [e]]

    after_multi_session_single_sign_out_url: str = ""
    """The URL to navigate to after a successful sign-out from multiple sessions."""

    after_sign_out_url: str = ""
    """The full URL or path to navigate to after a successful sign-out."""

    allowed_redirect_origins: list[str | str] = []
    """An optional list of domains to validate user-provided redirect URLs against."""

    allowed_redirect_protocols: list[str] = []
    """An optional list of protocols to validate user-provided redirect URLs against."""

    # NOTE: `apperance.base_theme` does not work yet.
    appearance: Appearance | None = None
    """Optional object to style your components. Will only affect Clerk components."""

    clerk_js_url: str = ""
    """Define the URL that @clerk/clerk-js should be hot-loaded from."""

    clerk_js_variant: str | None = None
    """If your web application only uses control components, set this to 'headless'."""

    clerk_js_version: str = ""
    """Define the npm version for @clerk/clerk-js."""

    # domain: str | JSCallable[[str], bool] = ""
    domain: str = ""
    """Required if your application is a satellite application. Sets the domain."""

    dynamic: bool = False
    """(For Next.js only) Indicates whether Clerk should make dynamic auth data available."""

    # initial_state: InitialState | None = None
    # """Provide an initial state of the Clerk client during server-side rendering."""

    # is_satellite: bool | JSCallable[[str], bool] = False
    is_satellite: bool = False
    """Whether the application is a satellite application."""

    # Not implemented
    # localization: Localization | None = None
    # See https://clerk.com/docs/customization/localization#clerk-localizations for more info.
    # """Optional object to localize your components. Will only affect Clerk components."""

    nonce: str = ""
    """Nonce value passed to the @clerk/clerk-js script tag for CSP implementation."""

    publishable_key: str = ""
    """The Clerk Publishable Key for your instance, found on the API keys page in the Clerk Dashboard."""

    # proxy_url: str | JSCallable[[str], str] = ""
    proxy_url: str = ""
    """The URL of the proxy server to use for Clerk API requests."""

    router_push: JSCallable[[str], None | Any] | None = None
    """A function to push a new route into the history stack for navigation."""

    router_replace: JSCallable[[str], None | Any] | None = None
    """A function to replace the current route in the history stack for navigation."""

    # sdk_metadata: dict[str, str] = {"name": "", "version": "", "environment": ""}
    # """Contains information about the SDK that the host application is using."""

    # select_initial_session: Callable[[Any], None | Any] | None = None
    # """Function to override the default behavior of using the last active session during client initialization."""

    sign_in_fallback_redirect_url: str = "/"
    """The fallback URL to redirect to after the user signs in if there's no redirect_url in the path."""

    sign_up_fallback_redirect_url: str = "/"
    """The fallback URL to redirect to after the user signs up if there's no redirect_url in the path."""

    sign_in_force_redirect_url: str = ""
    """URL to always redirect to after the user signs in."""

    sign_up_force_redirect_url: str = ""
    """URL to always redirect to after the user signs up."""

    sign_in_url: str = ""
    """URL used for any redirects that might happen, pointing to your primary application on the client-side."""

    sign_up_url: str = ""
    """URL used for any redirects that might happen, pointing to your primary application on the client-side."""

    standard_browser: bool = True
    """Indicates whether ClerkJS assumes cookies can be set (browser setup)."""

    support_email: str = ""
    """Optional support email for display in authentication screens."""

    sync_host: str = ""
    """URL of the web application that the Chrome Extension will sync the authentication state from."""

    telemetry: bool | dict[str, bool] | None = None
    """Controls whether Clerk will collect telemetry data."""

    touch_session: bool = True
    """Indicates whether the Clerk Frontend API touch endpoint is called during page focus to keep the last active session alive."""

    waitlist_url: str = ""
    """The full URL or path to the waitlist page."""

    @classmethod
    def create(cls, *children, **props) -> "ClerkProvider":
        return super().create(*children, **props)

    def add_custom_code(self) -> list[str]:
        return []

after_multi_session_single_sign_out_url = '' class-attribute instance-attribute

The URL to navigate to after a successful sign-out from multiple sessions.

after_sign_out_url = '' class-attribute instance-attribute

The full URL or path to navigate to after a successful sign-out.

allowed_redirect_origins = [] class-attribute instance-attribute

An optional list of domains to validate user-provided redirect URLs against.

allowed_redirect_protocols = [] class-attribute instance-attribute

An optional list of protocols to validate user-provided redirect URLs against.

appearance = None class-attribute instance-attribute

Optional object to style your components. Will only affect Clerk components.

clerk_js_url = '' class-attribute instance-attribute

Define the URL that @clerk/clerk-js should be hot-loaded from.

clerk_js_variant = None class-attribute instance-attribute

If your web application only uses control components, set this to 'headless'.

clerk_js_version = '' class-attribute instance-attribute

Define the npm version for @clerk/clerk-js.

domain = '' class-attribute instance-attribute

Required if your application is a satellite application. Sets the domain.

dynamic = False class-attribute instance-attribute

(For Next.js only) Indicates whether Clerk should make dynamic auth data available.

is_satellite = False class-attribute instance-attribute

Whether the application is a satellite application.

nonce = '' class-attribute instance-attribute

Nonce value passed to the @clerk/clerk-js script tag for CSP implementation.

publishable_key = '' class-attribute instance-attribute

The Clerk Publishable Key for your instance, found on the API keys page in the Clerk Dashboard.

proxy_url = '' class-attribute instance-attribute

The URL of the proxy server to use for Clerk API requests.

router_push = None class-attribute instance-attribute

A function to push a new route into the history stack for navigation.

router_replace = None class-attribute instance-attribute

A function to replace the current route in the history stack for navigation.

sign_in_fallback_redirect_url = '/' class-attribute instance-attribute

The fallback URL to redirect to after the user signs in if there's no redirect_url in the path.

sign_up_fallback_redirect_url = '/' class-attribute instance-attribute

The fallback URL to redirect to after the user signs up if there's no redirect_url in the path.

sign_in_force_redirect_url = '' class-attribute instance-attribute

URL to always redirect to after the user signs in.

sign_up_force_redirect_url = '' class-attribute instance-attribute

URL to always redirect to after the user signs up.

sign_in_url = '' class-attribute instance-attribute

URL used for any redirects that might happen, pointing to your primary application on the client-side.

sign_up_url = '' class-attribute instance-attribute

URL used for any redirects that might happen, pointing to your primary application on the client-side.

standard_browser = True class-attribute instance-attribute

Indicates whether ClerkJS assumes cookies can be set (browser setup).

support_email = '' class-attribute instance-attribute

Optional support email for display in authentication screens.

sync_host = '' class-attribute instance-attribute

URL of the web application that the Chrome Extension will sync the authentication state from.

telemetry = None class-attribute instance-attribute

Controls whether Clerk will collect telemetry data.

touch_session = True class-attribute instance-attribute

Indicates whether the Clerk Frontend API touch endpoint is called during page focus to keep the last active session alive.

waitlist_url = '' class-attribute instance-attribute

The full URL or path to the waitlist page.

on_load(on_load_events)

Use this to wrap any on_load events that should happen after Clerk has checked authentication.

Parameters:

Name Type Description Default
on_load_events EventType[] | None

The events to run after authentication is checked.

required

Examples:

app.add_page(..., on_load=clerk.on_load())

Source code in custom_components/reflex_clerk_api/clerk_provider.py
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
def on_load(on_load_events: EventType[()] | None) -> list[IndividualEventType[()]]:
    """Use this to wrap any on_load events that should happen after Clerk has checked authentication.

    Args:
        on_load_events: The events to run after authentication is checked.

    Examples:
        app.add_page(..., on_load=clerk.on_load(<events>))
    """
    if on_load_events is None:
        return []
    on_load_list = (
        on_load_events if isinstance(on_load_events, list) else [on_load_events]
    )

    # Add the on_load events to a registry in the ClerkState instead of actually passing them to on_load.
    #  Then, the wait_for_auth_check event will return the on_load events once auth_checked is True.
    #  Can't just use a blocking wait_for_auth_check because we are really waiting for the frontend event trigger to run,
    #  so we need to not block that while we wait.
    uid = uuid.uuid4()
    ClerkState._set_on_load_events(uid, on_load_list)
    return [ClerkState.wait_for_auth_check(uid)]

get_user(current_state) async

Get the User object from Clerk given the currently logged in user.

Note: Must be used within an event handler in order to get the appropriate clerk_state.

Parameters:

Name Type Description Default
current_state State

The self state from the current event handler.

required

Examples:

class State(rx.State):
    @rx.event
    async def handle_getting_user_email(self) -> EventType:
        user = await clerk.get_user(self)
        return rx.toast.info(f"User: {user.email}")
Source code in custom_components/reflex_clerk_api/clerk_provider.py
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
async def get_user(current_state: rx.State) -> clerk_backend_api.models.User:
    """Get the User object from Clerk given the currently logged in user.

    Note: Must be used within an event handler in order to get the appropriate clerk_state.

    Args:
        current_state: The `self` state from the current event handler.

    Examples:

    ```python
    class State(rx.State):
        @rx.event
        async def handle_getting_user_email(self) -> EventType:
            user = await clerk.get_user(self)
            return rx.toast.info(f"User: {user.email}")
    ```
    """
    clerk_state = await _get_state_within_handler(current_state, ClerkState)
    user_id = clerk_state.user_id
    if user_id is None:
        raise MissingUserError("No user_id to get user for")
    user = await clerk_state.client.users.get_async(user_id=user_id)
    if user is None:
        raise MissingUserError("No user found")
    return user

register_on_auth_change_handler(handler)

Register a handler to be called any time the user logs in or out.

Parameters:

Name Type Description Default
handler EventCallback

The event handler function to be called.

required
Source code in custom_components/reflex_clerk_api/clerk_provider.py
546
547
548
549
550
551
552
def register_on_auth_change_handler(handler: EventCallback) -> None:
    """Register a handler to be called any time the user logs in or out.

    Args:
        handler: The event handler function to be called.
    """
    ClerkState.register_dependent_handler(handler)

clerk_provider(*children, publishable_key, secret_key=None, register_user_state=False, appearance=None, **props)

Create a ClerkProvider component to wrap your app/page that uses clerk authentication.

Note: can also use wrap_app to wrap the entire app.

Parameters:

Name Type Description Default
children

The children components to wrap.

()
publishable_key str

The Clerk Publishable Key for your instance.

required
secret_key str | None

Your Clerk app's Secret Key, which you can find in the Clerk Dashboard. It will be prefixed with sk_test_ in development instances and sk_live_ in production instances. Do not expose this on the frontend with a public environment variable.

None
register_user_state bool

Whether to register the ClerkUser state to automatically load user information on login.

False
appearance Appearance | None

Optional object to style your components. Will only affect Clerk components.

None
Source code in custom_components/reflex_clerk_api/clerk_provider.py
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
def clerk_provider(
    *children,
    publishable_key: str,
    secret_key: str | None = None,
    register_user_state: bool = False,
    appearance: Appearance | None = None,
    **props,
) -> rx.Component:
    """
    Create a ClerkProvider component to wrap your app/page that uses clerk authentication.

    Note: can also use `wrap_app` to wrap the entire app.

    Args:
        children: The children components to wrap.
        publishable_key: The Clerk Publishable Key for your instance.
        secret_key: Your Clerk app's Secret Key, which you can find in the Clerk Dashboard. It will be prefixed with sk_test_ in development instances and sk_live_ in production instances. Do not expose this on the frontend with a public environment variable.
        register_user_state: Whether to register the ClerkUser state to automatically load user information on login.
        appearance: Optional object to style your components. Will only affect Clerk components.
    """
    if secret_key:
        ClerkState._set_secret_key(secret_key)

    if register_user_state:
        register_on_auth_change_handler(ClerkUser.load_user)

    return ClerkProvider.create(
        ClerkSessionSynchronizer.create(*children),
        publishable_key=publishable_key,
        appearance=appearance,
        **props,
    )

wrap_app(app, publishable_key, secret_key=None, register_user_state=False, appearance=None, **props)

Wraps the entire app with the ClerkProvider.

For multi-page apps where all pages require Clerk authentication components (including knowing if the user is not signed in).

Parameters:

Name Type Description Default
app App

The Reflex app to wrap.

required
publishable_key str

The Clerk Publishable Key for your instance.

required
secret_key str | None

Your Clerk app's Secret Key. (not needed for frontend only)

None
register_user_state bool

Whether to register the ClerkUser state to automatically load user information on login.

False
Source code in custom_components/reflex_clerk_api/clerk_provider.py
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
def wrap_app(
    app: rx.App,
    publishable_key: str,
    secret_key: str | None = None,
    register_user_state: bool = False,
    appearance: Appearance | None = None,
    **props,
) -> rx.App:
    """Wraps the entire app with the ClerkProvider.

    For multi-page apps where all pages require Clerk authentication components (including knowing if the user
    is **not** signed in).

    Args:
        app: The Reflex app to wrap.
        publishable_key: The Clerk Publishable Key for your instance.
        secret_key: Your Clerk app's Secret Key. (not needed for frontend only)
        register_user_state: Whether to register the ClerkUser state to automatically load user information on login.
    """
    # 1 makes this the first wrapper around the content
    #  (0 would place it after, 100 would also wrap default reflex wrappers)
    app.app_wraps[(1, "ClerkProvider")] = lambda _: clerk_provider(
        publishable_key=publishable_key,
        secret_key=secret_key,
        register_user_state=register_user_state,
        appearance=appearance,
        **props,
    )
    return app