- 순찰/점검 기능 개선 (zone-detail 페이지 추가) - 출근/근태 시스템 개선 (연차 조회, 근무현황) - 작업분석 대분류 그룹화 및 마이그레이션 스크립트 - 모바일 네비게이션 UI 추가 - NAS 배포 도구 및 문서 추가 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
531 lines
18 KiB
JavaScript
531 lines
18 KiB
JavaScript
const { getDb } = require('../dbPool');
|
|
|
|
class AttendanceModel {
|
|
// 일일 근태 기록 조회
|
|
static async getDailyAttendanceRecords(date, workerId = null) {
|
|
const db = await getDb();
|
|
let query = `
|
|
SELECT
|
|
dar.*,
|
|
w.worker_name,
|
|
w.job_type,
|
|
wat.type_name as attendance_type_name,
|
|
wat.type_code as attendance_type_code,
|
|
vt.type_name as vacation_type_name,
|
|
vt.type_code as vacation_type_code,
|
|
vt.deduct_days as vacation_days
|
|
FROM daily_attendance_records dar
|
|
LEFT JOIN workers w ON dar.worker_id = w.worker_id
|
|
LEFT JOIN work_attendance_types wat ON dar.attendance_type_id = wat.id
|
|
LEFT JOIN vacation_types vt ON dar.vacation_type_id = vt.id
|
|
WHERE dar.record_date = ?
|
|
`;
|
|
|
|
const params = [date];
|
|
|
|
if (workerId) {
|
|
query += ' AND dar.worker_id = ?';
|
|
params.push(workerId);
|
|
}
|
|
|
|
query += ' ORDER BY w.worker_name';
|
|
|
|
const [rows] = await db.execute(query, params);
|
|
return rows;
|
|
}
|
|
|
|
// 기간별 근태 기록 조회 (월별 조회용)
|
|
static async getDailyRecords(startDate, endDate, workerId = null) {
|
|
const db = await getDb();
|
|
let query = `
|
|
SELECT
|
|
dar.*,
|
|
w.worker_name,
|
|
w.job_type,
|
|
wat.type_name as attendance_type_name,
|
|
wat.type_code as attendance_type_code,
|
|
vt.type_name as vacation_type_name,
|
|
vt.type_code as vacation_type_code,
|
|
vt.deduct_days as vacation_days
|
|
FROM daily_attendance_records dar
|
|
LEFT JOIN workers w ON dar.worker_id = w.worker_id
|
|
LEFT JOIN work_attendance_types wat ON dar.attendance_type_id = wat.id
|
|
LEFT JOIN vacation_types vt ON dar.vacation_type_id = vt.id
|
|
WHERE dar.record_date BETWEEN ? AND ?
|
|
`;
|
|
|
|
const params = [startDate, endDate];
|
|
|
|
if (workerId) {
|
|
query += ' AND dar.worker_id = ?';
|
|
params.push(workerId);
|
|
}
|
|
|
|
query += ' ORDER BY dar.record_date ASC';
|
|
|
|
const [rows] = await db.execute(query, params);
|
|
return rows;
|
|
}
|
|
|
|
// 작업 보고서와 근태 기록 동기화 (시간 합산 및 상태 업데이트)
|
|
static async syncWithWorkReports(workerId, date) {
|
|
const db = await getDb();
|
|
|
|
// 1. 해당 날짜의 총 작업 시간 계산
|
|
const [reportStats] = await db.execute(`
|
|
SELECT
|
|
COALESCE(SUM(work_hours), 0) as total_hours,
|
|
COUNT(*) as report_count
|
|
FROM daily_work_reports
|
|
WHERE worker_id = ? AND report_date = ?
|
|
`, [workerId, date]);
|
|
|
|
const totalHours = parseFloat(reportStats[0].total_hours || 0);
|
|
const reportCount = reportStats[0].report_count;
|
|
|
|
// 2. 근태 유형 및 상태 결정
|
|
// 기본 규칙: 0시간 -> incomplete, <8시간 -> partial, 8시간 -> complete, >8시간 -> overtime
|
|
// (휴가는 별도 로직이지만 여기서 덮어쓰지 않도록 주의해야 함. 하지만 작업보고서가 추가되면 실 근무로 간주)
|
|
|
|
let status = 'incomplete';
|
|
let typeCode = 'REGULAR'; // 기본값
|
|
|
|
if (totalHours === 0) {
|
|
status = 'incomplete';
|
|
} else if (totalHours < 8) {
|
|
status = 'partial';
|
|
typeCode = 'PARTIAL';
|
|
} else if (totalHours === 8) {
|
|
status = 'complete';
|
|
typeCode = 'REGULAR';
|
|
} else {
|
|
status = 'overtime';
|
|
typeCode = 'OVERTIME';
|
|
}
|
|
|
|
// 근태 유형 ID 조회
|
|
const [types] = await db.execute('SELECT id FROM work_attendance_types WHERE type_code = ?', [typeCode]);
|
|
const typeId = types[0]?.id;
|
|
|
|
// 3. 기록 업데이트 (휴가 정보는 유지)
|
|
// 기존 기록 조회
|
|
const [existing] = await db.execute(
|
|
'SELECT id, vacation_type_id FROM daily_attendance_records WHERE worker_id = ? AND record_date = ?',
|
|
[workerId, date]
|
|
);
|
|
|
|
if (existing.length > 0) {
|
|
// 휴가가 설정되어 있고 시간이 0이면 휴가 상태 유지, 시간이 있으면 근무+휴가 복합 상태일 수 있음
|
|
// 여기서는 단순화하여 근무 시간이 있으면 근무 상태로 업데이트 (단, vacation_type_id는 유지)
|
|
|
|
const recordId = existing[0].id;
|
|
// 만약 기존 상태가 'vacation'이고 근무시간이 0이면 업데이트 건너뛸 수도 있지만,
|
|
// 작업보고서가 삭제되어 0이 된 경우도 있으므로 업데이트는 수행해야 함.
|
|
|
|
await db.execute(`
|
|
UPDATE daily_attendance_records
|
|
SET
|
|
total_work_hours = ?,
|
|
attendance_type_id = ?,
|
|
status = ?,
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE id = ?
|
|
`, [totalHours, typeId, status, recordId]);
|
|
|
|
return { synced: true, totalHours, status };
|
|
} else {
|
|
// 기록이 없으면 생성 (일반적으로는 initializeDailyRecords로 생성되어 있어야 함)
|
|
// 생성자가 명확하지 않으므로 시스템(1) 또는 알 수 없음 처리
|
|
await db.execute(`
|
|
INSERT INTO daily_attendance_records
|
|
(record_date, worker_id, total_work_hours, attendance_type_id, status, created_by)
|
|
VALUES (?, ?, ?, ?, ?, 1)
|
|
`, [date, workerId, totalHours, typeId, status]);
|
|
|
|
return { synced: true, totalHours, status, created: true };
|
|
}
|
|
}
|
|
|
|
// 일일 근태 기록 초기화 (모든 활성 작업자에 대한 기본 레코드 생성)
|
|
static async initializeDailyRecords(date, createdBy) {
|
|
const db = await getDb();
|
|
|
|
// 1. 활성 작업자 조회
|
|
const [workers] = await db.execute(
|
|
'SELECT worker_id FROM workers WHERE status = "active"' // is_active check not needed as status covers it based on previous fix? Wait, previous fix used status='active'.
|
|
);
|
|
|
|
if (workers.length === 0) return { inserted: 0 };
|
|
|
|
// 2. 일일 근태 레코드 일괄 생성 (이미 존재하면 무시)
|
|
// VALUES (...), (...), ...
|
|
const values = workers.map(w => [date, w.worker_id, 'incomplete', createdBy]);
|
|
|
|
// Bulk INSERT IGNORE
|
|
// Note: mysql2 execute doesn't support nested arrays for bulk insert easily with placeholder ?
|
|
// We should build the query or use query method for pool?
|
|
// Using simple loop for safety and compatibility or building string.
|
|
|
|
let insertedCount = 0;
|
|
|
|
// 트랜잭션 사용 권장
|
|
const conn = await db.getConnection();
|
|
try {
|
|
await conn.beginTransaction();
|
|
|
|
for (const w of workers) {
|
|
const [result] = await conn.execute(`
|
|
INSERT IGNORE INTO daily_attendance_records
|
|
(record_date, worker_id, status, created_by)
|
|
VALUES (?, ?, 'incomplete', ?)
|
|
`, [date, w.worker_id, createdBy]);
|
|
|
|
insertedCount += result.affectedRows;
|
|
}
|
|
|
|
await conn.commit();
|
|
} catch (err) {
|
|
await conn.rollback();
|
|
throw err;
|
|
} finally {
|
|
conn.release();
|
|
}
|
|
|
|
return { inserted: insertedCount, total_active_workers: workers.length };
|
|
}
|
|
|
|
// 근태 기록 생성 또는 업데이트
|
|
static async upsertAttendanceRecord(recordData) {
|
|
const db = await getDb();
|
|
|
|
const {
|
|
record_date,
|
|
worker_id,
|
|
total_work_hours = 8,
|
|
work_attendance_type_id = 1,
|
|
vacation_type_id = null,
|
|
is_overtime_approved = false,
|
|
created_by = 1
|
|
} = recordData;
|
|
|
|
const attendance_type_id = work_attendance_type_id;
|
|
|
|
// 기존 기록 확인
|
|
const [existing] = await db.execute(
|
|
'SELECT id FROM daily_attendance_records WHERE worker_id = ? AND record_date = ?',
|
|
[worker_id, record_date]
|
|
);
|
|
|
|
if (existing.length > 0) {
|
|
// 업데이트
|
|
const [result] = await db.execute(`
|
|
UPDATE daily_attendance_records
|
|
SET
|
|
total_work_hours = ?,
|
|
attendance_type_id = ?,
|
|
vacation_type_id = ?,
|
|
is_overtime_approved = ?,
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE id = ?
|
|
`, [
|
|
total_work_hours,
|
|
attendance_type_id,
|
|
vacation_type_id,
|
|
is_overtime_approved,
|
|
existing[0].id
|
|
]);
|
|
|
|
return { id: existing[0].id, affected: result.affectedRows };
|
|
} else {
|
|
// 생성
|
|
const [result] = await db.execute(`
|
|
INSERT INTO daily_attendance_records (
|
|
record_date, worker_id, total_work_hours, attendance_type_id,
|
|
vacation_type_id, is_overtime_approved, created_by
|
|
) VALUES (?, ?, ?, ?, ?, ?, ?)
|
|
`, [
|
|
record_date,
|
|
worker_id,
|
|
total_work_hours,
|
|
attendance_type_id,
|
|
vacation_type_id,
|
|
is_overtime_approved,
|
|
created_by
|
|
]);
|
|
|
|
return { id: result.insertId, affected: result.affectedRows };
|
|
}
|
|
}
|
|
|
|
// 작업자별 근태 현황 조회 (대시보드용)
|
|
static async getWorkerAttendanceStatus(date) {
|
|
const db = await getDb();
|
|
|
|
// 모든 작업자와 해당 날짜의 근태 기록을 조회
|
|
const [rows] = await db.execute(`
|
|
SELECT
|
|
w.worker_id,
|
|
w.worker_name,
|
|
w.job_type,
|
|
COALESCE(dar.total_work_hours, 0) as total_work_hours,
|
|
COALESCE(dar.status, 'incomplete') as status,
|
|
dar.is_vacation_processed,
|
|
dar.overtime_approved,
|
|
wat.type_name as attendance_type_name,
|
|
wat.type_code as attendance_type_code,
|
|
vt.type_name as vacation_type_name,
|
|
vt.type_code as vacation_type_code,
|
|
dar.notes,
|
|
-- 작업 건수 계산
|
|
(SELECT COUNT(*) FROM daily_work_reports dwr
|
|
WHERE dwr.worker_id = w.worker_id AND dwr.report_date = ?) as work_count,
|
|
-- 오류 건수 계산
|
|
(SELECT COUNT(*) FROM daily_work_reports dwr
|
|
WHERE dwr.worker_id = w.worker_id AND dwr.report_date = ? AND dwr.work_status_id = 2) as error_count
|
|
FROM workers w
|
|
LEFT JOIN daily_attendance_records dar ON w.worker_id = dar.worker_id AND dar.record_date = ?
|
|
LEFT JOIN work_attendance_types wat ON dar.attendance_type_id = wat.id
|
|
LEFT JOIN vacation_types vt ON dar.vacation_type_id = vt.id
|
|
WHERE w.is_active = TRUE
|
|
ORDER BY w.worker_name
|
|
`, [date, date, date]);
|
|
|
|
return rows;
|
|
}
|
|
|
|
// 휴가 처리
|
|
static async processVacation(workerId, date, vacationType, createdBy) {
|
|
const db = await getDb();
|
|
|
|
// 휴가 유형 정보 조회
|
|
const [vacationTypes] = await db.execute(
|
|
'SELECT id, type_code, type_name, deduct_days, is_active, created_at, updated_at FROM vacation_types WHERE type_code = ?',
|
|
[vacationType]
|
|
);
|
|
|
|
if (vacationTypes.length === 0) {
|
|
throw new Error('유효하지 않은 휴가 유형입니다.');
|
|
}
|
|
|
|
const vacationTypeInfo = vacationTypes[0];
|
|
|
|
// 현재 작업 시간 조회
|
|
const [workHours] = await db.execute(`
|
|
SELECT COALESCE(SUM(work_hours), 0) as total_hours
|
|
FROM daily_work_reports
|
|
WHERE worker_id = ? AND report_date = ?
|
|
`, [workerId, date]);
|
|
|
|
const currentHours = parseFloat(workHours[0].total_hours);
|
|
// deduct_days를 시간으로 변환 (1일 = 8시간)
|
|
const vacationHours = parseFloat(vacationTypeInfo.deduct_days) * 8;
|
|
const totalHours = currentHours + vacationHours;
|
|
|
|
// 근로 유형 결정
|
|
let attendanceTypeCode = 'VACATION';
|
|
let status = 'vacation';
|
|
|
|
if (totalHours >= 8) {
|
|
attendanceTypeCode = totalHours > 8 ? 'OVERTIME' : 'REGULAR';
|
|
status = totalHours > 8 ? 'overtime' : 'complete';
|
|
}
|
|
|
|
const [attendanceTypes] = await db.execute(
|
|
'SELECT id FROM work_attendance_types WHERE type_code = ?',
|
|
[attendanceTypeCode]
|
|
);
|
|
|
|
// 휴가 작업 기록 생성 (프로젝트 ID 13 = "연차/휴무", work_type_id 1 = 기본)
|
|
await db.execute(`
|
|
INSERT INTO daily_work_reports (
|
|
report_date, worker_id, project_id, work_type_id, work_status_id,
|
|
work_hours, description, created_by
|
|
) VALUES (?, ?, 13, 1, 1, ?, ?, ?)
|
|
`, [
|
|
date,
|
|
workerId,
|
|
vacationHours,
|
|
`${vacationTypeInfo.type_name} 처리`,
|
|
createdBy
|
|
]);
|
|
|
|
// 근태 기록 업데이트
|
|
const attendanceData = {
|
|
record_date: date,
|
|
worker_id: workerId,
|
|
total_work_hours: totalHours,
|
|
work_attendance_type_id: attendanceTypes[0]?.id,
|
|
vacation_type_id: vacationTypeInfo.id,
|
|
is_overtime_approved: false,
|
|
created_by: createdBy
|
|
};
|
|
|
|
return await this.upsertAttendanceRecord(attendanceData);
|
|
}
|
|
|
|
// 초과근무 승인
|
|
static async approveOvertime(workerId, date, approvedBy) {
|
|
const db = await getDb();
|
|
|
|
const [result] = await db.execute(`
|
|
UPDATE daily_attendance_records
|
|
SET
|
|
overtime_approved = TRUE,
|
|
overtime_approved_by = ?,
|
|
overtime_approved_at = CURRENT_TIMESTAMP,
|
|
updated_by = ?,
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE worker_id = ? AND record_date = ?
|
|
`, [approvedBy, approvedBy, workerId, date]);
|
|
|
|
return result.affectedRows > 0;
|
|
}
|
|
|
|
// 근로 유형 목록 조회
|
|
static async getAttendanceTypes() {
|
|
const db = await getDb();
|
|
const [rows] = await db.execute(
|
|
'SELECT id, type_code, type_name, description, is_active, created_at, updated_at FROM work_attendance_types WHERE is_active = TRUE ORDER BY id'
|
|
);
|
|
return rows;
|
|
}
|
|
|
|
// 휴가 유형 목록 조회
|
|
static async getVacationTypes() {
|
|
const db = await getDb();
|
|
const [rows] = await db.execute(
|
|
'SELECT id, type_code, type_name, deduct_days, is_active, created_at, updated_at FROM vacation_types WHERE is_active = TRUE ORDER BY deduct_days DESC'
|
|
);
|
|
return rows;
|
|
}
|
|
|
|
// 작업자 휴가 잔여 조회
|
|
static async getWorkerVacationBalance(workerId, year = null) {
|
|
const db = await getDb();
|
|
const currentYear = year || new Date().getFullYear();
|
|
|
|
const [rows] = await db.execute(`
|
|
SELECT id, worker_id, year, total_annual_leave, used_annual_leave, notes, created_at, updated_at FROM worker_vacation_balance
|
|
WHERE worker_id = ? AND year = ?
|
|
`, [workerId, currentYear]);
|
|
|
|
if (rows.length === 0) {
|
|
// 기본 연차 생성 (15일)
|
|
await db.execute(`
|
|
INSERT INTO worker_vacation_balance (worker_id, year, total_annual_leave)
|
|
VALUES (?, ?, 15.0)
|
|
`, [workerId, currentYear]);
|
|
|
|
return {
|
|
worker_id: workerId,
|
|
year: currentYear,
|
|
total_annual_leave: 15.0,
|
|
used_annual_leave: 0,
|
|
remaining_annual_leave: 15.0
|
|
};
|
|
}
|
|
|
|
return rows[0];
|
|
}
|
|
|
|
// 월별 근태 통계
|
|
static async getMonthlyAttendanceStats(year, month, workerId = null) {
|
|
const db = await getDb();
|
|
|
|
// work_attendance_types: 1=NORMAL, 2=LATE, 3=EARLY_LEAVE, 4=ABSENT, 5=VACATION
|
|
// vacation_types: 1=ANNUAL(연차), 2=HALF_ANNUAL(반차), 3=SICK(병가), 4=SPECIAL(경조사)
|
|
let query = `
|
|
SELECT
|
|
w.worker_id,
|
|
w.worker_name,
|
|
COUNT(CASE WHEN dar.attendance_type_id = 1 AND (dar.is_overtime_approved = 0 OR dar.is_overtime_approved IS NULL) THEN 1 END) as regular_days,
|
|
COUNT(CASE WHEN dar.is_overtime_approved = 1 OR dar.total_work_hours > 8 THEN 1 END) as overtime_days,
|
|
COUNT(CASE WHEN dar.attendance_type_id = 5 AND dar.vacation_type_id = 1 THEN 1 END) as vacation_days,
|
|
COUNT(CASE WHEN dar.vacation_type_id = 2 THEN 1 END) as partial_days,
|
|
COUNT(CASE WHEN dar.attendance_type_id = 4 THEN 1 END) as incomplete_days,
|
|
COALESCE(SUM(dar.total_work_hours), 0) as total_work_hours,
|
|
COALESCE(AVG(dar.total_work_hours), 0) as avg_work_hours
|
|
FROM workers w
|
|
LEFT JOIN daily_attendance_records dar ON w.worker_id = dar.worker_id
|
|
AND YEAR(dar.record_date) = ? AND MONTH(dar.record_date) = ?
|
|
WHERE w.employment_status = 'employed'
|
|
`;
|
|
|
|
const params = [year, month];
|
|
|
|
if (workerId) {
|
|
query += ' AND w.worker_id = ?';
|
|
params.push(workerId);
|
|
}
|
|
|
|
query += ' GROUP BY w.worker_id, w.worker_name ORDER BY w.worker_name';
|
|
|
|
const [rows] = await db.execute(query, params);
|
|
return rows;
|
|
}
|
|
|
|
// 출근 체크 기록 생성 또는 업데이트
|
|
static async upsertCheckin(checkinData) {
|
|
const db = await getDb();
|
|
const { worker_id, record_date, is_present } = checkinData;
|
|
|
|
// 해당 날짜에 기록이 있는지 확인
|
|
const [existing] = await db.execute(
|
|
'SELECT id FROM daily_attendance_records WHERE worker_id = ? AND record_date = ?',
|
|
[worker_id, record_date]
|
|
);
|
|
|
|
if (existing.length > 0) {
|
|
// 업데이트
|
|
await db.execute(
|
|
'UPDATE daily_attendance_records SET is_present = ? WHERE id = ?',
|
|
[is_present, existing[0].id]
|
|
);
|
|
return existing[0].id;
|
|
} else {
|
|
// 새로 생성 (기본값으로)
|
|
const [result] = await db.execute(
|
|
`INSERT INTO daily_attendance_records
|
|
(worker_id, record_date, is_present, attendance_type_id, created_by)
|
|
VALUES (?, ?, ?, 1, 1)`,
|
|
[worker_id, record_date, is_present]
|
|
);
|
|
return result.insertId;
|
|
}
|
|
}
|
|
|
|
// 특정 날짜의 출근 체크 목록 조회 (휴가 정보 포함)
|
|
static async getCheckinList(date) {
|
|
const db = await getDb();
|
|
const query = `
|
|
SELECT
|
|
w.worker_id,
|
|
w.worker_name,
|
|
w.job_type,
|
|
w.employment_status,
|
|
COALESCE(dar.is_present, TRUE) as is_present,
|
|
dar.id as record_id,
|
|
vr.request_id as vacation_request_id,
|
|
vr.status as vacation_status,
|
|
vt.type_name as vacation_type_name,
|
|
vt.type_code as vacation_type_code,
|
|
vr.days_used as vacation_days
|
|
FROM workers w
|
|
LEFT JOIN daily_attendance_records dar
|
|
ON w.worker_id = dar.worker_id AND dar.record_date = ?
|
|
LEFT JOIN vacation_requests vr
|
|
ON w.worker_id = vr.worker_id
|
|
AND ? BETWEEN vr.start_date AND vr.end_date
|
|
AND vr.status = 'approved'
|
|
LEFT JOIN vacation_types vt ON vr.vacation_type_id = vt.id
|
|
WHERE w.employment_status = 'employed'
|
|
ORDER BY w.worker_name
|
|
`;
|
|
|
|
const [rows] = await db.execute(query, [date, date]);
|
|
return rows;
|
|
}
|
|
}
|
|
|
|
module.exports = AttendanceModel;
|