feat(tkfb): 공정표 + 생산회의록 시스템 추가
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
182
system1-factory/api/controllers/meetingController.js
Normal file
182
system1-factory/api/controllers/meetingController.js
Normal file
@@ -0,0 +1,182 @@
|
||||
const MeetingModel = require('../models/meetingModel');
|
||||
const logger = require('../utils/logger');
|
||||
|
||||
const MeetingController = {
|
||||
// 회의록 목록
|
||||
getAll: async (req, res) => {
|
||||
try {
|
||||
const { year, month, search } = req.query;
|
||||
const rows = await MeetingModel.getAll({ year, month, search });
|
||||
res.json({ success: true, data: rows });
|
||||
} catch (err) {
|
||||
logger.error('Meeting getAll error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// 회의록 상세
|
||||
getById: async (req, res) => {
|
||||
try {
|
||||
const meeting = await MeetingModel.getById(req.params.id);
|
||||
if (!meeting) return res.status(404).json({ success: false, message: '회의록을 찾을 수 없습니다.' });
|
||||
res.json({ success: true, data: meeting });
|
||||
} catch (err) {
|
||||
logger.error('Meeting getById error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// 회의록 생성
|
||||
create: async (req, res) => {
|
||||
try {
|
||||
const { meeting_date, title } = req.body;
|
||||
if (!meeting_date || !title) {
|
||||
return res.status(400).json({ success: false, message: '날짜와 제목은 필수입니다.' });
|
||||
}
|
||||
const id = await MeetingModel.create({
|
||||
...req.body,
|
||||
created_by: req.user.user_id || req.user.id
|
||||
});
|
||||
res.status(201).json({ success: true, data: { meeting_id: id }, message: '회의록이 생성되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Meeting create error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// 회의록 수정
|
||||
update: async (req, res) => {
|
||||
try {
|
||||
const meetingId = req.params.id;
|
||||
const status = await MeetingModel.getStatus(meetingId);
|
||||
if (!status) return res.status(404).json({ success: false, message: '회의록을 찾을 수 없습니다.' });
|
||||
|
||||
// published 상태면 admin만 수정 가능
|
||||
const userLevel = req.user.access_level || req.user.role;
|
||||
if (status === 'published' && !['admin', 'system'].includes(userLevel)) {
|
||||
return res.status(403).json({ success: false, message: '발행된 회의록은 관리자만 수정할 수 있습니다.' });
|
||||
}
|
||||
|
||||
await MeetingModel.update(meetingId, req.body);
|
||||
res.json({ success: true, message: '회의록이 수정되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Meeting update error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// 회의록 발행
|
||||
publish: async (req, res) => {
|
||||
try {
|
||||
await MeetingModel.publish(req.params.id);
|
||||
res.json({ success: true, message: '회의록이 발행되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Meeting publish error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// 발행 취소 (admin only)
|
||||
unpublish: async (req, res) => {
|
||||
try {
|
||||
await MeetingModel.unpublish(req.params.id);
|
||||
res.json({ success: true, message: '발행이 취소되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Meeting unpublish error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// 회의록 삭제
|
||||
delete: async (req, res) => {
|
||||
try {
|
||||
await MeetingModel.delete(req.params.id);
|
||||
res.json({ success: true, message: '회의록이 삭제되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Meeting delete error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// === 안건 ===
|
||||
addItem: async (req, res) => {
|
||||
try {
|
||||
const { content } = req.body;
|
||||
if (!content) return res.status(400).json({ success: false, message: '안건 내용을 입력해주세요.' });
|
||||
|
||||
// published 체크
|
||||
const status = await MeetingModel.getStatus(req.params.id);
|
||||
const userLevel = req.user.access_level || req.user.role;
|
||||
if (status === 'published' && !['admin', 'system'].includes(userLevel)) {
|
||||
return res.status(403).json({ success: false, message: '발행된 회의록은 관리자만 수정할 수 있습니다.' });
|
||||
}
|
||||
|
||||
const id = await MeetingModel.addItem(req.params.id, req.body);
|
||||
res.status(201).json({ success: true, data: { item_id: id }, message: '안건이 추가되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Meeting addItem error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
updateItem: async (req, res) => {
|
||||
try {
|
||||
// published 체크
|
||||
const status = await MeetingModel.getStatus(req.params.id);
|
||||
const userLevel = req.user.access_level || req.user.role;
|
||||
if (status === 'published' && !['admin', 'system'].includes(userLevel)) {
|
||||
return res.status(403).json({ success: false, message: '발행된 회의록은 관리자만 수정할 수 있습니다.' });
|
||||
}
|
||||
|
||||
await MeetingModel.updateItem(req.params.itemId, req.body);
|
||||
res.json({ success: true, message: '안건이 수정되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Meeting updateItem error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
deleteItem: async (req, res) => {
|
||||
try {
|
||||
// published 체크
|
||||
const status = await MeetingModel.getStatus(req.params.id);
|
||||
const userLevel = req.user.access_level || req.user.role;
|
||||
if (status === 'published' && !['admin', 'system'].includes(userLevel)) {
|
||||
return res.status(403).json({ success: false, message: '발행된 회의록은 관리자만 수정할 수 있습니다.' });
|
||||
}
|
||||
|
||||
await MeetingModel.deleteItem(req.params.itemId);
|
||||
res.json({ success: true, message: '안건이 삭제되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Meeting deleteItem error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// 조치상태 업데이트 (group_leader+)
|
||||
updateItemStatus: async (req, res) => {
|
||||
try {
|
||||
const { status } = req.body;
|
||||
if (!status) return res.status(400).json({ success: false, message: '상태를 선택해주세요.' });
|
||||
await MeetingModel.updateItemStatus(req.params.itemId, status);
|
||||
res.json({ success: true, message: '조치상태가 업데이트되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Meeting updateItemStatus error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// 미완료 조치사항
|
||||
getActionItems: async (req, res) => {
|
||||
try {
|
||||
const { status, responsible_user_id } = req.query;
|
||||
const rows = await MeetingModel.getActionItems({ status, responsible_user_id });
|
||||
res.json({ success: true, data: rows });
|
||||
} catch (err) {
|
||||
logger.error('Meeting getActionItems error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
module.exports = MeetingController;
|
||||
228
system1-factory/api/controllers/scheduleController.js
Normal file
228
system1-factory/api/controllers/scheduleController.js
Normal file
@@ -0,0 +1,228 @@
|
||||
const ScheduleModel = require('../models/scheduleModel');
|
||||
const logger = require('../utils/logger');
|
||||
|
||||
const ScheduleController = {
|
||||
// === 공정 단계 ===
|
||||
getPhases: async (req, res) => {
|
||||
try {
|
||||
const rows = await ScheduleModel.getPhases();
|
||||
res.json({ success: true, data: rows });
|
||||
} catch (err) {
|
||||
logger.error('Schedule getPhases error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
createPhase: async (req, res) => {
|
||||
try {
|
||||
const { phase_name, display_order, color } = req.body;
|
||||
if (!phase_name) return res.status(400).json({ success: false, message: '단계명을 입력해주세요.' });
|
||||
const id = await ScheduleModel.createPhase({ phase_name, display_order, color });
|
||||
res.status(201).json({ success: true, data: { phase_id: id }, message: '공정 단계가 추가되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Schedule createPhase error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
updatePhase: async (req, res) => {
|
||||
try {
|
||||
await ScheduleModel.updatePhase(req.params.id, req.body);
|
||||
res.json({ success: true, message: '공정 단계가 수정되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Schedule updatePhase error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// === 작업 템플릿 ===
|
||||
getTemplates: async (req, res) => {
|
||||
try {
|
||||
const rows = await ScheduleModel.getTemplates(req.query.phase_id);
|
||||
res.json({ success: true, data: rows });
|
||||
} catch (err) {
|
||||
logger.error('Schedule getTemplates error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// === 공정표 항목 ===
|
||||
getEntries: async (req, res) => {
|
||||
try {
|
||||
const { project_id, year, month } = req.query;
|
||||
const rows = await ScheduleModel.getEntries({ project_id, year, month });
|
||||
res.json({ success: true, data: rows });
|
||||
} catch (err) {
|
||||
logger.error('Schedule getEntries error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
getGanttData: async (req, res) => {
|
||||
try {
|
||||
const year = req.query.year || new Date().getFullYear();
|
||||
const data = await ScheduleModel.getGanttData(year);
|
||||
res.json({ success: true, data });
|
||||
} catch (err) {
|
||||
logger.error('Schedule getGanttData error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
createEntry: async (req, res) => {
|
||||
try {
|
||||
const { project_id, phase_id, task_name, start_date, end_date } = req.body;
|
||||
if (!project_id || !phase_id || !task_name || !start_date || !end_date) {
|
||||
return res.status(400).json({ success: false, message: '필수 항목을 모두 입력해주세요.' });
|
||||
}
|
||||
const id = await ScheduleModel.createEntry({
|
||||
...req.body,
|
||||
created_by: req.user.user_id || req.user.id
|
||||
});
|
||||
res.status(201).json({ success: true, data: { entry_id: id }, message: '공정표 항목이 추가되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Schedule createEntry error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
createBatchEntries: async (req, res) => {
|
||||
try {
|
||||
const { project_id, phase_id, entries } = req.body;
|
||||
if (!project_id || !phase_id || !entries || entries.length === 0) {
|
||||
return res.status(400).json({ success: false, message: '프로젝트, 단계, 항목 정보를 입력해주세요.' });
|
||||
}
|
||||
const ids = await ScheduleModel.createBatchEntries(
|
||||
project_id, phase_id, entries, req.user.user_id || req.user.id
|
||||
);
|
||||
res.status(201).json({ success: true, data: { entry_ids: ids }, message: `${ids.length}개 항목이 일괄 추가되었습니다.` });
|
||||
} catch (err) {
|
||||
logger.error('Schedule createBatchEntries error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
updateEntry: async (req, res) => {
|
||||
try {
|
||||
await ScheduleModel.updateEntry(req.params.id, req.body);
|
||||
res.json({ success: true, message: '공정표 항목이 수정되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Schedule updateEntry error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
updateProgress: async (req, res) => {
|
||||
try {
|
||||
const { progress } = req.body;
|
||||
if (progress === undefined || progress < 0 || progress > 100) {
|
||||
return res.status(400).json({ success: false, message: '진행률은 0~100 사이의 값이어야 합니다.' });
|
||||
}
|
||||
await ScheduleModel.updateProgress(req.params.id, progress);
|
||||
res.json({ success: true, message: '진행률이 업데이트되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Schedule updateProgress error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
deleteEntry: async (req, res) => {
|
||||
try {
|
||||
await ScheduleModel.deleteEntry(req.params.id);
|
||||
res.json({ success: true, message: '공정표 항목이 삭제되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Schedule deleteEntry error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// === 의존관계 ===
|
||||
addDependency: async (req, res) => {
|
||||
try {
|
||||
const { depends_on_entry_id } = req.body;
|
||||
if (!depends_on_entry_id) {
|
||||
return res.status(400).json({ success: false, message: '선행 작업을 선택해주세요.' });
|
||||
}
|
||||
await ScheduleModel.addDependency(req.params.id, depends_on_entry_id);
|
||||
res.status(201).json({ success: true, message: '의존관계가 추가되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Schedule addDependency error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
removeDependency: async (req, res) => {
|
||||
try {
|
||||
await ScheduleModel.removeDependency(req.params.id, req.params.depId);
|
||||
res.json({ success: true, message: '의존관계가 삭제되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Schedule removeDependency error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// === 마일스톤 ===
|
||||
getMilestones: async (req, res) => {
|
||||
try {
|
||||
const rows = await ScheduleModel.getMilestones({ project_id: req.query.project_id });
|
||||
res.json({ success: true, data: rows });
|
||||
} catch (err) {
|
||||
logger.error('Schedule getMilestones error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
createMilestone: async (req, res) => {
|
||||
try {
|
||||
const { project_id, milestone_name, milestone_date } = req.body;
|
||||
if (!project_id || !milestone_name || !milestone_date) {
|
||||
return res.status(400).json({ success: false, message: '필수 항목을 모두 입력해주세요.' });
|
||||
}
|
||||
const id = await ScheduleModel.createMilestone({
|
||||
...req.body,
|
||||
created_by: req.user.user_id || req.user.id
|
||||
});
|
||||
res.status(201).json({ success: true, data: { milestone_id: id }, message: '마일스톤이 추가되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Schedule createMilestone error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
updateMilestone: async (req, res) => {
|
||||
try {
|
||||
await ScheduleModel.updateMilestone(req.params.id, req.body);
|
||||
res.json({ success: true, message: '마일스톤이 수정되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Schedule updateMilestone error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
deleteMilestone: async (req, res) => {
|
||||
try {
|
||||
await ScheduleModel.deleteMilestone(req.params.id);
|
||||
res.json({ success: true, message: '마일스톤이 삭제되었습니다.' });
|
||||
} catch (err) {
|
||||
logger.error('Schedule deleteMilestone error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
},
|
||||
|
||||
// === 부적합 연동 ===
|
||||
getNonconformance: async (req, res) => {
|
||||
try {
|
||||
const { project_id } = req.query;
|
||||
if (!project_id) {
|
||||
return res.status(400).json({ success: false, message: '프로젝트를 선택해주세요.' });
|
||||
}
|
||||
const rows = await ScheduleModel.getNonconformanceByProject(project_id);
|
||||
res.json({ success: true, data: rows });
|
||||
} catch (err) {
|
||||
logger.error('Schedule getNonconformance error:', err);
|
||||
res.status(500).json({ success: false, message: '서버 오류가 발생했습니다.' });
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
module.exports = ScheduleController;
|
||||
Reference in New Issue
Block a user