관리자 기능과 태그 표시 설정 추가
This commit is contained in:
43
server/routes/admin/api/auth/login.post.js
Normal file
43
server/routes/admin/api/auth/login.post.js
Normal file
@@ -0,0 +1,43 @@
|
||||
import { z } from 'zod'
|
||||
import { createError, readBody } from 'h3'
|
||||
import { safeCompare, setAdminSession } from '../../../../utils/admin-auth'
|
||||
|
||||
const loginSchema = z.object({
|
||||
email: z.string().email(),
|
||||
password: z.string().min(1)
|
||||
})
|
||||
|
||||
/**
|
||||
* 관리자 로그인 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<{ email: string }>} 관리자 세션 정보
|
||||
*/
|
||||
export default defineEventHandler(async (event) => {
|
||||
const parsedBody = loginSchema.safeParse(await readBody(event))
|
||||
const config = useRuntimeConfig()
|
||||
|
||||
if (!parsedBody.success) {
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
message: '로그인 요청 형식이 올바르지 않습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
const body = parsedBody.data
|
||||
|
||||
if (
|
||||
!safeCompare(body.email, config.adminEmail) ||
|
||||
!safeCompare(body.password, config.adminPassword)
|
||||
) {
|
||||
throw createError({
|
||||
statusCode: 401,
|
||||
message: '이메일 또는 비밀번호가 올바르지 않습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
setAdminSession(event, body.email)
|
||||
|
||||
return {
|
||||
email: body.email
|
||||
}
|
||||
})
|
||||
14
server/routes/admin/api/auth/logout.post.js
Normal file
14
server/routes/admin/api/auth/logout.post.js
Normal file
@@ -0,0 +1,14 @@
|
||||
import { clearAdminSession } from '../../../../utils/admin-auth'
|
||||
|
||||
/**
|
||||
* 관리자 로그아웃 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {{ ok: boolean }} 로그아웃 결과
|
||||
*/
|
||||
export default defineEventHandler((event) => {
|
||||
clearAdminSession(event)
|
||||
|
||||
return {
|
||||
ok: true
|
||||
}
|
||||
})
|
||||
8
server/routes/admin/api/auth/me.get.js
Normal file
8
server/routes/admin/api/auth/me.get.js
Normal file
@@ -0,0 +1,8 @@
|
||||
import { requireAdminSession } from '../../../../utils/admin-auth'
|
||||
|
||||
/**
|
||||
* 관리자 세션 조회 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {{ email: string }} 관리자 세션 정보
|
||||
*/
|
||||
export default defineEventHandler((event) => requireAdminSession(event))
|
||||
13
server/routes/admin/api/posts.get.js
Normal file
13
server/routes/admin/api/posts.get.js
Normal file
@@ -0,0 +1,13 @@
|
||||
import { requireAdminSession } from '../../../utils/admin-auth'
|
||||
import { listAdminPosts } from '../../../repositories/content-repository'
|
||||
|
||||
/**
|
||||
* 관리자 게시물 목록 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<Array>} 게시물 목록
|
||||
*/
|
||||
export default defineEventHandler((event) => {
|
||||
requireAdminSession(event)
|
||||
|
||||
return listAdminPosts()
|
||||
})
|
||||
35
server/routes/admin/api/posts.post.js
Normal file
35
server/routes/admin/api/posts.post.js
Normal file
@@ -0,0 +1,35 @@
|
||||
import { createError, readBody } from 'h3'
|
||||
import { requireAdminSession } from '../../../utils/admin-auth'
|
||||
import { parseAdminPostInput } from '../../../utils/admin-post-input'
|
||||
import { createAdminPost } from '../../../repositories/content-repository'
|
||||
|
||||
/**
|
||||
* 관리자 게시물 생성 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<Object>} 생성된 게시물
|
||||
*/
|
||||
export default defineEventHandler(async (event) => {
|
||||
requireAdminSession(event)
|
||||
|
||||
const parsedBody = parseAdminPostInput(await readBody(event))
|
||||
|
||||
if (!parsedBody.success) {
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
message: '게시물 입력 형식이 올바르지 않습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
try {
|
||||
return await createAdminPost(parsedBody.data)
|
||||
} catch (error) {
|
||||
if (error?.code === '23505') {
|
||||
throw createError({
|
||||
statusCode: 409,
|
||||
message: '이미 사용 중인 슬러그입니다.'
|
||||
})
|
||||
}
|
||||
|
||||
throw error
|
||||
}
|
||||
})
|
||||
26
server/routes/admin/api/posts/[id].delete.js
Normal file
26
server/routes/admin/api/posts/[id].delete.js
Normal file
@@ -0,0 +1,26 @@
|
||||
import { createError, getRouterParam } from 'h3'
|
||||
import { requireAdminSession } from '../../../../utils/admin-auth'
|
||||
import { deleteAdminPost } from '../../../../repositories/content-repository'
|
||||
|
||||
/**
|
||||
* 관리자 게시물 삭제 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<{ id: string }>} 삭제된 게시물 ID
|
||||
*/
|
||||
export default defineEventHandler(async (event) => {
|
||||
requireAdminSession(event)
|
||||
|
||||
const id = getRouterParam(event, 'id')
|
||||
const deleted = await deleteAdminPost(id)
|
||||
|
||||
if (!deleted) {
|
||||
throw createError({
|
||||
statusCode: 404,
|
||||
message: '게시물을 찾을 수 없습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
return {
|
||||
id
|
||||
}
|
||||
})
|
||||
24
server/routes/admin/api/posts/[id].get.js
Normal file
24
server/routes/admin/api/posts/[id].get.js
Normal file
@@ -0,0 +1,24 @@
|
||||
import { createError, getRouterParam } from 'h3'
|
||||
import { requireAdminSession } from '../../../../utils/admin-auth'
|
||||
import { getAdminPostById } from '../../../../repositories/content-repository'
|
||||
|
||||
/**
|
||||
* 관리자 게시물 상세 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<Object>} 게시물 상세
|
||||
*/
|
||||
export default defineEventHandler(async (event) => {
|
||||
requireAdminSession(event)
|
||||
|
||||
const id = getRouterParam(event, 'id')
|
||||
const post = await getAdminPostById(id)
|
||||
|
||||
if (!post) {
|
||||
throw createError({
|
||||
statusCode: 404,
|
||||
message: '게시물을 찾을 수 없습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
return post
|
||||
})
|
||||
45
server/routes/admin/api/posts/[id].put.js
Normal file
45
server/routes/admin/api/posts/[id].put.js
Normal file
@@ -0,0 +1,45 @@
|
||||
import { createError, getRouterParam, readBody } from 'h3'
|
||||
import { requireAdminSession } from '../../../../utils/admin-auth'
|
||||
import { parseAdminPostInput } from '../../../../utils/admin-post-input'
|
||||
import { updateAdminPost } from '../../../../repositories/content-repository'
|
||||
|
||||
/**
|
||||
* 관리자 게시물 수정 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<Object>} 수정된 게시물
|
||||
*/
|
||||
export default defineEventHandler(async (event) => {
|
||||
requireAdminSession(event)
|
||||
|
||||
const id = getRouterParam(event, 'id')
|
||||
const parsedBody = parseAdminPostInput(await readBody(event))
|
||||
|
||||
if (!parsedBody.success) {
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
message: '게시물 입력 형식이 올바르지 않습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
try {
|
||||
const post = await updateAdminPost(id, parsedBody.data)
|
||||
|
||||
if (!post) {
|
||||
throw createError({
|
||||
statusCode: 404,
|
||||
message: '게시물을 찾을 수 없습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
return post
|
||||
} catch (error) {
|
||||
if (error?.code === '23505') {
|
||||
throw createError({
|
||||
statusCode: 409,
|
||||
message: '이미 사용 중인 슬러그입니다.'
|
||||
})
|
||||
}
|
||||
|
||||
throw error
|
||||
}
|
||||
})
|
||||
13
server/routes/admin/api/tags.get.js
Normal file
13
server/routes/admin/api/tags.get.js
Normal file
@@ -0,0 +1,13 @@
|
||||
import { requireAdminSession } from '../../../utils/admin-auth'
|
||||
import { listTags } from '../../../repositories/content-repository'
|
||||
|
||||
/**
|
||||
* 관리자 태그 목록 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<Array>} 태그 목록
|
||||
*/
|
||||
export default defineEventHandler((event) => {
|
||||
requireAdminSession(event)
|
||||
|
||||
return listTags()
|
||||
})
|
||||
35
server/routes/admin/api/tags.post.js
Normal file
35
server/routes/admin/api/tags.post.js
Normal file
@@ -0,0 +1,35 @@
|
||||
import { createError, readBody } from 'h3'
|
||||
import { requireAdminSession } from '../../../utils/admin-auth'
|
||||
import { parseAdminTagInput } from '../../../utils/admin-tag-input'
|
||||
import { createAdminTag } from '../../../repositories/content-repository'
|
||||
|
||||
/**
|
||||
* 관리자 태그 생성 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<Object>} 생성된 태그
|
||||
*/
|
||||
export default defineEventHandler(async (event) => {
|
||||
requireAdminSession(event)
|
||||
|
||||
const parsedBody = parseAdminTagInput(await readBody(event))
|
||||
|
||||
if (!parsedBody.success) {
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
message: '태그 입력 형식이 올바르지 않습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
try {
|
||||
return await createAdminTag(parsedBody.data)
|
||||
} catch (error) {
|
||||
if (error?.code === '23505') {
|
||||
throw createError({
|
||||
statusCode: 409,
|
||||
message: '이미 사용 중인 태그 슬러그입니다.'
|
||||
})
|
||||
}
|
||||
|
||||
throw error
|
||||
}
|
||||
})
|
||||
26
server/routes/admin/api/tags/[id].delete.js
Normal file
26
server/routes/admin/api/tags/[id].delete.js
Normal file
@@ -0,0 +1,26 @@
|
||||
import { createError, getRouterParam } from 'h3'
|
||||
import { requireAdminSession } from '../../../../utils/admin-auth'
|
||||
import { deleteAdminTag } from '../../../../repositories/content-repository'
|
||||
|
||||
/**
|
||||
* 관리자 태그 삭제 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<{ id: string }>} 삭제된 태그 ID
|
||||
*/
|
||||
export default defineEventHandler(async (event) => {
|
||||
requireAdminSession(event)
|
||||
|
||||
const id = getRouterParam(event, 'id')
|
||||
const deleted = await deleteAdminTag(id)
|
||||
|
||||
if (!deleted) {
|
||||
throw createError({
|
||||
statusCode: 404,
|
||||
message: '태그를 찾을 수 없습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
return {
|
||||
id
|
||||
}
|
||||
})
|
||||
24
server/routes/admin/api/tags/[id].get.js
Normal file
24
server/routes/admin/api/tags/[id].get.js
Normal file
@@ -0,0 +1,24 @@
|
||||
import { createError, getRouterParam } from 'h3'
|
||||
import { requireAdminSession } from '../../../../utils/admin-auth'
|
||||
import { getAdminTagById } from '../../../../repositories/content-repository'
|
||||
|
||||
/**
|
||||
* 관리자 태그 상세 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<Object>} 태그 상세
|
||||
*/
|
||||
export default defineEventHandler(async (event) => {
|
||||
requireAdminSession(event)
|
||||
|
||||
const id = getRouterParam(event, 'id')
|
||||
const tag = await getAdminTagById(id)
|
||||
|
||||
if (!tag) {
|
||||
throw createError({
|
||||
statusCode: 404,
|
||||
message: '태그를 찾을 수 없습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
return tag
|
||||
})
|
||||
45
server/routes/admin/api/tags/[id].put.js
Normal file
45
server/routes/admin/api/tags/[id].put.js
Normal file
@@ -0,0 +1,45 @@
|
||||
import { createError, getRouterParam, readBody } from 'h3'
|
||||
import { requireAdminSession } from '../../../../utils/admin-auth'
|
||||
import { parseAdminTagInput } from '../../../../utils/admin-tag-input'
|
||||
import { updateAdminTag } from '../../../../repositories/content-repository'
|
||||
|
||||
/**
|
||||
* 관리자 태그 수정 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<Object>} 수정된 태그
|
||||
*/
|
||||
export default defineEventHandler(async (event) => {
|
||||
requireAdminSession(event)
|
||||
|
||||
const id = getRouterParam(event, 'id')
|
||||
const parsedBody = parseAdminTagInput(await readBody(event))
|
||||
|
||||
if (!parsedBody.success) {
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
message: '태그 입력 형식이 올바르지 않습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
try {
|
||||
const tag = await updateAdminTag(id, parsedBody.data)
|
||||
|
||||
if (!tag) {
|
||||
throw createError({
|
||||
statusCode: 404,
|
||||
message: '태그를 찾을 수 없습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
return tag
|
||||
} catch (error) {
|
||||
if (error?.code === '23505') {
|
||||
throw createError({
|
||||
statusCode: 409,
|
||||
message: '이미 사용 중인 태그 슬러그입니다.'
|
||||
})
|
||||
}
|
||||
|
||||
throw error
|
||||
}
|
||||
})
|
||||
Reference in New Issue
Block a user