UNPKG

@alauda-fe/common

Version:

Alauda frontend team common codes.

189 lines 24.1 kB
import { getTopWindow, noop, NOT_NOTIFY_ON_ERROR_HEADERS, } from '../core/public-api'; import { getInitUrl, recordInitUrl } from './app-init-url'; import { CALLBACK_API, CODE_KEY, STATE_KEY, EXTERNAL_REDIRECT_API, ID_TOKEN_KEY, LOGIN_API, LOGOUT_API, REDIRECT_URIS, } from './constants'; import { cleanStorageToken, readStorageToken, refreshStorageAliveRecord, writeStorageToken, } from './storage-token'; import { fetchTokenApi, requestTokenInfo, requestTokenRefresh, } from './token-client'; let ACCOUNT_INFO; let checkTokenPromise; let refreshTokenPromise; export function getCachedAccountInfo() { return ACCOUNT_INFO; } export async function fetchAuthorizationState() { recordInitUrl(); await getExistingToken(); const info = await resolveTokenInfoWithRefresh(); if (!info) { await logout(true); return new Promise(() => undefined); } return { info }; } export function attachAuthorizationHeader() { const idToken = readStorageToken(); return idToken ? { Authorization: `Bearer ${idToken}` } : {}; } export async function checkAuthorizationToken() { if (checkTokenPromise) { return checkTokenPromise; } checkTokenPromise = requestTokenInfo() .then(info => { ACCOUNT_INFO = info; if (!info) { throw new Error('Authorization token is invalid'); } return info; }) .finally(() => { checkTokenPromise = null; }); return checkTokenPromise; } export async function refreshAuthorizationToken() { if (refreshTokenPromise) { return refreshTokenPromise; } refreshTokenPromise = requestTokenRefresh() .then(tokenResponse => { if (!tokenResponse) { throw new Error('Failed to refresh token'); } return tokenResponse; }) .finally(() => { refreshTokenPromise = null; }); return refreshTokenPromise; } export async function logoutAudit() { return fetchTokenApi(LOGOUT_API, { headers: NOT_NOTIFY_ON_ERROR_HEADERS, }) .then(res => res?.json()) .catch(() => null); } export async function logout(returnCurrentPage = false) { cleanStorageToken(); let logoutUrl; if (typeof returnCurrentPage === 'string' && returnCurrentPage !== '') { logoutUrl = returnCurrentPage; } else { try { const { authUrl } = await getAuthConfiguration(); logoutUrl = dexLogoutUrl(authUrl, !!returnCurrentPage); } catch { logout(returnCurrentPage); } } try { getTopWindow().location.href = logoutUrl; } catch { location.href = logoutUrl; } return new Promise(noop); } export function redirectSSOEntry(entry) { const hasQuery = entry.includes('?'); if (hasQuery) { const [url, query] = entry.split('?'); return `${EXTERNAL_REDIRECT_API}?redirect_url=${url}&${query}`; } return `${EXTERNAL_REDIRECT_API}?redirect_url=${entry}`; } async function resolveTokenInfoWithRefresh() { try { const info = await requestTokenInfo(); ACCOUNT_INFO = info; if (info) { return info; } const refreshed = await requestTokenRefresh(); if (!refreshed) { ACCOUNT_INFO = null; return null; } const newInfo = await requestTokenInfo(); ACCOUNT_INFO = newInfo; return newInfo; } catch { ACCOUNT_INFO = null; return null; } } function dexLogoutUrl(authUrl, returnCurrentPage = false) { return replaceRedirectUrl(authUrl, returnCurrentPage ? location.href : location.origin + '/console-portal'); } function replaceRedirectUrl(dexUrl, redirectUrl) { const [path, queryParams] = dexUrl.split('?'); const replacedQueryParams = queryParams .split('&') .map(pair => { const [key, value] = pair.split('='); return REDIRECT_URIS.has(key) ? `${key}=${encodeURIComponent(redirectUrl)}` : `${key}=${value}`; }) .join('&'); return `${path}?${replacedQueryParams}`; } async function getAuthConfiguration() { const { auth_url: authUrl } = await fetch(LOGIN_API).then(res => res.json()); return { authUrl }; } async function getExistingToken() { const { queryParams, hashParams } = getParams(); const code = queryParams[CODE_KEY] || hashParams[CODE_KEY]; const state = queryParams[STATE_KEY] || hashParams[STATE_KEY]; const idToken = queryParams[ID_TOKEN_KEY] || hashParams[ID_TOKEN_KEY]; if (!code && !idToken) { return; } refreshStorageAliveRecord(); if (code) { await setCookieByCode(code, state); } else { await setCookieByToken(idToken); } } async function setCookieByCode(code, state) { const queryParams = new URLSearchParams({ code, state }); const tokenResponse = await fetch(`${CALLBACK_API}?${queryParams.toString()}`).then(res => res.json()); writeStorageToken(tokenResponse?.id_token); } async function setCookieByToken(idToken) { const queryParams = new URLSearchParams({ id_token: idToken }); const tokenResponse = await fetch(`${CALLBACK_API}?${queryParams.toString()}`).then(res => res.json()); writeStorageToken(tokenResponse?.id_token); } function getParams() { const initUrl = getInitUrl(); const initLocation = new URL(initUrl); const queryParams = parseParams(initLocation.search ? initLocation.search.slice(1) : ''); const hashParams = parseParams(initLocation.hash ? initLocation.hash.slice(1) : ''); return { queryParams, hashParams }; } function parseParams(query) { if (!query) { return {}; } return query.split('&').reduce((acc, param) => { if (!param) { return acc; } const separatorIndex = param.indexOf('='); const key = separatorIndex === -1 ? param : param.slice(0, separatorIndex); const value = separatorIndex === -1 ? '' : param.slice(separatorIndex + 1); if (!key) { return acc; } acc[key] = value; return acc; }, {}); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"authorization.js","sourceRoot":"","sources":["../../../../../libs/common/src/authorization/authorization.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  getTopWindow,\n  noop,\n  NOT_NOTIFY_ON_ERROR_HEADERS,\n  StringMap,\n} from '../core/public-api';\n\nimport { getInitUrl, recordInitUrl } from './app-init-url';\nimport {\n  CALLBACK_API,\n  CODE_KEY,\n  STATE_KEY,\n  EXTERNAL_REDIRECT_API,\n  ID_TOKEN_KEY,\n  LOGIN_API,\n  LOGOUT_API,\n  REDIRECT_URIS,\n} from './constants';\nimport {\n  cleanStorageToken,\n  readStorageToken,\n  refreshStorageAliveRecord,\n  writeStorageToken,\n} from './storage-token';\nimport {\n  fetchTokenApi,\n  requestTokenInfo,\n  requestTokenRefresh,\n} from './token-client';\nimport type { AccountInfo, TokenResponse } from './token-client';\n\nlet ACCOUNT_INFO: AccountInfo;\nlet checkTokenPromise: Promise<AccountInfo>;\nlet refreshTokenPromise: Promise<TokenResponse>;\n\nexport function getCachedAccountInfo() {\n  return ACCOUNT_INFO;\n}\n\nexport async function fetchAuthorizationState() {\n  recordInitUrl();\n\n  await getExistingToken();\n\n  const info = await resolveTokenInfoWithRefresh();\n\n  if (!info) {\n    await logout(true);\n    return new Promise<never>(() => undefined);\n  }\n\n  return { info };\n}\n\nexport function attachAuthorizationHeader(): { Authorization?: string } {\n  const idToken = readStorageToken();\n  return idToken ? { Authorization: `Bearer ${idToken}` } : {};\n}\n\nexport async function checkAuthorizationToken(): Promise<AccountInfo> {\n  if (checkTokenPromise) {\n    return checkTokenPromise;\n  }\n\n  checkTokenPromise = requestTokenInfo()\n    .then(info => {\n      ACCOUNT_INFO = info;\n\n      if (!info) {\n        throw new Error('Authorization token is invalid');\n      }\n\n      return info;\n    })\n    .finally(() => {\n      checkTokenPromise = null;\n    });\n\n  return checkTokenPromise;\n}\n\nexport async function refreshAuthorizationToken(): Promise<TokenResponse> {\n  if (refreshTokenPromise) {\n    return refreshTokenPromise;\n  }\n\n  refreshTokenPromise = requestTokenRefresh()\n    .then(tokenResponse => {\n      if (!tokenResponse) {\n        throw new Error('Failed to refresh token');\n      }\n\n      return tokenResponse;\n    })\n    .finally(() => {\n      refreshTokenPromise = null;\n    });\n\n  return refreshTokenPromise;\n}\n\nexport async function logoutAudit(): Promise<{ logout_redirect_url?: string }> {\n  return fetchTokenApi(LOGOUT_API, {\n    headers: NOT_NOTIFY_ON_ERROR_HEADERS,\n  })\n    .then(res => res?.json())\n    .catch(() => null);\n}\n\nexport async function logout(\n  returnCurrentPage: boolean | string = false,\n): Promise<never> {\n  cleanStorageToken();\n\n  let logoutUrl: string;\n\n  if (typeof returnCurrentPage === 'string' && returnCurrentPage !== '') {\n    logoutUrl = returnCurrentPage;\n  } else {\n    try {\n      const { authUrl } = await getAuthConfiguration();\n      logoutUrl = dexLogoutUrl(authUrl, !!returnCurrentPage);\n    } catch {\n      logout(returnCurrentPage);\n    }\n  }\n\n  try {\n    getTopWindow().location.href = logoutUrl;\n  } catch {\n    location.href = logoutUrl;\n  }\n\n  return new Promise<never>(noop);\n}\n\nexport function redirectSSOEntry(entry: string) {\n  const hasQuery = entry.includes('?');\n\n  if (hasQuery) {\n    const [url, query] = entry.split('?');\n    return `${EXTERNAL_REDIRECT_API}?redirect_url=${url}&${query}`;\n  }\n\n  return `${EXTERNAL_REDIRECT_API}?redirect_url=${entry}`;\n}\n\nasync function resolveTokenInfoWithRefresh() {\n  try {\n    const info = await requestTokenInfo();\n    ACCOUNT_INFO = info;\n\n    if (info) {\n      return info;\n    }\n\n    const refreshed = await requestTokenRefresh();\n\n    if (!refreshed) {\n      ACCOUNT_INFO = null;\n      return null;\n    }\n\n    const newInfo = await requestTokenInfo();\n    ACCOUNT_INFO = newInfo;\n    return newInfo;\n  } catch {\n    ACCOUNT_INFO = null;\n    return null;\n  }\n}\n\nfunction dexLogoutUrl(authUrl: string, returnCurrentPage = false) {\n  return replaceRedirectUrl(\n    authUrl,\n    returnCurrentPage ? location.href : location.origin + '/console-portal',\n  );\n}\n\nfunction replaceRedirectUrl(dexUrl: string, redirectUrl: string) {\n  const [path, queryParams] = dexUrl.split('?');\n\n  const replacedQueryParams = queryParams\n    .split('&')\n    .map(pair => {\n      const [key, value] = pair.split('=');\n      return REDIRECT_URIS.has(key)\n        ? `${key}=${encodeURIComponent(redirectUrl)}`\n        : `${key}=${value}`;\n    })\n    .join('&');\n\n  return `${path}?${replacedQueryParams}`;\n}\n\nasync function getAuthConfiguration(): Promise<{\n  authUrl: string;\n}> {\n  const { auth_url: authUrl } = await fetch(LOGIN_API).then(res => res.json());\n  return { authUrl };\n}\n\nasync function getExistingToken(): Promise<void> {\n  const { queryParams, hashParams } = getParams();\n\n  const code = queryParams[CODE_KEY] || hashParams[CODE_KEY];\n  const state = queryParams[STATE_KEY] || hashParams[STATE_KEY];\n  const idToken = queryParams[ID_TOKEN_KEY] || hashParams[ID_TOKEN_KEY];\n\n  if (!code && !idToken) {\n    return;\n  }\n\n  refreshStorageAliveRecord();\n\n  if (code) {\n    await setCookieByCode(code, state);\n  } else {\n    await setCookieByToken(idToken);\n  }\n}\n\nasync function setCookieByCode(code: string, state: string): Promise<void> {\n  const queryParams = new URLSearchParams({ code, state });\n\n  const tokenResponse: TokenResponse = await fetch(\n    `${CALLBACK_API}?${queryParams.toString()}`,\n  ).then(res => res.json());\n\n  writeStorageToken(tokenResponse?.id_token);\n}\n\nasync function setCookieByToken(idToken: string): Promise<void> {\n  const queryParams = new URLSearchParams({ id_token: idToken });\n\n  const tokenResponse: TokenResponse = await fetch(\n    `${CALLBACK_API}?${queryParams.toString()}`,\n  ).then(res => res.json());\n\n  writeStorageToken(tokenResponse?.id_token);\n}\n\nfunction getParams() {\n  const initUrl = getInitUrl();\n\n  const initLocation = new URL(initUrl);\n\n  const queryParams = parseParams(\n    initLocation.search ? initLocation.search.slice(1) : '',\n  );\n\n  const hashParams = parseParams(\n    initLocation.hash ? initLocation.hash.slice(1) : '',\n  );\n\n  return { queryParams, hashParams };\n}\n\nfunction parseParams(query: string): StringMap {\n  if (!query) {\n    return {};\n  }\n\n  return query.split('&').reduce((acc, param) => {\n    if (!param) {\n      return acc;\n    }\n\n    const separatorIndex = param.indexOf('=');\n    const key = separatorIndex === -1 ? param : param.slice(0, separatorIndex);\n    const value = separatorIndex === -1 ? '' : param.slice(separatorIndex + 1);\n\n    if (!key) {\n      return acc;\n    }\n\n    acc[key] = value;\n\n    return acc;\n  }, {} as StringMap);\n}\n"]}