Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[안혜정] week15 #540

Conversation

hyejungan
Copy link
Collaborator

요구사항

기본

  • [링크 공유 페이지] 링크 공유 페이지의 url path를 ‘/shared’에서 ‘/shared/{folderId}’로 변경했나요?
  • [링크 공유 페이지] 폴더의 정보는 ‘/api/folders/{folderId}’, 폴더 소유자의 정보는 ‘/api/users/{userId}’를 활용했나요?
  • [링크 공유 페이지] 링크 공유 페이지에서 폴더의 링크 데이터는 ‘/api/users/{userId}/links?folderId={folderId}’를 사용했나요?
  • [폴더 페이지] 폴더 페이지에서 유저가 access token이 없는 경우 ‘/signin’페이지로 이동하나요?
  • [폴더 페이지] 폴더 페이지의 url path가 ‘/folder’일 경우 폴더 목록에서 “전체” 가 선택되어 있고, ‘/folder/{folderId}’일 경우 폴더 목록에서 {folderId} 에 해당하는 폴더가 선택되어 있고 폴더에 있는 링크들을 볼 수 있나요?
  • [폴더 페이지] 폴더 페이지에서 현재 유저의 폴더 목록 데이터를 받아올 때 ‘/api/folders’를 활용했나요?
  • [폴더 페이지] 폴더 페이지에서 전체 링크 데이터를 받아올 때 ‘/api/links’, 특정 폴더의 링크를 받아올 때 ‘/api/links?folderId=1’를 활용했나요?
  • [상단 네비게이션] 유효한 access token이 있는 경우 ‘/api/users’로 현재 로그인한 유저 정보를 받아 상단 네비게이션 유저 프로필을 보여주나요?

심화

  • [심화] 리퀘스트 헤더에 인증 토큰을 첨부할 때 axios interceptors 또는 이와 유사한 기능을 활용 했나요?

주요 변경사항

  • 필요없는 fragment, console.log정리했습니다.
  • 로그인 인풋, 폼을 재사용할 수 있게 만들려고 했는데 아직 많이 복잡하게 구현되어 수정이 필요한 것 같습니다.

스크린샷

ezgif com-speed

멘토에게

  • 멘토링때 말씀해주신건 차근차근 고쳐보겠습니다. (에러핸들링, 토큰 proxy 전달)
  • 토큰을 axios 인터셉터로 저장해서 사용하고 싶은데 그렇게 하는것도 괜찮을까요? 토큰때문에 무조건 로그인부터 들어와야한다라는 조건이 붙어서 이 방향도 아닌것 같아 고민이 됩니다.
  • 많이 알려주셨는데 구현을 못해봐서 아쉬워요ㅠ

@hyejungan hyejungan requested a review from jayjnu December 16, 2023 19:21
@hyejungan hyejungan added 순한맛🐑  마음이 많이 여립니다.. 미완성 죄송합니다.. labels Dec 16, 2023
@hyejungan hyejungan self-assigned this Dec 16, 2023
@hyejungan hyejungan added 매운맛🔥 뒤는 없습니다. 그냥 필터 없이 말해주세요. 책임은 제가 집니다. and removed 순한맛🐑  마음이 많이 여립니다.. labels Dec 16, 2023
Copy link
Collaborator

@jayjnu jayjnu left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

고생하셨습니다.

멘토링때 말씀해주신건 차근차근 고쳐보겠습니다. (에러핸들링, 토큰 proxy 전달)

👍

토큰을 axios 인터셉터로 저장해서 사용하고 싶은데 그렇게 하는것도 괜찮을까요? 토큰때문에 무조건 로그인부터 들어와야한다라는 조건이 붙어서 이 방향도 아닌것 같아 고민이 됩니다.

심화과제 내용이니 해보는것도 좋을것 같네요. 모든 인증이 필요한 요청에 대해서는 어쨌든 실제 backend api로 리퀘스트 하기 전에 토큰 유효성 등을 체크하는 로직이 공통으로 들어가게 되어 있는데요, 이를 server side가 아니라 client side에서 수행할때에는 interceptor가 그나마 활용하기 좋은 레이어이긴 합니다. 다만 말씀드린대로 인증이 필요한 요청에 대해서만 제약을 걸고 싶다면 axios global보다는 instance단위로 interceptor를 구현하시는 것이 좋을것같네요.

많이 알려주셨는데 구현을 못해봐서 아쉬워요ㅠ

다 구현할 수 있으면 3년차 이상일겁니다. 고생하셨어요

