From a44d888bf05d3607baec2a3add629ccb823082f7 Mon Sep 17 00:00:00 2001 From: Mohammed Taha <62352949+6km@users.noreply.github.com> Date: Wed, 29 May 2024 12:18:55 +0300 Subject: [PATCH] refactor(helper/adapter): improve runtime detection (#2846) * refactor: improve runtime detection detect runtime using navigator.userAgent when possible * fix typescript types * export knownUserAgents * add return type for checkUserAgentEquals --- src/helper/adapter/index.ts | 39 +++++++++++++++++++++++++++++-------- 1 file changed, 31 insertions(+), 8 deletions(-) diff --git a/src/helper/adapter/index.ts b/src/helper/adapter/index.ts index 4f2f01e68..01e73538d 100644 --- a/src/helper/adapter/index.ts +++ b/src/helper/adapter/index.ts @@ -35,28 +35,51 @@ export const env = , C extends Context = Conte return runtimeEnvHandlers[runtime]() } +export const knownUserAgents: Partial> = { + deno: 'Deno', + bun: 'Bun', + workerd: 'Cloudflare-Workers', + node: 'Node.js', +} + export const getRuntimeKey = (): Runtime => { // eslint-disable-next-line @typescript-eslint/no-explicit-any const global = globalThis as any - if (global?.Deno !== undefined) { - return 'deno' - } - if (global?.Bun !== undefined) { - return 'bun' - } - if (typeof global?.WebSocketPair === 'function') { - return 'workerd' + // check if the current runtime supports navigator.userAgent + const userAgentSupported = + typeof navigator !== 'undefined' && typeof navigator.userAgent === 'string' + + // if supported, check the user agent + if (userAgentSupported) { + for (const [runtimeKey, userAgent] of Object.entries(knownUserAgents)) { + if (checkUserAgentEquals(userAgent)) { + return runtimeKey as Runtime + } + } } + + // check if running on Edge Runtime if (typeof global?.EdgeRuntime === 'string') { return 'edge-light' } + + // check if running on Fastly if (global?.fastly !== undefined) { return 'fastly' } + + // userAgent isn't supported before Node v21.1.0; so fallback to the old way if (global?.process?.release?.name === 'node') { return 'node' } + // couldn't detect the runtime return 'other' } + +export const checkUserAgentEquals = (platform: string): boolean => { + const userAgent = navigator.userAgent + + return userAgent.startsWith(platform) +}