.
All checks were successful
Deploy blog-frontend / build-and-deploy (push) Successful in 1m47s

This commit is contained in:
ParkWonYeop
2025-12-27 21:47:34 +09:00
parent 3b1160295f
commit 1b7a26432d

View File

@@ -8,17 +8,16 @@ import { getCategories } from '@/api/category';
import { uploadImage } from '@/api/image';
import { useAuthStore } from '@/store/authStore';
import { Loader2, Save, Image as ImageIcon, ArrowLeft, Folder, UploadCloud } from 'lucide-react';
// 🎨 UX 개선: 토스트 알림 사용
import toast from 'react-hot-toast';
import dynamic from 'next/dynamic';
import axios from 'axios'; // 🆕 직접 갱신 요청을 위해 추가
import axios from 'axios';
const MDEditor = dynamic(
() => import('@uiw/react-md-editor').then((mod) => mod.default),
{ ssr: false }
);
// 🛠️ 1. 토큰 만료 체크 유틸리티 (JWT 디코딩)
// 🛠️ 1. 토큰 만료 체크 유틸리티
function isTokenExpired(token: string) {
try {
const base64Url = token.split('.')[1];
@@ -30,10 +29,9 @@ function isTokenExpired(token: string) {
.join('')
);
const { exp } = JSON.parse(jsonPayload);
// 현재 시간보다 만료 시간이 적게 남았거나 지났으면 true (여유시간 30초)
return Date.now() / 1000 >= exp - 30;
} catch (e) {
return true; // 파싱 실패 시 만료된 것으로 간주
return true;
}
}
@@ -41,16 +39,17 @@ function WritePageContent() {
const router = useRouter();
const searchParams = useSearchParams();
const queryClient = useQueryClient();
const { role, _hasHydrated, accessToken, refreshToken, login } = useAuthStore(); // 🆕 토큰 관련 상태 가져오기
const { role, _hasHydrated, accessToken, refreshToken, login } = useAuthStore();
const editSlug = searchParams.get('slug');
const isEditMode = !!editSlug;
const [title, setTitle] = useState('');
const [content, setContent] = useState('');
const [content, setContent] = useState('**Hello world!**');
const [categoryId, setCategoryId] = useState<number | ''>('');
const [tags, setTags] = useState('');
const [isUploading, setIsUploading] = useState(false);
const [isSubmitting, setIsSubmitting] = useState(false); // 🆕 제출 중 상태 관리
const [isSubmitting, setIsSubmitting] = useState(false);
useEffect(() => {
if (_hasHydrated && (!role || !role.includes('ADMIN'))) {
@@ -68,13 +67,17 @@ function WritePageContent() {
queryKey: ['post', editSlug],
queryFn: () => getPost(editSlug!),
enabled: isEditMode,
staleTime: 0,
gcTime: 0,
refetchOnMount: 'always'
});
useEffect(() => {
if (existingPost) {
// 🛠️ undefined 방지 처리 추가
setTitle(existingPost.title || '');
setContent(existingPost.content || '');
setTags(existingPost.tags ? existingPost.tags.join(', ') : '');
if (categories && existingPost.categoryName) {
const found = findCategoryByName(categories, existingPost.categoryName);
if (found) setCategoryId(found.id);
@@ -95,38 +98,50 @@ function WritePageContent() {
const mutation = useMutation({
mutationFn: (data: any) => isEditMode ? updatePost(existingPost!.id, data) : createPost(data),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['posts'] });
if (isEditMode) {
queryClient.invalidateQueries({ queryKey: ['post', editSlug] });
onSuccess: async (response: any) => {
const savedPost = response?.data;
const newSlug = savedPost?.slug || editSlug;
// 🚨 [핵심 수정] 목록 캐시를 '초기화(Reset)'합니다.
// 이렇게 하면 목록 페이지(Home, Category 등)로 이동했을 때
// 기존 캐시(옛날 글 목록)를 보여주지 않고, 즉시 서버에서 새 데이터를 가져옵니다.
await queryClient.resetQueries({ queryKey: ['posts'] });
// 카테고리 데이터(글 개수 등)도 갱신
await queryClient.invalidateQueries({ queryKey: ['categories'] });
// 상세 페이지 캐시 삭제 (다시 들어가면 새로 받도록)
if (editSlug) {
queryClient.removeQueries({ queryKey: ['post', editSlug] });
}
if (newSlug && newSlug !== editSlug) {
queryClient.removeQueries({ queryKey: ['post', newSlug] });
}
toast.success(isEditMode ? '게시글이 수정되었습니다.' : '게시글이 발행되었습니다!');
router.push(isEditMode ? `/posts/${editSlug}` : '/');
router.push(isEditMode ? `/posts/${newSlug}` : '/');
},
onError: (err: any) => {
toast.error('저장 실패: ' + (err.response?.data?.message || err.message));
},
onSettled: () => {
setIsSubmitting(false); // 🆕 완료 시 로딩 해제
setIsSubmitting(false);
}
});
// 🛡️ 2. [핵심 로직] 토큰 검사 및 갱신 보장 함수
const ensureAuthToken = async (): Promise<boolean> => {
// 1. 토큰이 아예 없으면 실패
if (!accessToken || !refreshToken) {
toast.error('로그인이 필요합니다.');
return false;
}
// 2. 토큰이 아직 싱싱하면 바로 통과
if (!isTokenExpired(accessToken)) {
return true;
}
// 3. 만료되었다면 갱신 시도
try {
// console.log('🔄 Access token expired during write. Refreshing...');
console.log('🔄 Access token expired during write. Refreshing...');
const BASE_URL = process.env.NEXT_PUBLIC_API_URL || 'http://localhost:8080';
const { data } = await axios.post(
@@ -137,13 +152,12 @@ function WritePageContent() {
if (data.code === 'SUCCESS' && data.data) {
login(data.data.accessToken, data.data.refreshToken);
// console.log('✅ Token refreshed successfully before save.');
console.log('✅ Token refreshed successfully before save.');
return true;
}
return false;
} catch (error) {
// console.error('❌ Failed to refresh token before save:', error);
// 갱신 실패: 사용자가 내용을 백업할 수 있도록 경고
console.error('❌ Failed to refresh token before save:', error);
toast.error('세션이 만료되었습니다.\n작성 중인 글을 복사해두고 다시 로그인해주세요!', { duration: 5000 });
return false;
}
@@ -159,20 +173,19 @@ function WritePageContent() {
return;
}
setIsSubmitting(true); // 버튼 비활성화
setIsSubmitting(true);
// 🛡️ 저장 전 토큰 체크!
const isTokenValid = await ensureAuthToken();
if (!isTokenValid) {
setIsSubmitting(false);
return; // 토큰 갱신 실패 시 중단
return;
}
mutation.mutate({
title,
content,
categoryId: Number(categoryId),
tags: [], // 🆕 타입 오류 방지용 빈 배열 (필요 시 태그 입력 기능 추가)
tags: tags.split(',').map(t => t.trim()).filter(t => t),
});
};
@@ -184,7 +197,6 @@ function WritePageContent() {
const uploadToast = toast.loading('이미지 업로드 중...');
try {
// 이미지 업로드 전에도 토큰 체크 (선택 사항이지만 안전함)
await ensureAuthToken();
const res = await uploadImage(file);
@@ -216,7 +228,7 @@ function WritePageContent() {
const uploadToast = toast.loading('이미지 업로드 중...');
try {
await ensureAuthToken(); // 토큰 체크
await ensureAuthToken();
const res = await uploadImage(file);
if (res.code === 'SUCCESS' && res.data) {
@@ -312,7 +324,21 @@ function WritePageContent() {
</div>
</div>
<div className="bg-white p-5 rounded-xl border border-gray-100 shadow-sm sticky top-[300px]">
<div className="bg-white p-5 rounded-xl border border-gray-100 shadow-sm sticky top-[280px]">
<h3 className="font-bold text-gray-700 mb-3 flex items-center gap-2">
🏷
</h3>
<input
type="text"
value={tags}
onChange={(e) => setTags(e.target.value)}
placeholder="태그 입력 (콤마 , 로 구분)"
className="w-full px-3 py-2 border border-gray-200 rounded-lg text-sm focus:ring-2 focus:ring-blue-100 focus:border-blue-500 outline-none"
/>
<p className="text-xs text-gray-400 mt-2">: React, Next.js, </p>
</div>
<div className="bg-white p-5 rounded-xl border border-gray-100 shadow-sm sticky top-[450px]">
<h3 className="font-bold text-gray-700 mb-3 flex items-center gap-2">
<ImageIcon size={18} />
</h3>