@alauda-fe/common
Version:
Alauda frontend team common codes.
189 lines • 24.1 kB
JavaScript
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"]}