diff --git a/.changeset/sad-turkeys-rhyme.md b/.changeset/sad-turkeys-rhyme.md
new file mode 100644
index 00000000000..34b19fa4825
--- /dev/null
+++ b/.changeset/sad-turkeys-rhyme.md
@@ -0,0 +1,10 @@
+---
+'@clerk/clerk-js': patch
+'@clerk/types': patch
+---
+
+Add optional `isExternal` to `ApplicationLogo`
+
+Add optional `oAuthApplicationUrl` parameter to OAuth Consent mounting (which is used to provide a link to the OAuth App homepage).
+
+Harden `Link` component so it sanitizes the given `href` to avoid dangerous protocols.
diff --git a/.vscode/tasks.json b/.vscode/tasks.json
new file mode 100644
index 00000000000..71f66cc4083
--- /dev/null
+++ b/.vscode/tasks.json
@@ -0,0 +1,13 @@
+{
+ "version": "2.0.0",
+ "tasks": [
+ {
+ "type": "npm",
+ "script": "dev:sandbox",
+ "path": "packages/clerk-js",
+ "problemMatcher": [],
+ "label": "Dev: Sandbox",
+ "detail": "npm: dev:sandbox - packages/clerk-js"
+ }
+ ]
+}
\ No newline at end of file
diff --git a/packages/clerk-js/sandbox/app.ts b/packages/clerk-js/sandbox/app.ts
index db52e1ac505..15d45afc722 100644
--- a/packages/clerk-js/sandbox/app.ts
+++ b/packages/clerk-js/sandbox/app.ts
@@ -334,6 +334,8 @@ void (async () => {
scopes,
oAuthApplicationName: searchParams.get('oauth-application-name'),
redirectUrl: searchParams.get('redirect_uri'),
+ oAuthApplicationLogoUrl: searchParams.get('logo-url'),
+ oAuthApplicationUrl: searchParams.get('app-url'),
},
);
},
diff --git a/packages/clerk-js/src/ui/components/OAuthConsent/OAuthConsent.tsx b/packages/clerk-js/src/ui/components/OAuthConsent/OAuthConsent.tsx
index dc7a55fac4c..d40ed41d158 100644
--- a/packages/clerk-js/src/ui/components/OAuthConsent/OAuthConsent.tsx
+++ b/packages/clerk-js/src/ui/components/OAuthConsent/OAuthConsent.tsx
@@ -17,7 +17,7 @@ import { common } from '@/ui/styledSystem';
import { colors } from '@/ui/utils/colors';
export function OAuthConsentInternal() {
- const { scopes, oAuthApplicationName, oAuthApplicationLogoUrl, redirectUrl, onDeny, onAllow } =
+ const { scopes, oAuthApplicationName, oAuthApplicationLogoUrl, oAuthApplicationUrl, redirectUrl, onDeny, onAllow } =
useOAuthConsentContext();
const { user } = useUser();
const { applicationName, logoImageUrl } = useEnvironment().displayConfig;
@@ -46,6 +46,8 @@ export function OAuthConsentInternal() {
@@ -65,6 +67,8 @@ export function OAuthConsentInternal() {
& {
* The URL to navigate to when the logo is clicked.
*/
href?: string;
+ /**
+ * Whether the href should be treated as an external link.
+ * When true, uses a Link component with target="_blank" and proper security attributes.
+ * When false or undefined, uses RouterLink for internal navigation.
+ */
+ isExternal?: boolean;
};
export const ApplicationLogo: React.FC = (props: ApplicationLogoProps): JSX.Element | null => {
- const { src, alt, href, sx, ...rest } = props;
+ const { src, alt, href, isExternal, sx, ...rest } = props;
const imageRef = React.useRef(null);
const [loaded, setLoaded] = React.useState(false);
const { logoImageUrl, applicationName, homeUrl } = useEnvironment().displayConfig;
@@ -80,12 +87,22 @@ export const ApplicationLogo: React.FC = (props: Applicati
]}
>
{logoUrl ? (
-
- {image}
-
+ isExternal ? (
+
+ {image}
+
+ ) : (
+
+ {image}
+
+ )
) : (
image
)}
diff --git a/packages/clerk-js/src/ui/primitives/Link.tsx b/packages/clerk-js/src/ui/primitives/Link.tsx
index b57e89caef2..c724e659779 100644
--- a/packages/clerk-js/src/ui/primitives/Link.tsx
+++ b/packages/clerk-js/src/ui/primitives/Link.tsx
@@ -1,5 +1,6 @@
import React from 'react';
+import { sanitizeHref } from '../../utils/url';
import type { PrimitiveProps, StyleVariants } from '../styledSystem';
import { common, createVariants } from '../styledSystem';
import { applyDataStateProps } from './applyDataStateProps';
@@ -57,9 +58,12 @@ export type LinkProps = PrimitiveProps<'a'> & OwnProps & StyleVariants {
const { isExternal, children, href, onClick, ...rest } = props;
+ // Sanitize href to prevent dangerous protocols
+ const sanitizedHref = sanitizeHref(href);
+
const onClickHandler = onClick
? (e: React.MouseEvent) => {
- if (!href) {
+ if (!sanitizedHref) {
e.preventDefault();
}
onClick(e);
@@ -70,9 +74,9 @@ export const Link = (props: LinkProps): JSX.Element => {
{children}
diff --git a/packages/clerk-js/src/utils/__tests__/url.spec.ts b/packages/clerk-js/src/utils/__tests__/url.spec.ts
index 9e37b01cc08..3c368500718 100644
--- a/packages/clerk-js/src/utils/__tests__/url.spec.ts
+++ b/packages/clerk-js/src/utils/__tests__/url.spec.ts
@@ -7,6 +7,7 @@ import {
createAllowedRedirectOrigins,
getETLDPlusOneFromFrontendApi,
getSearchParameterFromHash,
+ hasBannedHrefProtocol,
hasBannedProtocol,
hasExternalAccountSignUpError,
isAllowedRedirect,
@@ -18,6 +19,7 @@ import {
mergeFragmentIntoUrl,
relativeToAbsoluteUrl,
requiresUserInput,
+ sanitizeHref,
trimLeadingSlash,
trimTrailingSlash,
} from '../url';
@@ -36,7 +38,7 @@ describe('isDevAccountPortalOrigin(url)', () => {
];
test.each(goodUrls)('.isDevAccountPortalOrigin(%s)', (a, expected) => {
- // @ts-ignore
+ // @ts-ignore - Type assertion for test parameter
expect(isDevAccountPortalOrigin(a)).toBe(expected);
});
});
@@ -147,6 +149,79 @@ describe('hasBannedProtocol(url)', () => {
});
});
+describe('hasBannedHrefProtocol(url)', () => {
+ const cases: Array<[string, boolean]> = [
+ ['https://www.clerk.com/', false],
+ ['http://www.clerk.com/', false],
+ ['/sign-in', false],
+ ['/sign-in?test=1', false],
+ ['/?test', false],
+ ['javascript:console.log(document.cookies)', true],
+ ['', true],
+ ['vbscript:alert("xss")', true],
+ ['blob:https://example.com/12345678-1234-1234-1234-123456789012', true],
+ ['ftp://files.example.com/file.txt', false],
+ ['mailto:user@example.com', false],
+ ];
+
+ test.each(cases)('.hasBannedHrefProtocol(%s)', (a, expected) => {
+ expect(hasBannedHrefProtocol(a)).toBe(expected);
+ });
+});
+
+describe('sanitizeHref(href)', () => {
+ const cases: Array<[string | undefined | null, string | null]> = [
+ // Null/undefined/empty cases
+ [null, null],
+ [undefined, null],
+ ['', null],
+ [' ', null],
+
+ // Safe relative URLs
+ ['/path/to/page', '/path/to/page'],
+ ['#anchor', '#anchor'],
+ ['?query=param', '?query=param'],
+ ['../relative/path', '../relative/path'],
+ ['relative/path', 'relative/path'],
+ ['path/page#anchor', 'path/page#anchor'],
+
+ // Safe absolute URLs
+ ['https://www.clerk.com/', 'https://www.clerk.com/'],
+ ['http://localhost:3000/path', 'http://localhost:3000/path'],
+ ['ftp://files.example.com/file.txt', 'ftp://files.example.com/file.txt'],
+ ['mailto:user@example.com', 'mailto:user@example.com'],
+
+ // Dangerous protocols - should return null
+ ['javascript:alert("xss")', null],
+ ['javascript:console.log(document.cookies)', null],
+ ['data:text/html,', null],
+ ['data:text/html;base64,PHNjcmlwdD5hbGVydCgnWFNTIGZyb20gZGF0YSBVUkknKTwvc2NyaXB0Pg==', null],
+ ['', null],
+ ['vbscript:alert("xss")', null],
+ ['blob:https://example.com/12345678-1234-1234-1234-123456789012', null],
+
+ // Sneaky cases with dangerous protocols
+ ['JAVASCRIPT:alert("xss")', null], // All caps protocol
+ ['JavaScript:alert("xss")', null], // Mixed case
+ [' javascript:alert("xss") ', null], // Whitespace
+ ['javascript: alert("xss") ', null], // Whitespace
+
+ // Malformed URLs that might be relative paths
+ ['not-a-url', 'not-a-url'],
+ ['path:with:colons', 'path:with:colons'],
+ ];
+
+ test.each(cases)('.sanitizeHref(%s)', (href, expected) => {
+ expect(sanitizeHref(href)).toBe(expected);
+ });
+
+ it('handles malformed URLs gracefully', () => {
+ // These should not throw errors and should be allowed as potential relative URLs
+ expect(sanitizeHref(':::invalid:::')).toBe(':::invalid:::');
+ expect(sanitizeHref('malformed:url:here')).toBe('malformed:url:here');
+ });
+});
+
describe('buildURL(options: URLParams, skipOrigin)', () => {
it('builds a URL()', () => {
expect(buildURL({}, { stringify: true })).toBe('http://localhost:3000/');
diff --git a/packages/clerk-js/src/utils/url.ts b/packages/clerk-js/src/utils/url.ts
index d5b69503538..9f511e82e6b 100644
--- a/packages/clerk-js/src/utils/url.ts
+++ b/packages/clerk-js/src/utils/url.ts
@@ -21,6 +21,9 @@ const DUMMY_URL_BASE = 'http://clerk-dummy';
const BANNED_URI_PROTOCOLS = ['javascript:'] as const;
+// Protocols that are dangerous specifically for href attributes in links
+const BANNED_HREF_PROTOCOLS = ['javascript:', 'data:', 'vbscript:', 'blob:'] as const;
+
const { isDevOrStagingUrl } = createDevOrStagingUrlCache();
export { isDevOrStagingUrl };
const accountPortalCache = new Map();
@@ -276,6 +279,16 @@ export function isDataUri(val?: string): val is string {
return new URL(val).protocol === 'data:';
}
+/**
+ * Checks if a URL uses javascript: protocol.
+ * This prevents some XSS attacks through javascript: URLs.
+ *
+ * IMPORTANT: This does not check for `data:` or other protocols which
+ * are dangerous if used for links or setting the window location.
+ *
+ * @param val - The URL to check
+ * @returns True if the URL contains a banned protocol, false otherwise
+ */
export function hasBannedProtocol(val: string | URL) {
if (!isValidUrl(val)) {
return false;
@@ -284,6 +297,58 @@ export function hasBannedProtocol(val: string | URL) {
return BANNED_URI_PROTOCOLS.some(bp => bp === protocol);
}
+/**
+ * Checks if a URL contains a banned protocol for href attributes in links.
+ * This prevents some XSS attacks through javascript:, data:, vbscript:, and blob: URLs.
+ *
+ * @param val - The URL to check
+ * @returns True if the URL contains a banned protocol, false otherwise
+ */
+export function hasBannedHrefProtocol(val: string | URL): boolean {
+ if (!isValidUrl(val)) {
+ return false;
+ }
+ const protocol = new URL(val).protocol;
+ return BANNED_HREF_PROTOCOLS.some(bp => bp === protocol);
+}
+
+/**
+ * Sanitizes an href value by checking for dangerous protocols.
+ * Returns null if the href contains a dangerous protocol, otherwise returns the original href.
+ * This prevents some XSS attacks through javascript:, data:, vbscript:, and blob: URLs.
+ *
+ * @param href - The href value to sanitize
+ * @returns The sanitized href or null if dangerous
+ */
+export function sanitizeHref(href: string | undefined | null): string | null {
+ if (!href || href.trim() === '') {
+ return null;
+ }
+
+ // For relative URLs (starting with / or # or ?), allow them through
+ if (href.startsWith('/') || href.startsWith('#') || href.startsWith('?')) {
+ return href;
+ }
+
+ // For relative URLs without leading slash, allow them through
+ if (!href.includes(':')) {
+ return href;
+ }
+
+ // Check if it's a valid URL with a dangerous protocol
+ try {
+ const url = new URL(href);
+ if (hasBannedHrefProtocol(url)) {
+ return null;
+ }
+ return href;
+ } catch {
+ // If URL parsing fails, it's likely a relative URL or malformed
+ // Allow relative URLs through, but be cautious with malformed ones
+ return href;
+ }
+}
+
export const hasUrlInFragment = (_url: URL | string) => {
return new URL(_url, DUMMY_URL_BASE).hash.startsWith('#/');
};
diff --git a/packages/types/src/clerk.ts b/packages/types/src/clerk.ts
index bb5ab633869..c60738237a9 100644
--- a/packages/types/src/clerk.ts
+++ b/packages/types/src/clerk.ts
@@ -2002,6 +2002,10 @@ export type __internal_OAuthConsentProps = {
* Logo URL of the OAuth application.
*/
oAuthApplicationLogoUrl?: string;
+ /**
+ * URL of the OAuth application.
+ */
+ oAuthApplicationUrl?: string;
/**
* Scopes requested by the OAuth application.
*/