mirror of
https://gitee.com/nocobase/nocobase.git
synced 2025-05-06 05:59:25 +08:00
* feat(auth): add skipAuth option to API requests and update auth middleware * feat(auth): implement removeBasename function to streamline pathname handling
81 lines
2.4 KiB
TypeScript
81 lines
2.4 KiB
TypeScript
/**
|
|
* This file is part of the NocoBase (R) project.
|
|
* Copyright (c) 2020-2024 NocoBase Co., Ltd.
|
|
* Authors: NocoBase Team.
|
|
*
|
|
* This project is dual-licensed under AGPL-3.0 and NocoBase Commercial License.
|
|
* For more information, please refer to: https://www.nocobase.com/agreement.
|
|
*/
|
|
|
|
import React, { createContext, useContext, useMemo } from 'react';
|
|
import { Navigate } from 'react-router-dom';
|
|
import { useACLRoleContext } from '../acl';
|
|
import { ReturnTypeOfUseRequest, useAPIClient, useRequest } from '../api-client';
|
|
import { useAppSpin, useLocationNoUpdate } from '../application';
|
|
import { useCompile } from '../schema-component';
|
|
|
|
export const CurrentUserContext = createContext<ReturnTypeOfUseRequest>(null);
|
|
CurrentUserContext.displayName = 'CurrentUserContext';
|
|
|
|
export const useCurrentUserContext = () => {
|
|
return useContext(CurrentUserContext);
|
|
};
|
|
|
|
export const useCurrentRoles = () => {
|
|
const { allowAnonymous } = useACLRoleContext();
|
|
const { data } = useCurrentUserContext();
|
|
const compile = useCompile();
|
|
const options = useMemo(() => {
|
|
const roles = (data?.data?.roles || []).map(({ name, title }) => ({ name, title: compile(title) }));
|
|
if (allowAnonymous) {
|
|
roles.push({
|
|
title: 'Anonymous',
|
|
name: 'anonymous',
|
|
});
|
|
}
|
|
return roles;
|
|
}, [allowAnonymous, data?.data?.roles]);
|
|
return options;
|
|
};
|
|
|
|
export const CurrentUserProvider = (props) => {
|
|
const api = useAPIClient();
|
|
const result = useRequest<any>(() =>
|
|
api
|
|
.request({
|
|
url: '/auth:check',
|
|
skipNotify: (error) => {
|
|
const errs = api.toErrMessages(error);
|
|
if (errs.find((error: { code?: string }) => error.code === 'EMPTY_TOKEN')) {
|
|
return true;
|
|
}
|
|
return false;
|
|
},
|
|
skipAuth: true,
|
|
})
|
|
.then((res) => res?.data),
|
|
);
|
|
const { render } = useAppSpin();
|
|
|
|
if (result.loading) {
|
|
return render();
|
|
}
|
|
|
|
return <CurrentUserContext.Provider value={result}>{props.children}</CurrentUserContext.Provider>;
|
|
};
|
|
|
|
export const NavigateToSigninWithRedirect = () => {
|
|
const { pathname, search } = useLocationNoUpdate();
|
|
const redirect = `?redirect=${pathname}${search}`;
|
|
return <Navigate replace to={`/signin${redirect}`} />;
|
|
};
|
|
|
|
export const NavigateIfNotSignIn = ({ children }) => {
|
|
const result = useCurrentUserContext();
|
|
|
|
if (result.loading === false && !result.data?.data?.id) {
|
|
return <NavigateToSigninWithRedirect />;
|
|
}
|
|
return <>{children}</>;
|
|
};
|