2026-01-05 22:16:02 +09:00
|
|
|
import express from "express";
|
|
|
|
|
import pool from "../lib/db.js";
|
2026-01-06 08:22:43 +09:00
|
|
|
import { searchSchedules } from "../services/meilisearch.js";
|
2026-01-05 22:16:02 +09:00
|
|
|
|
|
|
|
|
const router = express.Router();
|
|
|
|
|
|
2026-01-06 08:22:43 +09:00
|
|
|
// 공개 일정 목록 조회 (검색 포함)
|
2026-01-05 22:16:02 +09:00
|
|
|
router.get("/", async (req, res) => {
|
|
|
|
|
try {
|
2026-01-06 08:22:43 +09:00
|
|
|
const { search } = req.query;
|
|
|
|
|
|
|
|
|
|
// 검색어가 있으면 Meilisearch 사용
|
|
|
|
|
if (search && search.trim()) {
|
|
|
|
|
const results = await searchSchedules(search.trim());
|
|
|
|
|
return res.json(results);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 검색어 없으면 DB에서 전체 조회
|
2026-01-05 22:16:02 +09:00
|
|
|
const [schedules] = await pool.query(`
|
|
|
|
|
SELECT
|
|
|
|
|
s.id,
|
|
|
|
|
s.title,
|
|
|
|
|
s.description,
|
|
|
|
|
s.date,
|
|
|
|
|
s.time,
|
|
|
|
|
s.category_id,
|
|
|
|
|
s.source_url,
|
2026-01-06 00:27:35 +09:00
|
|
|
s.source_name,
|
2026-01-05 22:16:02 +09:00
|
|
|
s.location_name,
|
|
|
|
|
c.name as category_name,
|
2026-01-06 00:27:35 +09:00
|
|
|
c.color as category_color,
|
|
|
|
|
GROUP_CONCAT(m.name ORDER BY m.id SEPARATOR ',') as member_names
|
2026-01-05 22:16:02 +09:00
|
|
|
FROM schedules s
|
|
|
|
|
LEFT JOIN schedule_categories c ON s.category_id = c.id
|
2026-01-06 00:27:35 +09:00
|
|
|
LEFT JOIN schedule_members sm ON s.id = sm.schedule_id
|
|
|
|
|
LEFT JOIN members m ON sm.member_id = m.id
|
|
|
|
|
GROUP BY s.id
|
2026-01-05 22:16:02 +09:00
|
|
|
ORDER BY s.date DESC, s.time DESC
|
|
|
|
|
`);
|
|
|
|
|
|
|
|
|
|
res.json(schedules);
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error("일정 목록 조회 오류:", error);
|
|
|
|
|
res.status(500).json({ error: "일정 목록 조회 중 오류가 발생했습니다." });
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// 카테고리 목록 조회
|
|
|
|
|
router.get("/categories", async (req, res) => {
|
|
|
|
|
try {
|
|
|
|
|
const [categories] = await pool.query(`
|
|
|
|
|
SELECT id, name, color, sort_order
|
|
|
|
|
FROM schedule_categories
|
|
|
|
|
ORDER BY sort_order ASC
|
|
|
|
|
`);
|
|
|
|
|
|
|
|
|
|
res.json(categories);
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error("카테고리 조회 오류:", error);
|
|
|
|
|
res.status(500).json({ error: "카테고리 조회 중 오류가 발생했습니다." });
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// 개별 일정 조회
|
|
|
|
|
router.get("/:id", async (req, res) => {
|
|
|
|
|
try {
|
|
|
|
|
const { id } = req.params;
|
|
|
|
|
|
|
|
|
|
const [schedules] = await pool.query(
|
|
|
|
|
`
|
|
|
|
|
SELECT
|
|
|
|
|
s.*,
|
|
|
|
|
c.name as category_name,
|
|
|
|
|
c.color as category_color
|
|
|
|
|
FROM schedules s
|
|
|
|
|
LEFT JOIN schedule_categories c ON s.category_id = c.id
|
|
|
|
|
WHERE s.id = ?
|
|
|
|
|
`,
|
|
|
|
|
[id]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (schedules.length === 0) {
|
|
|
|
|
return res.status(404).json({ error: "일정을 찾을 수 없습니다." });
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
res.json(schedules[0]);
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error("일정 조회 오류:", error);
|
|
|
|
|
res.status(500).json({ error: "일정 조회 중 오류가 발생했습니다." });
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
2026-01-06 08:22:43 +09:00
|
|
|
// Meilisearch 동기화 API
|
|
|
|
|
router.post("/sync-search", async (req, res) => {
|
|
|
|
|
try {
|
|
|
|
|
const { syncAllSchedules } = await import("../services/meilisearch.js");
|
|
|
|
|
|
|
|
|
|
// DB에서 모든 일정 조회
|
|
|
|
|
const [schedules] = await pool.query(`
|
|
|
|
|
SELECT
|
|
|
|
|
s.id,
|
|
|
|
|
s.title,
|
|
|
|
|
s.description,
|
|
|
|
|
s.date,
|
|
|
|
|
s.time,
|
|
|
|
|
s.category_id,
|
|
|
|
|
s.source_url,
|
|
|
|
|
s.source_name,
|
|
|
|
|
c.name as category_name,
|
|
|
|
|
c.color as category_color,
|
|
|
|
|
GROUP_CONCAT(m.name ORDER BY m.id SEPARATOR ' ') as member_names
|
|
|
|
|
FROM schedules s
|
|
|
|
|
LEFT JOIN schedule_categories c ON s.category_id = c.id
|
|
|
|
|
LEFT JOIN schedule_members sm ON s.id = sm.schedule_id
|
|
|
|
|
LEFT JOIN members m ON sm.member_id = m.id
|
|
|
|
|
GROUP BY s.id
|
|
|
|
|
`);
|
|
|
|
|
|
|
|
|
|
const count = await syncAllSchedules(schedules);
|
|
|
|
|
res.json({ success: true, synced: count });
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error("Meilisearch 동기화 오류:", error);
|
|
|
|
|
res.status(500).json({ error: "동기화 중 오류가 발생했습니다." });
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
2026-01-05 22:16:02 +09:00
|
|
|
export default router;
|