index.tsx 1.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778
  1. import { useEffect, useState } from "react";
  2. import { getClientConfig } from "@/app/config/client";
  3. import { useAccessStore } from "@/app/store";
  4. import { api } from "@/app/client/api";
  5. import dayjs from "dayjs";
  6. export type ITokenData = {
  7. access_token: string;
  8. token_type: string;
  9. expires_in: number;
  10. scope?: string[];
  11. };
  12. /**
  13. * 使用oauth2登录站点
  14. */
  15. export const EnsureToken = (props: any) => {
  16. const accessStore = useAccessStore();
  17. const [loading, setLoading] = useState(true);
  18. const config = getClientConfig();
  19. useEffect(() => {
  20. const parseToken = () => {
  21. const url = new URL(window.location.href);
  22. if (url.hash && url.hash.startsWith("#token=")) {
  23. try {
  24. setLoading(true);
  25. const token = JSON.parse(
  26. decodeURIComponent(url.hash.substring(7)),
  27. ) as ITokenData;
  28. accessStore.setAccessToken({
  29. accessToken: token.access_token,
  30. tokenType: token.token_type,
  31. expiresIn: dayjs().add(token.expires_in, "s").unix(),
  32. });
  33. setTimeout(() => {
  34. setLoading(false);
  35. url.hash = "";
  36. window.location.href = url.toString();
  37. }, 1000);
  38. return;
  39. } catch (e) {
  40. setLoading(false);
  41. console.log("parse token fail", e);
  42. }
  43. } else if (accessStore.hasAccessToken()) {
  44. api.user.userinfo().finally(() => {
  45. setLoading(false);
  46. });
  47. } else {
  48. accessStore.clearToken();
  49. setLoading(false);
  50. }
  51. };
  52. parseToken();
  53. const handler = setInterval(parseToken, 10000);
  54. return () => {
  55. clearInterval(handler);
  56. };
  57. }, []);
  58. if (loading) {
  59. return (
  60. <div>
  61. <span>Loading...</span>
  62. </div>
  63. );
  64. }
  65. if (!accessStore?.hasAccessToken()) {
  66. return (
  67. <div>
  68. <a href={config?.authorizeUrl ?? "/login"}>登录</a>
  69. </div>
  70. );
  71. }
  72. return <>{props.children}</>;
  73. };