관리자 페이지 관리 추가
This commit is contained in:
@@ -338,6 +338,114 @@ export const listPages = async () => {
|
||||
return rows.map(mapPageRow)
|
||||
}
|
||||
|
||||
/**
|
||||
* 관리자 고정 페이지 목록 조회
|
||||
* @returns {Promise<Array>} 관리자 고정 페이지 목록
|
||||
*/
|
||||
export const listAdminPages = async () => listPages()
|
||||
|
||||
/**
|
||||
* 관리자 고정 페이지 상세 조회
|
||||
* @param {string} id - 페이지 ID
|
||||
* @returns {Promise<Object | null>} 관리자 고정 페이지 상세
|
||||
*/
|
||||
export const getAdminPageById = async (id) => {
|
||||
const sql = getPostgresClient()
|
||||
|
||||
if (!sql) {
|
||||
return getSamplePages().find((page) => page.id === id) || null
|
||||
}
|
||||
|
||||
const rows = await sql`
|
||||
SELECT *
|
||||
FROM pages
|
||||
WHERE id = ${id}
|
||||
LIMIT 1
|
||||
`
|
||||
|
||||
return rows[0] ? mapPageRow(rows[0]) : null
|
||||
}
|
||||
|
||||
/**
|
||||
* 관리자 고정 페이지 생성
|
||||
* @param {Object} input - 페이지 입력값
|
||||
* @returns {Promise<Object>} 생성된 페이지
|
||||
*/
|
||||
export const createAdminPage = async (input) => {
|
||||
const sql = getPostgresClient()
|
||||
|
||||
if (!sql) {
|
||||
throw new Error('DATABASE_REQUIRED')
|
||||
}
|
||||
|
||||
const rows = await sql`
|
||||
INSERT INTO pages (
|
||||
title,
|
||||
slug,
|
||||
content,
|
||||
featured_image
|
||||
)
|
||||
VALUES (
|
||||
${input.title},
|
||||
${input.slug},
|
||||
${input.content},
|
||||
${input.featuredImage}
|
||||
)
|
||||
RETURNING *
|
||||
`
|
||||
|
||||
return mapPageRow(rows[0])
|
||||
}
|
||||
|
||||
/**
|
||||
* 관리자 고정 페이지 수정
|
||||
* @param {string} id - 페이지 ID
|
||||
* @param {Object} input - 페이지 입력값
|
||||
* @returns {Promise<Object | null>} 수정된 페이지
|
||||
*/
|
||||
export const updateAdminPage = async (id, input) => {
|
||||
const sql = getPostgresClient()
|
||||
|
||||
if (!sql) {
|
||||
throw new Error('DATABASE_REQUIRED')
|
||||
}
|
||||
|
||||
const rows = await sql`
|
||||
UPDATE pages
|
||||
SET
|
||||
title = ${input.title},
|
||||
slug = ${input.slug},
|
||||
content = ${input.content},
|
||||
featured_image = ${input.featuredImage},
|
||||
updated_at = now()
|
||||
WHERE id = ${id}
|
||||
RETURNING *
|
||||
`
|
||||
|
||||
return rows[0] ? mapPageRow(rows[0]) : null
|
||||
}
|
||||
|
||||
/**
|
||||
* 관리자 고정 페이지 삭제
|
||||
* @param {string} id - 페이지 ID
|
||||
* @returns {Promise<boolean>} 삭제 여부
|
||||
*/
|
||||
export const deleteAdminPage = async (id) => {
|
||||
const sql = getPostgresClient()
|
||||
|
||||
if (!sql) {
|
||||
throw new Error('DATABASE_REQUIRED')
|
||||
}
|
||||
|
||||
const rows = await sql`
|
||||
DELETE FROM pages
|
||||
WHERE id = ${id}
|
||||
RETURNING id
|
||||
`
|
||||
|
||||
return Boolean(rows[0])
|
||||
}
|
||||
|
||||
/**
|
||||
* 공개 고정 페이지 상세 조회
|
||||
* @param {string} slug - 페이지 슬러그
|
||||
|
||||
13
server/routes/admin/api/pages.get.js
Normal file
13
server/routes/admin/api/pages.get.js
Normal file
@@ -0,0 +1,13 @@
|
||||
import { requireAdminSession } from '../../../utils/admin-auth'
|
||||
import { listAdminPages } from '../../../repositories/content-repository'
|
||||
|
||||
/**
|
||||
* 관리자 고정 페이지 목록 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<Array>} 고정 페이지 목록
|
||||
*/
|
||||
export default defineEventHandler((event) => {
|
||||
requireAdminSession(event)
|
||||
|
||||
return listAdminPages()
|
||||
})
|
||||
35
server/routes/admin/api/pages.post.js
Normal file
35
server/routes/admin/api/pages.post.js
Normal file
@@ -0,0 +1,35 @@
|
||||
import { createError, readBody } from 'h3'
|
||||
import { requireAdminSession } from '../../../utils/admin-auth'
|
||||
import { parseAdminPageInput } from '../../../utils/admin-page-input'
|
||||
import { createAdminPage } from '../../../repositories/content-repository'
|
||||
|
||||
/**
|
||||
* 관리자 고정 페이지 생성 API
|
||||
* @param {import('h3').H3Event} event - 요청 이벤트
|
||||
* @returns {Promise<Object>} 생성된 페이지
|
||||
*/
|
||||
export default defineEventHandler(async (event) => {
|
||||
requireAdminSession(event)
|
||||
|
||||
const parsedBody = parseAdminPageInput(await readBody(event))
|
||||
|
||||
if (!parsedBody.success) {
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
message: '페이지 입력 형식이 올바르지 않습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
try {
|
||||
return await createAdminPage(parsedBody.data)
|
||||
} catch (error) {
|
||||
if (error?.code === '23505') {
|
||||
throw createError({
|
||||
statusCode: 409,
|
||||
message: '이미 사용 중인 페이지 슬러그입니다.'
|
||||
})
|
||||
}
|
||||
|
||||
throw error
|
||||
}
|
||||
})
|
||||
26
server/routes/admin/api/pages/[id].delete.js
Normal file
26
server/routes/admin/api/pages/[id].delete.js
Normal file
@@ -0,0 +1,26 @@
|
||||
import { createError, getRouterParam } from 'h3'
|
||||
import { requireAdminSession } from '../../../../utils/admin-auth'
|
||||
import { deleteAdminPage } 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 deleteAdminPage(id)
|
||||
|
||||
if (!deleted) {
|
||||
throw createError({
|
||||
statusCode: 404,
|
||||
message: '페이지를 찾을 수 없습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
return {
|
||||
id
|
||||
}
|
||||
})
|
||||
24
server/routes/admin/api/pages/[id].get.js
Normal file
24
server/routes/admin/api/pages/[id].get.js
Normal file
@@ -0,0 +1,24 @@
|
||||
import { createError, getRouterParam } from 'h3'
|
||||
import { requireAdminSession } from '../../../../utils/admin-auth'
|
||||
import { getAdminPageById } 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 page = await getAdminPageById(id)
|
||||
|
||||
if (!page) {
|
||||
throw createError({
|
||||
statusCode: 404,
|
||||
message: '페이지를 찾을 수 없습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
return page
|
||||
})
|
||||
45
server/routes/admin/api/pages/[id].put.js
Normal file
45
server/routes/admin/api/pages/[id].put.js
Normal file
@@ -0,0 +1,45 @@
|
||||
import { createError, getRouterParam, readBody } from 'h3'
|
||||
import { requireAdminSession } from '../../../../utils/admin-auth'
|
||||
import { parseAdminPageInput } from '../../../../utils/admin-page-input'
|
||||
import { updateAdminPage } 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 = parseAdminPageInput(await readBody(event))
|
||||
|
||||
if (!parsedBody.success) {
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
message: '페이지 입력 형식이 올바르지 않습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
try {
|
||||
const page = await updateAdminPage(id, parsedBody.data)
|
||||
|
||||
if (!page) {
|
||||
throw createError({
|
||||
statusCode: 404,
|
||||
message: '페이지를 찾을 수 없습니다.'
|
||||
})
|
||||
}
|
||||
|
||||
return page
|
||||
} catch (error) {
|
||||
if (error?.code === '23505') {
|
||||
throw createError({
|
||||
statusCode: 409,
|
||||
message: '이미 사용 중인 페이지 슬러그입니다.'
|
||||
})
|
||||
}
|
||||
|
||||
throw error
|
||||
}
|
||||
})
|
||||
15
server/utils/admin-page-input.js
Normal file
15
server/utils/admin-page-input.js
Normal file
@@ -0,0 +1,15 @@
|
||||
import { z } from 'zod'
|
||||
|
||||
export const adminPageInputSchema = z.object({
|
||||
title: z.string().trim().min(1),
|
||||
slug: z.string().trim().min(1).regex(/^[a-z0-9가-힣]+(?:-[a-z0-9가-힣]+)*$/),
|
||||
content: z.string().default(''),
|
||||
featuredImage: z.string().trim().nullable().default(null)
|
||||
})
|
||||
|
||||
/**
|
||||
* 관리자 페이지 입력값 정리
|
||||
* @param {unknown} body - 요청 본문
|
||||
* @returns {import('zod').SafeParseReturnType<unknown, Object>} 검증 결과
|
||||
*/
|
||||
export const parseAdminPageInput = (body) => adminPageInputSchema.safeParse(body)
|
||||
Reference in New Issue
Block a user