payload
Version:
Node, React and MongoDB Headless CMS and Application Framework
359 lines (358 loc) • 34.6 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
function _export(target, all) {
for(var name in all)Object.defineProperty(target, name, {
enumerable: true,
get: all[name]
});
}
_export(exports, {
AuthProvider: function() {
return AuthProvider;
},
useAuth: function() {
return useAuth;
}
});
const _modal = require("@faceless-ui/modal");
const _qs = /*#__PURE__*/ _interop_require_default(require("qs"));
const _react = /*#__PURE__*/ _interop_require_wildcard(require("react"));
const _reacti18next = require("react-i18next");
const _reactrouterdom = require("react-router-dom");
const _reacttoastify = require("react-toastify");
const _api = require("../../../api");
const _useDebounce = /*#__PURE__*/ _interop_require_default(require("../../../hooks/useDebounce"));
const _Config = require("../Config");
const _Locale = require("../Locale");
function _interop_require_default(obj) {
return obj && obj.__esModule ? obj : {
default: obj
};
}
function _getRequireWildcardCache(nodeInterop) {
if (typeof WeakMap !== "function") return null;
var cacheBabelInterop = new WeakMap();
var cacheNodeInterop = new WeakMap();
return (_getRequireWildcardCache = function(nodeInterop) {
return nodeInterop ? cacheNodeInterop : cacheBabelInterop;
})(nodeInterop);
}
function _interop_require_wildcard(obj, nodeInterop) {
if (!nodeInterop && obj && obj.__esModule) {
return obj;
}
if (obj === null || typeof obj !== "object" && typeof obj !== "function") {
return {
default: obj
};
}
var cache = _getRequireWildcardCache(nodeInterop);
if (cache && cache.has(obj)) {
return cache.get(obj);
}
var newObj = {
__proto__: null
};
var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor;
for(var key in obj){
if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) {
var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null;
if (desc && (desc.get || desc.set)) {
Object.defineProperty(newObj, key, desc);
} else {
newObj[key] = obj[key];
}
}
}
newObj.default = obj;
if (cache) {
cache.set(obj, newObj);
}
return newObj;
}
const Context = /*#__PURE__*/ (0, _react.createContext)({});
const maxTimeoutTime = 2147483647;
const AuthProvider = ({ children })=>{
const [user, setUser] = (0, _react.useState)();
const [tokenInMemory, setTokenInMemory] = (0, _react.useState)();
const [tokenExpiration, setTokenExpiration] = (0, _react.useState)();
const { pathname } = (0, _reactrouterdom.useLocation)();
const { push } = (0, _reactrouterdom.useHistory)();
const { code } = (0, _Locale.useLocale)();
const config = (0, _Config.useConfig)();
const { admin: { autoLogin, inactivityRoute: logoutInactivityRoute, user: userSlug }, routes: { admin, api }, serverURL } = config;
const [permissions, setPermissions] = (0, _react.useState)();
const { i18n } = (0, _reacti18next.useTranslation)();
const { closeAllModals, openModal } = (0, _modal.useModal)();
const [lastLocationChange, setLastLocationChange] = (0, _react.useState)(0);
const debouncedLocationChange = (0, _useDebounce.default)(lastLocationChange, 10000);
const id = user?.id;
const redirectToInactivityRoute = (0, _react.useCallback)(()=>{
if (window.location.pathname.startsWith(admin)) {
const redirectParam = `?redirect=${encodeURIComponent(window.location.pathname.replace(admin, ''))}`;
push(`${admin}${logoutInactivityRoute}${redirectParam}`);
} else {
push(`${admin}${logoutInactivityRoute}`);
}
closeAllModals();
}, [
push,
admin,
logoutInactivityRoute,
closeAllModals
]);
const revokeTokenAndExpire = (0, _react.useCallback)(()=>{
setTokenInMemory(undefined);
setTokenExpiration(undefined);
}, []);
const setTokenAndExpiration = (0, _react.useCallback)((json)=>{
const token = json?.token || json?.refreshedToken;
if (token && json?.exp) {
setTokenInMemory(token);
setTokenExpiration(json.exp);
} else {
revokeTokenAndExpire();
}
}, [
revokeTokenAndExpire
]);
const refreshCookie = (0, _react.useCallback)((forceRefresh)=>{
const now = Math.round(new Date().getTime() / 1000);
const remainingTime = (typeof tokenExpiration === 'number' ? tokenExpiration : 0) - now;
if (forceRefresh || tokenExpiration && remainingTime < 120) {
setTimeout(async ()=>{
try {
const request = await _api.requests.post(`${serverURL}${api}/${userSlug}/refresh-token`, {
headers: {
'Accept-Language': i18n.language
}
});
if (request.status === 200) {
const json = await request.json();
setUser(json.user);
setTokenAndExpiration(json);
} else {
setUser(null);
redirectToInactivityRoute();
}
} catch (e) {
_reacttoastify.toast.error(e.message);
}
}, 1000);
}
}, [
tokenExpiration,
serverURL,
api,
userSlug,
i18n,
redirectToInactivityRoute,
setTokenAndExpiration
]);
const refreshCookieAsync = (0, _react.useCallback)(async (skipSetUser)=>{
try {
const request = await _api.requests.post(`${serverURL}${api}/${userSlug}/refresh-token`, {
headers: {
'Accept-Language': i18n.language
}
});
if (request.status === 200) {
const json = await request.json();
if (!skipSetUser) {
setUser(json.user);
setTokenAndExpiration(json);
}
return json.user;
}
setUser(null);
redirectToInactivityRoute();
return null;
} catch (e) {
_reacttoastify.toast.error(`Refreshing token failed: ${e.message}`);
return null;
}
}, [
serverURL,
api,
userSlug,
i18n,
redirectToInactivityRoute,
setTokenAndExpiration
]);
const logOut = (0, _react.useCallback)(()=>{
setUser(null);
revokeTokenAndExpire();
_api.requests.post(`${serverURL}${api}/${userSlug}/logout`);
}, [
serverURL,
api,
userSlug,
revokeTokenAndExpire
]);
const refreshPermissions = (0, _react.useCallback)(async ()=>{
const params = {
locale: code
};
try {
const request = await _api.requests.get(`${serverURL}${api}/access?${_qs.default.stringify(params)}`, {
headers: {
'Accept-Language': i18n.language
}
});
if (request.status === 200) {
const json = await request.json();
setPermissions(json);
} else {
throw new Error(`Fetching permissions failed with status code ${request.status}`);
}
} catch (e) {
_reacttoastify.toast.error(`Refreshing permissions failed: ${e.message}`);
}
}, [
serverURL,
api,
i18n,
code
]);
const fetchFullUser = _react.default.useCallback(async ()=>{
try {
const request = await _api.requests.get(`${serverURL}${api}/${userSlug}/me`, {
headers: {
'Accept-Language': i18n.language
}
});
if (request.status === 200) {
const json = await request.json();
if (json?.user) {
setUser(json.user);
if (json?.token) {
setTokenAndExpiration(json);
}
} else if (autoLogin && autoLogin.prefillOnly !== true) {
// auto log-in with the provided autoLogin credentials. This is used in dev mode
// so you don't have to log in over and over again
const autoLoginResult = await _api.requests.post(`${serverURL}${api}/${userSlug}/login`, {
body: JSON.stringify({
email: autoLogin.email,
password: autoLogin.password
}),
headers: {
'Accept-Language': i18n.language,
'Content-Type': 'application/json'
}
});
if (autoLoginResult.status === 200) {
const autoLoginJson = await autoLoginResult.json();
setUser(autoLoginJson.user);
if (autoLoginJson?.token) {
setTokenAndExpiration(autoLoginJson);
}
} else {
setUser(null);
revokeTokenAndExpire();
}
} else {
setUser(null);
revokeTokenAndExpire();
}
}
} catch (e) {
_reacttoastify.toast.error(`Fetching user failed: ${e.message}`);
}
}, [
serverURL,
api,
userSlug,
i18n,
autoLogin,
setTokenAndExpiration,
revokeTokenAndExpire
]);
// On mount, get user and set
(0, _react.useEffect)(()=>{
fetchFullUser();
}, [
fetchFullUser
]);
// When location changes, refresh cookie
(0, _react.useEffect)(()=>{
if (id) {
refreshCookie();
}
}, [
debouncedLocationChange,
refreshCookie,
id
]);
(0, _react.useEffect)(()=>{
setLastLocationChange(Date.now());
}, [
pathname
]);
// When user changes, get new access
(0, _react.useEffect)(()=>{
if (id) {
refreshPermissions();
}
}, [
i18n,
id,
api,
serverURL,
refreshPermissions
]);
(0, _react.useEffect)(()=>{
let reminder;
const now = Math.round(new Date().getTime() / 1000);
const remainingTime = typeof tokenExpiration === 'number' ? tokenExpiration - now : 0;
if (remainingTime > 0) {
reminder = setTimeout(()=>{
openModal('stay-logged-in');
}, Math.max(Math.min((remainingTime - 60) * 1000, maxTimeoutTime)));
}
return ()=>{
if (reminder) clearTimeout(reminder);
};
}, [
tokenExpiration,
openModal
]);
(0, _react.useEffect)(()=>{
let forceLogOut;
const now = Math.round(new Date().getTime() / 1000);
const remainingTime = typeof tokenExpiration === 'number' ? tokenExpiration - now : 0;
if (remainingTime > 0) {
forceLogOut = setTimeout(()=>{
setUser(null);
revokeTokenAndExpire();
redirectToInactivityRoute();
}, Math.max(Math.min(remainingTime * 1000, maxTimeoutTime), 0));
}
return ()=>{
if (forceLogOut) clearTimeout(forceLogOut);
};
}, [
tokenExpiration,
closeAllModals,
i18n,
redirectToInactivityRoute,
revokeTokenAndExpire
]);
return /*#__PURE__*/ _react.default.createElement(Context.Provider, {
value: {
fetchFullUser,
logOut,
permissions,
refreshCookie,
refreshCookieAsync,
refreshPermissions,
setUser,
token: tokenInMemory,
user
}
}, children);
};
const useAuth = ()=>(0, _react.useContext)(Context);
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../../src/admin/components/utilities/Auth/index.tsx"],"sourcesContent":["import { useModal } from '@faceless-ui/modal'\nimport qs from 'qs'\nimport React, { createContext, useCallback, useContext, useEffect, useState } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport { useHistory, useLocation } from 'react-router-dom'\nimport { toast } from 'react-toastify'\n\nimport type { Permissions, User } from '../../../../auth/types'\nimport type { AuthContext } from './types'\n\nimport { requests } from '../../../api'\nimport useDebounce from '../../../hooks/useDebounce'\nimport { useConfig } from '../Config'\nimport { useLocale } from '../Locale'\n\nconst Context = createContext({} as AuthContext)\n\nconst maxTimeoutTime = 2147483647\n\nexport const AuthProvider: React.FC<{ children: React.ReactNode }> = ({ children }) => {\n  const [user, setUser] = useState<User | null>()\n  const [tokenInMemory, setTokenInMemory] = useState<string>()\n  const [tokenExpiration, setTokenExpiration] = useState<number>()\n  const { pathname } = useLocation()\n  const { push } = useHistory()\n  const { code } = useLocale()\n\n  const config = useConfig()\n\n  const {\n    admin: { autoLogin, inactivityRoute: logoutInactivityRoute, user: userSlug },\n    routes: { admin, api },\n    serverURL,\n  } = config\n\n  const [permissions, setPermissions] = useState<Permissions>()\n\n  const { i18n } = useTranslation()\n  const { closeAllModals, openModal } = useModal()\n  const [lastLocationChange, setLastLocationChange] = useState(0)\n  const debouncedLocationChange = useDebounce(lastLocationChange, 10000)\n\n  const id = user?.id\n\n  const redirectToInactivityRoute = useCallback(() => {\n    if (window.location.pathname.startsWith(admin)) {\n      const redirectParam = `?redirect=${encodeURIComponent(\n        window.location.pathname.replace(admin, ''),\n      )}`\n      push(`${admin}${logoutInactivityRoute}${redirectParam}`)\n    } else {\n      push(`${admin}${logoutInactivityRoute}`)\n    }\n    closeAllModals()\n  }, [push, admin, logoutInactivityRoute, closeAllModals])\n\n  const revokeTokenAndExpire = useCallback(() => {\n    setTokenInMemory(undefined)\n    setTokenExpiration(undefined)\n  }, [])\n\n  const setTokenAndExpiration = useCallback(\n    (json) => {\n      const token = json?.token || json?.refreshedToken\n      if (token && json?.exp) {\n        setTokenInMemory(token)\n        setTokenExpiration(json.exp)\n      } else {\n        revokeTokenAndExpire()\n      }\n    },\n    [revokeTokenAndExpire],\n  )\n\n  const refreshCookie = useCallback(\n    (forceRefresh?: boolean) => {\n      const now = Math.round(new Date().getTime() / 1000)\n      const remainingTime = (typeof tokenExpiration === 'number' ? tokenExpiration : 0) - now\n\n      if (forceRefresh || (tokenExpiration && remainingTime < 120)) {\n        setTimeout(async () => {\n          try {\n            const request = await requests.post(`${serverURL}${api}/${userSlug}/refresh-token`, {\n              headers: {\n                'Accept-Language': i18n.language,\n              },\n            })\n\n            if (request.status === 200) {\n              const json = await request.json()\n              setUser(json.user)\n              setTokenAndExpiration(json)\n            } else {\n              setUser(null)\n              redirectToInactivityRoute()\n            }\n          } catch (e) {\n            toast.error(e.message)\n          }\n        }, 1000)\n      }\n    },\n    [\n      tokenExpiration,\n      serverURL,\n      api,\n      userSlug,\n      i18n,\n      redirectToInactivityRoute,\n      setTokenAndExpiration,\n    ],\n  )\n\n  const refreshCookieAsync = useCallback(\n    async (skipSetUser?: boolean): Promise<User> => {\n      try {\n        const request = await requests.post(`${serverURL}${api}/${userSlug}/refresh-token`, {\n          headers: {\n            'Accept-Language': i18n.language,\n          },\n        })\n\n        if (request.status === 200) {\n          const json = await request.json()\n          if (!skipSetUser) {\n            setUser(json.user)\n            setTokenAndExpiration(json)\n          }\n          return json.user\n        }\n\n        setUser(null)\n        redirectToInactivityRoute()\n        return null\n      } catch (e) {\n        toast.error(`Refreshing token failed: ${e.message}`)\n        return null\n      }\n    },\n    [serverURL, api, userSlug, i18n, redirectToInactivityRoute, setTokenAndExpiration],\n  )\n\n  const logOut = useCallback(() => {\n    setUser(null)\n    revokeTokenAndExpire()\n    requests.post(`${serverURL}${api}/${userSlug}/logout`)\n  }, [serverURL, api, userSlug, revokeTokenAndExpire])\n\n  const refreshPermissions = useCallback(async () => {\n    const params = {\n      locale: code,\n    }\n    try {\n      const request = await requests.get(`${serverURL}${api}/access?${qs.stringify(params)}`, {\n        headers: {\n          'Accept-Language': i18n.language,\n        },\n      })\n\n      if (request.status === 200) {\n        const json: Permissions = await request.json()\n        setPermissions(json)\n      } else {\n        throw new Error(`Fetching permissions failed with status code ${request.status}`)\n      }\n    } catch (e) {\n      toast.error(`Refreshing permissions failed: ${e.message}`)\n    }\n  }, [serverURL, api, i18n, code])\n\n  const fetchFullUser = React.useCallback(async () => {\n    try {\n      const request = await requests.get(`${serverURL}${api}/${userSlug}/me`, {\n        headers: {\n          'Accept-Language': i18n.language,\n        },\n      })\n\n      if (request.status === 200) {\n        const json = await request.json()\n\n        if (json?.user) {\n          setUser(json.user)\n          if (json?.token) {\n            setTokenAndExpiration(json)\n          }\n        } else if (autoLogin && autoLogin.prefillOnly !== true) {\n          // auto log-in with the provided autoLogin credentials. This is used in dev mode\n          // so you don't have to log in over and over again\n          const autoLoginResult = await requests.post(`${serverURL}${api}/${userSlug}/login`, {\n            body: JSON.stringify({\n              email: autoLogin.email,\n              password: autoLogin.password,\n            }),\n            headers: {\n              'Accept-Language': i18n.language,\n              'Content-Type': 'application/json',\n            },\n          })\n          if (autoLoginResult.status === 200) {\n            const autoLoginJson = await autoLoginResult.json()\n            setUser(autoLoginJson.user)\n            if (autoLoginJson?.token) {\n              setTokenAndExpiration(autoLoginJson)\n            }\n          } else {\n            setUser(null)\n            revokeTokenAndExpire()\n          }\n        } else {\n          setUser(null)\n          revokeTokenAndExpire()\n        }\n      }\n    } catch (e) {\n      toast.error(`Fetching user failed: ${e.message}`)\n    }\n  }, [serverURL, api, userSlug, i18n, autoLogin, setTokenAndExpiration, revokeTokenAndExpire])\n\n  // On mount, get user and set\n  useEffect(() => {\n    fetchFullUser()\n  }, [fetchFullUser])\n\n  // When location changes, refresh cookie\n  useEffect(() => {\n    if (id) {\n      refreshCookie()\n    }\n  }, [debouncedLocationChange, refreshCookie, id])\n\n  useEffect(() => {\n    setLastLocationChange(Date.now())\n  }, [pathname])\n\n  // When user changes, get new access\n  useEffect(() => {\n    if (id) {\n      refreshPermissions()\n    }\n  }, [i18n, id, api, serverURL, refreshPermissions])\n\n  useEffect(() => {\n    let reminder: ReturnType<typeof setTimeout>\n    const now = Math.round(new Date().getTime() / 1000)\n    const remainingTime = typeof tokenExpiration === 'number' ? tokenExpiration - now : 0\n\n    if (remainingTime > 0) {\n      reminder = setTimeout(\n        () => {\n          openModal('stay-logged-in')\n        },\n        Math.max(Math.min((remainingTime - 60) * 1000, maxTimeoutTime)),\n      )\n    }\n\n    return () => {\n      if (reminder) clearTimeout(reminder)\n    }\n  }, [tokenExpiration, openModal])\n\n  useEffect(() => {\n    let forceLogOut: ReturnType<typeof setTimeout>\n    const now = Math.round(new Date().getTime() / 1000)\n    const remainingTime = typeof tokenExpiration === 'number' ? tokenExpiration - now : 0\n\n    if (remainingTime > 0) {\n      forceLogOut = setTimeout(\n        () => {\n          setUser(null)\n          revokeTokenAndExpire()\n          redirectToInactivityRoute()\n        },\n        Math.max(Math.min(remainingTime * 1000, maxTimeoutTime), 0),\n      )\n    }\n\n    return () => {\n      if (forceLogOut) clearTimeout(forceLogOut)\n    }\n  }, [tokenExpiration, closeAllModals, i18n, redirectToInactivityRoute, revokeTokenAndExpire])\n\n  return (\n    <Context.Provider\n      value={{\n        fetchFullUser,\n        logOut,\n        permissions,\n        refreshCookie,\n        refreshCookieAsync,\n        refreshPermissions,\n        setUser,\n        token: tokenInMemory,\n        user,\n      }}\n    >\n      {children}\n    </Context.Provider>\n  )\n}\n\nexport const useAuth = <T = User,>(): AuthContext<T> => useContext(Context) as AuthContext<T>\n"],"names":["AuthProvider","useAuth","Context","createContext","maxTimeoutTime","children","user","setUser","useState","tokenInMemory","setTokenInMemory","tokenExpiration","setTokenExpiration","pathname","useLocation","push","useHistory","code","useLocale","config","useConfig","admin","autoLogin","inactivityRoute","logoutInactivityRoute","userSlug","routes","api","serverURL","permissions","setPermissions","i18n","useTranslation","closeAllModals","openModal","useModal","lastLocationChange","setLastLocationChange","debouncedLocationChange","useDebounce","id","redirectToInactivityRoute","useCallback","window","location","startsWith","redirectParam","encodeURIComponent","replace","revokeTokenAndExpire","undefined","setTokenAndExpiration","json","token","refreshedToken","exp","refreshCookie","forceRefresh","now","Math","round","Date","getTime","remainingTime","setTimeout","request","requests","post","headers","language","status","e","toast","error","message","refreshCookieAsync","skipSetUser","logOut","refreshPermissions","params","locale","get","qs","stringify","Error","fetchFullUser","React","prefillOnly","autoLoginResult","body","JSON","email","password","autoLoginJson","useEffect","reminder","max","min","clearTimeout","forceLogOut","Provider","value","useContext"],"mappings":"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"}