Skip to content

Commit

Permalink
feat(remix): Refactor getEnvVariable (#1862)
Browse files Browse the repository at this point in the history
* feat(remix): Refactor getEnvVariable to support Cloudflare Pages

* chore(repo): Adds changeset

* chore(repo): Update Changeset

* fix(remix): Updated all uses of getEnvVariable to use the Remix context
  • Loading branch information
octoper authored Oct 14, 2023
1 parent a11f962 commit 0175607
Show file tree
Hide file tree
Showing 6 changed files with 51 additions and 46 deletions.
5 changes: 5 additions & 0 deletions .changeset/thin-geese-peel.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@clerk/remix': patch
---

Internal improvements for retrieving environment variables.
47 changes: 15 additions & 32 deletions packages/remix/src/ssr/authenticateRequest.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,10 +27,10 @@ export function authenticateRequest(args: LoaderFunctionArgs, opts: RootAuthLoad
// 2. Then try from process.env if exists (Node).
// 3. Then try from globalThis (Cloudflare Workers).
// 4. Then from loader context (Cloudflare Pages).
const secretKey = opts.secretKey || getEnvVariable('CLERK_SECRET_KEY') || (context?.CLERK_SECRET_KEY as string) || '';
const apiKey = opts.apiKey || getEnvVariable('CLERK_API_KEY') || (context?.CLERK_API_KEY as string) || '';
const secretKey = opts.secretKey || getEnvVariable('CLERK_SECRET_KEY', context) || '';
const apiKey = opts.apiKey || getEnvVariable('CLERK_API_KEY', context) || '';
if (apiKey) {
if (getEnvVariable('CLERK_API_KEY')) {
if (getEnvVariable('CLERK_API_KEY', context)) {
deprecated('CLERK_API_KEY', 'Use `CLERK_SECRET_KEY` instead.');
} else {
deprecated('apiKey', 'Use `secretKey` instead.');
Expand All @@ -41,41 +41,34 @@ export function authenticateRequest(args: LoaderFunctionArgs, opts: RootAuthLoad
throw new Error(noSecretKeyOrApiKeyError);
}

const frontendApi =
opts.frontendApi || getEnvVariable('CLERK_FRONTEND_API') || (context?.CLERK_FRONTEND_API as string) || '';
const frontendApi = opts.frontendApi || getEnvVariable('CLERK_FRONTEND_API', context) || '';
if (frontendApi) {
if (getEnvVariable('CLERK_FRONTEND_API')) {
if (getEnvVariable('CLERK_FRONTEND_API', context)) {
deprecated('CLERK_FRONTEND_API', 'Use `CLERK_PUBLISHABLE_KEY` instead.');
} else {
deprecated('frontendApi', 'Use `publishableKey` instead.');
}
}

const publishableKey =
opts.publishableKey || getEnvVariable('CLERK_PUBLISHABLE_KEY') || (context?.CLERK_PUBLISHABLE_KEY as string) || '';
const publishableKey = opts.publishableKey || getEnvVariable('CLERK_PUBLISHABLE_KEY', context) || '';

const jwtKey = opts.jwtKey || getEnvVariable('CLERK_JWT_KEY') || (context?.CLERK_JWT_KEY as string);
const jwtKey = opts.jwtKey || getEnvVariable('CLERK_JWT_KEY', context);

const apiUrl = getEnvVariable('CLERK_API_URL') || (context?.CLERK_API_URL as string);
const apiUrl = getEnvVariable('CLERK_API_URL', context);

const domain =
handleValueOrFn(opts.domain, new URL(request.url)) ||
getEnvVariable('CLERK_DOMAIN') ||
(context?.CLERK_DOMAIN as string) ||
'';
const domain = handleValueOrFn(opts.domain, new URL(request.url)) || getEnvVariable('CLERK_DOMAIN', context) || '';

const isSatellite =
handleValueOrFn(opts.isSatellite, new URL(request.url)) ||
getEnvVariable('CLERK_IS_SATELLITE') === 'true' ||
(context?.CLERK_IS_SATELLITE as string) === 'true' ||
getEnvVariable('CLERK_IS_SATELLITE', context) === 'true' ||
false;

const requestURL = buildRequestUrl(request);

const relativeOrAbsoluteProxyUrl = handleValueOrFn(
opts?.proxyUrl,
requestURL,
getEnvVariable('CLERK_PROXY_URL') || (context?.CLERK_PROXY_URL as string),
getEnvVariable('CLERK_PROXY_URL', context),
);

let proxyUrl;
Expand All @@ -85,23 +78,13 @@ export function authenticateRequest(args: LoaderFunctionArgs, opts: RootAuthLoad
proxyUrl = relativeOrAbsoluteProxyUrl;
}

const signInUrl =
opts.signInUrl || getEnvVariable('CLERK_SIGN_IN_URL') || (context?.CLERK_SIGN_IN_URL as string) || '';
const signInUrl = opts.signInUrl || getEnvVariable('CLERK_SIGN_IN_URL', context) || '';

const signUpUrl =
opts.signUpUrl || getEnvVariable('CLERK_SIGN_UP_URL') || (context?.CLERK_SIGN_UP_URL as string) || '';
const signUpUrl = opts.signUpUrl || getEnvVariable('CLERK_SIGN_UP_URL', context) || '';

const afterSignInUrl =
opts.afterSignInUrl ||
getEnvVariable('CLERK_AFTER_SIGN_IN_URL') ||
(context?.CLERK_AFTER_SIGN_IN_URL as string) ||
'';
const afterSignInUrl = opts.afterSignInUrl || getEnvVariable('CLERK_AFTER_SIGN_IN_URL', context) || '';

const afterSignUpUrl =
opts.afterSignUpUrl ||
getEnvVariable('CLERK_AFTER_SIGN_UP_URL') ||
(context?.CLERK_AFTER_SIGN_UP_URL as string) ||
'';
const afterSignUpUrl = opts.afterSignUpUrl || getEnvVariable('CLERK_AFTER_SIGN_UP_URL', context) || '';

if (isSatellite && !proxyUrl && !domain) {
throw new Error(satelliteAndMissingProxyUrlAndDomain);
Expand Down
2 changes: 1 addition & 1 deletion packages/remix/src/ssr/getAuth.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ export async function getAuth(args: LoaderFunctionArgs, opts?: GetAuthOptions):
}

if (requestState.isInterstitial) {
throw interstitialJsonResponse(requestState, { loader: 'nested' });
throw interstitialJsonResponse(requestState, { loader: 'nested' }, args.context);
}

return sanitizeAuthObject(requestState.toAuth());
Expand Down
8 changes: 4 additions & 4 deletions packages/remix/src/ssr/rootAuthLoader.ts
Original file line number Diff line number Diff line change
Expand Up @@ -53,12 +53,12 @@ export const rootAuthLoader: RootAuthLoader = async (
}

if (requestState.isInterstitial) {
throw interstitialJsonResponse(requestState, { loader: 'root' });
throw interstitialJsonResponse(requestState, { loader: 'root' }, args.context);
}

if (!handler) {
// if the user did not provide a handler, simply inject requestState into an empty response
return injectRequestStateIntoResponse(new Response(JSON.stringify({})), requestState);
return injectRequestStateIntoResponse(new Response(JSON.stringify({})), requestState, args.context);
}

const handlerResult = await handler(injectAuthIntoRequest(args, sanitizeAuthObject(requestState.toAuth())));
Expand All @@ -72,7 +72,7 @@ export const rootAuthLoader: RootAuthLoader = async (
}
// clone and try to inject requestState into all json-like responses
// if this fails, the user probably didn't return a json object or a valid json string
return injectRequestStateIntoResponse(handlerResult, requestState);
return injectRequestStateIntoResponse(handlerResult, requestState, args.context);
} catch (e) {
throw new Error(invalidRootLoaderCallbackReturn);
}
Expand All @@ -81,5 +81,5 @@ export const rootAuthLoader: RootAuthLoader = async (
// if the return value of the user's handler is null or a plain object, create an empty response to inject Clerk's state into
const responseBody = JSON.stringify(handlerResult ?? {});

return injectRequestStateIntoResponse(new Response(responseBody), requestState);
return injectRequestStateIntoResponse(new Response(responseBody), requestState, args.context);
};
21 changes: 15 additions & 6 deletions packages/remix/src/ssr/utils.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import type { AuthObject, RequestState } from '@clerk/backend';
import { constants, debugRequestState, loadInterstitialFromLocal } from '@clerk/backend';
import type { AppLoadContext } from '@remix-run/server-runtime';
import { json } from '@remix-run/server-runtime';
import cookie from 'cookie';

Expand Down Expand Up @@ -75,7 +76,11 @@ export const unknownResponse = (requestState: RequestState) => {
return json(null, { status: 401, headers: observabilityHeadersFromRequestState(requestState) });
};

export const interstitialJsonResponse = (requestState: RequestState, opts: { loader: 'root' | 'nested' }) => {
export const interstitialJsonResponse = (
requestState: RequestState,
opts: { loader: 'root' | 'nested' },
context: AppLoadContext,
) => {
return json(
wrapWithClerkState({
__loader: opts.loader,
Expand All @@ -85,8 +90,8 @@ export const interstitialJsonResponse = (requestState: RequestState, opts: { loa
publishableKey: requestState.publishableKey,
// TODO: This needs to be the version of clerk/remix not clerk/react
// pkgVersion: LIB_VERSION,
clerkJSUrl: getEnvVariable('CLERK_JS'),
clerkJSVersion: getEnvVariable('CLERK_JS_VERSION'),
clerkJSUrl: getEnvVariable('CLERK_JS', context),
clerkJSVersion: getEnvVariable('CLERK_JS_VERSION', context),
proxyUrl: requestState.proxyUrl,
isSatellite: requestState.isSatellite,
domain: requestState.domain,
Expand All @@ -97,7 +102,11 @@ export const interstitialJsonResponse = (requestState: RequestState, opts: { loa
);
};

export const injectRequestStateIntoResponse = async (response: Response, requestState: RequestState) => {
export const injectRequestStateIntoResponse = async (
response: Response,
requestState: RequestState,
context: AppLoadContext,
) => {
// eslint-disable-next-line @typescript-eslint/no-unused-vars
const { reason, message, isSignedIn, isInterstitial, ...rest } = requestState;
const clone = response.clone();
Expand All @@ -114,8 +123,8 @@ export const injectRequestStateIntoResponse = async (response: Response, request
__afterSignInUrl: requestState.afterSignInUrl,
__afterSignUpUrl: requestState.afterSignUpUrl,
__clerk_debug: debugRequestState(requestState),
__clerkJSUrl: getEnvVariable('CLERK_JS'),
__clerkJSVersion: getEnvVariable('CLERK_JS_VERSION'),
__clerkJSUrl: getEnvVariable('CLERK_JS', context),
__clerkJSVersion: getEnvVariable('CLERK_JS_VERSION', context),
});
// set the correct content-type header in case the user returned a `Response` directly
// without setting the header, instead of using the `json()` helper
Expand Down
14 changes: 11 additions & 3 deletions packages/remix/src/utils.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
import type { AppLoadContext } from '@remix-run/server-runtime';

import type { ClerkState } from './client/types';
import { invalidClerkStatePropError, noClerkStateError } from './errors';

Expand Down Expand Up @@ -29,16 +31,22 @@ export function assertValidClerkState(val: any): asserts val is ClerkState | und
* @param name
* @returns
*/
export const getEnvVariable = (name: string): string => {
export const getEnvVariable = (name: string, context: AppLoadContext | undefined): string => {
// Node envs
if (typeof process !== 'undefined') {
return (process.env && process.env[name]) || '';
}

// Cloudflare pages
if (typeof context !== 'undefined') {
const contextEnv = context?.env as Record<string, string>;

return contextEnv[name] || (context[name] as string) || '';
}

// Cloudflare workers
try {
// @ts-expect-error
return globalThis[name];
return globalThis[name as keyof typeof globalThis];
} catch (_) {
// This will raise an error in Cloudflare Pages
}
Expand Down

0 comments on commit 0175607

Please sign in to comment.