UNPKG

payload

Version:

Node, React and MongoDB Headless CMS and Application Framework

359 lines (358 loc) • 34.6 kB
"use strict"; 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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"}