Skip to content

Commit

Permalink
Merge pull request #3 from Team-INSERT/feature/BMWK-0003
Browse files Browse the repository at this point in the history
Implement oauth page
  • Loading branch information
Ubinquitous authored Jan 3, 2025
2 parents 3d7ee32 + ebc0b33 commit 33c1a67
Show file tree
Hide file tree
Showing 15 changed files with 337 additions and 4 deletions.
7 changes: 5 additions & 2 deletions app/layout.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,10 @@ import type { Metadata } from 'next';
import './globals.css';
import classNames from 'classnames';
import { pretendard } from '@/styles/fonts';
import TanstackQueryProvider from '@/providers/TanstackQueryProvider';

export const metadata: Metadata = {
title: '역사의 고서',
title: '부마위키 | 역사의 고서',
description: '우리의 손으로 써내려 나가는 역사의 고서, 부마위키',
};

Expand All @@ -15,7 +16,9 @@ export default function RootLayout({
}>) {
return (
<html lang="en">
<body className={classNames(pretendard.variable, 'max-w-screen h-screen')}>{children}</body>
<body className={classNames(pretendard.variable, 'max-w-screen h-screen')}>
<TanstackQueryProvider>{children}</TanstackQueryProvider>
</body>
</html>
);
}
11 changes: 11 additions & 0 deletions app/oauth/_entities/api/axios.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
import { http, refreshTokenHeader } from '@/modules/services';

export const requestLogin = async (authCode: string) => {
const { data } = await http.post('/auth/oauth/bsm', {}, { headers: { authCode } });
return data;
};

export const requestLogout = async () => {
const { data } = await http.delete('/auth/bsm/logout', refreshTokenHeader());
return data;
};
26 changes: 26 additions & 0 deletions app/oauth/_entities/api/mutation.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
import { useMutation } from '@tanstack/react-query';
import { requestLogin, requestLogout } from './axios';
import { Storage } from '@/modules/services';
import { Bumawiki } from '@/modules/constants';

export const useLoginMutation = () => {
return useMutation({
mutationFn: requestLogin,
onSuccess: ({ accessToken, refreshToken }) => {
Storage.setItem(Bumawiki.token.access, accessToken);
Storage.setItem(Bumawiki.token.refresh, refreshToken);
window.history.go(-2);
},
});
};

export const useLogoutMutation = () => {
return useMutation({
mutationFn: requestLogout,
onSuccess: window.location.reload,
onSettled: () => {
Storage.delItem(Bumawiki.token.access);
Storage.delItem(Bumawiki.token.refresh);
},
});
};
1 change: 1 addition & 0 deletions app/oauth/_entities/model/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export {};
1 change: 1 addition & 0 deletions app/oauth/_features/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export { AuthenticationPage } from './ui/AuthenticationPage';
24 changes: 24 additions & 0 deletions app/oauth/_features/lib/withAuthentication.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
import { useLoginMutation } from '../../_entities/api/mutation';
import { useMount } from '@/hooks/useMount';
import { useSearchParams } from 'next/navigation';
import React, { useEffect } from 'react';

export const withAuthentication = <P extends object>(Component: React.ComponentType<P>) => {
const WrappedComponent: React.FC<P> = (props) => {
const { mutate: login } = useLoginMutation();
const isMounted = useMount();
const authCode = useSearchParams().get('code') || '';

useEffect(() => {
if (isMounted) login(authCode);
}, [isMounted, authCode, login]);

return <Component {...props} />;
};

WrappedComponent.displayName = `withAuthentication(${
Component.displayName || Component.name || 'Component'
})`;

return WrappedComponent;
};
13 changes: 13 additions & 0 deletions app/oauth/_features/ui/AuthenticationPage.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
'use client';

import { MoonLoader } from 'react-spinners';
import { withAuthentication } from '../lib/withAuthentication';

export const AuthenticationPage = withAuthentication(() => {
return (
<main className="flex flex-col items-center justify-center w-full h-screen gap-4">
<MoonLoader size={40} color="#274168" />
<span className="text-label-medium font-semibold">로그인 중...</span>
</main>
);
});
18 changes: 18 additions & 0 deletions app/oauth/page.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
import { Suspense } from 'react';
import { AuthenticationPage } from './_features';
import { Metadata, NextPage } from 'next';

export const metadata: Metadata = {
title: '부마위키 | 로그인',
description: '로그인 후 부마위키 문서에 직접 기여해보세요.',
};

const Page: NextPage = () => {
return (
<Suspense>
<AuthenticationPage />
</Suspense>
);
};

export default Page;
2 changes: 1 addition & 1 deletion next.config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ const nextConfig: NextConfig = {
return [
{
source: '/insert-proxy/:path*',
destination: 'https://buma.wiki/api/path*',
destination: 'https://buma.wiki/api/:path*',
basePath: false,
},
];
Expand Down
5 changes: 4 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -9,10 +9,13 @@
"lint": "next lint"
},
"dependencies": {
"@tanstack/react-query": "^5.62.12",
"axios": "^1.7.9",
"classnames": "^2.5.1",
"next": "15.1.3",
"react": "^19.0.0",
"react-dom": "^19.0.0"
"react-dom": "^19.0.0",
"react-spinners": "^0.15.0"
},
"devDependencies": {
"@eslint/eslintrc": "^3",
Expand Down
Loading

0 comments on commit 33c1a67

Please sign in to comment.