<Script
src="https://t1.kakaocdn.net/kakao_js_sdk/2.5.0/kakao.min.js"
onLoad={initKakao}
/>
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

App에서 로드할 경우 불필요한 페이지에서도 해당 js를 로드해야 하는데요, 필요한 페이지에서만 로드하도록 하면 좋을것같아요


export default function App({ Component, pageProps }: AppProps) {
function initKakao() {
if (!window.Kakao.isInitialized()) {
window.Kakao.init(process.env.NEXT_PUBLIC_KAKAO_KEY);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

토큰을 환경변수로 관리하셨네요. 잘하셨습니다. 👍

export const FooterContext = createContext({
isFooterVisible: false,
setIsFooterVisible: (visible: boolean) => {},
});
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Context 관련된 파일은 별도의 context 파일로 분리해주시는 것이 좋을것같습니다.

if (localStorage.getItem('access_token')) {
router.push('/folder');
useEffect (() => {
if(user) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

localStorage 관련 코드를 잘 추상화 하셨습니다.

router.push('/folder');
useEffect (() => {
if(user) {
router.push('/folder')
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

push 보다는 redirect를 하면 좋을것같아요

});
setCheckItem(checkItems);
} else if (value.length === 0) {
setCheckItem(links);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

component prop으로 setter를 받기보다는 어떤 이벤트인지를 받는것이 좋습니다.

또한 외부에서 받는 value에 대해서 데이터를 정제한 뒤 setter로 올리는 코드라서 Search의 관심사가 아닐 가능성이 높지 않을까 해요. 해당 value를 상태관리하는 코드의 관심사가 맞지 않을까요?

useEffect(() => {
Kakao.cleanup();
Kakao.init('');
}, []);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

SocialShare 코드와 실제 kakaosdk를 로드하는 코드가 멀리 떨어져있고, 다른 social share vendor 코드와 섞여있으므로 카카오는 카카오끼리 묶을 수 있겠죠?

import kakaoIcon from '@/src/assets/Kakao.svg';

function KakaoShareButton() {
  return (
     // SDK js 로드 및 init, cleanup 담당
    <KakaoSDKLoader>
      {(sdk) => <SocialShareButton src={kakaoIcon} onClick={() => {
         // sdk 호출
       }}/>}
    </KakaoSDKLoader>
  )
}

const response = await axios.get(`/folders${folderId ? folderId : ''}`);
const nextFolders = response.data.data.folder;
setValues((prev) => ({ ...prev, folders: nextFolders }));
}
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

getFolders, getLinks는 auth의 관심사가 아닌데 같이 있을 필요는 없습니다.

auth에서는 말그대로 인증에 대한 처리 (signup, signin) 만 있으면 좋을것같고,
각각의 데이터 소스에 대한 내용은 각각의 데이터 소스에서 관리하면 됩니다.

const { email, password } = data;
const response = await axios.post('/sign-in', { email, password });
const { accessToken } = response.data.data;
axios.defaults.headers.common['Authorization'] = `Bearer ${accessToken}`;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

지금처럼 모든 api요청에 대해서는 글로벌로 처리하신 것이 효과적이므로 잘 하셨습니다.
다만 codeit이아닌 외부 api 요청에 대해서도 해당 accessToken이 유출될 여지가 있습니다.

그러한 상황에서 저같으면 아래처럼 제공할것같네요. best는 아닙니다. 애초에 accessToken을 js에서 만지면 안되니까요

function useFolders() {
  const {headers} = useAuth();

  return useFetch('/folders', {
    //인증 관련 header를 넘김
    headers: headers.extend({
      //...custom header
    })
  })
}

if (require && !context.user && !context.isPending) {
router.push('/signin');
}
}, [require, context.user, context.isPending, router]);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

만약 인증이 필요한 페이지에만 AuthProvider를 감쌀것이라면
AuthProvider 내부에서 위의 처리를 한 뒤에 children을 그리면 더 깔끔하지 않을까요?
hook은 무조건 한번은 화면이 렌더링 되어야 하는 문제가 있습니다.

function AuthProvider({children}) {
  return (
    <AuthContext.Provider>
       // 이 컴포넌트 내부에서 인증 여부에 따라 redirect 수행
      <RedirectOnNoAuth>{children}</RedirectOnNoAuth>
    </AuthContext.Provider>
  );
}

@jayjnu jayjnu merged commit 130d819 into codeit-bootcamp-frontend:part3-안혜정 Dec 19, 2023
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
매운맛🔥 뒤는 없습니다. 그냥 필터 없이 말해주세요. 책임은 제가 집니다. 미완성 죄송합니다..
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants