기능구현중

This commit is contained in:
2026-01-11 17:01:01 +09:00
parent 375d5bf91a
commit 954ba21211
148 changed files with 2276 additions and 0 deletions

View File

@@ -1,64 +0,0 @@
import { query, execute } from '../../../utils/db'
import { requireAdmin } from '../../../utils/session'
/**
* 직원 삭제
* DELETE /api/employee/[id]/delete
*/
export default defineEventHandler(async (event) => {
// 관리자 권한 체크 (role 기반)
const currentUserId = await requireAdmin(event)
const employeeId = getRouterParam(event, 'id')
if (!employeeId) {
throw createError({ statusCode: 400, message: '직원 ID가 필요합니다.' })
}
// 본인 삭제 방지
if (parseInt(employeeId) === currentUserId) {
throw createError({ statusCode: 400, message: '본인은 삭제할 수 없습니다.' })
}
// 직원 존재 여부 확인
const employee = await query<any>(`
SELECT employee_id, employee_name FROM wr_employee_info WHERE employee_id = $1
`, [employeeId])
if (!employee[0]) {
throw createError({ statusCode: 404, message: '직원을 찾을 수 없습니다.' })
}
// 주간보고 존재 여부 확인
const reports = await query<any>(`
SELECT COUNT(*) as cnt FROM wr_weekly_report WHERE author_id = $1
`, [employeeId])
const reportCount = parseInt(reports[0].cnt)
if (reportCount > 0) {
// 주간보고가 있으면 비활성화만
await execute(`
UPDATE wr_employee_info
SET is_active = false, updated_at = NOW()
WHERE employee_id = $1
`, [employeeId])
return {
success: true,
action: 'deactivated',
message: `${employee[0].employee_name}님이 비활성화되었습니다. (주간보고 ${reportCount}건 보존)`
}
} else {
// 주간보고가 없으면 완전 삭제 (관련 데이터 포함)
await execute(`DELETE FROM wr_employee_role WHERE employee_id = $1`, [employeeId])
await execute(`DELETE FROM wr_session WHERE employee_id = $1`, [employeeId])
await execute(`DELETE FROM wr_login_history WHERE employee_id = $1`, [employeeId])
await execute(`DELETE FROM wr_employee_info WHERE employee_id = $1`, [employeeId])
return {
success: true,
action: 'deleted',
message: `${employee[0].employee_name}님이 삭제되었습니다.`
}
}
})

View File

@@ -1,72 +0,0 @@
import { queryOne, query } from '../../../utils/db'
import { requireAuth, getSessionIdFromCookie, getDbSession } from '../../../utils/session'
/**
* 직원 상세 조회
* GET /api/employee/[id]/detail
*/
export default defineEventHandler(async (event) => {
await requireAuth(event)
const employeeId = getRouterParam(event, 'id')
// 세션에서 현재 로그인 히스토리 ID 가져오기
const sessionId = getSessionIdFromCookie(event)
const session = sessionId ? await getDbSession(sessionId) : null
const currentHistoryId = session?.loginHistoryId || null
const employee = await queryOne<any>(`
SELECT * FROM wr_employee_info WHERE employee_id = $1
`, [employeeId])
if (!employee) {
throw createError({ statusCode: 404, message: '직원을 찾을 수 없습니다.' })
}
// 로그인 이력 조회 (최근 20건)
const loginHistory = await query<any>(`
SELECT
history_id,
login_at,
login_ip,
logout_at,
logout_ip
FROM wr_login_history
WHERE employee_id = $1
ORDER BY login_at DESC
LIMIT 20
`, [employeeId])
return {
employee: {
employeeId: employee.employee_id,
employeeName: employee.employee_name,
employeeEmail: employee.employee_email,
employeePhone: employee.employee_phone,
employeePosition: employee.employee_position,
company: employee.company,
joinDate: employee.join_date,
isActive: employee.is_active,
createdAt: employee.created_at,
createdIp: employee.created_ip,
updatedAt: employee.updated_at,
updatedIp: employee.updated_ip,
// 계정 관련 필드
hasPassword: !!employee.password_hash,
googleId: employee.google_id,
googleEmail: employee.google_email,
googleLinkedAt: employee.google_linked_at,
lastLoginAt: employee.last_login_at,
lastLoginIp: employee.last_login_ip
},
loginHistory: loginHistory.map(h => ({
historyId: h.history_id,
loginAt: h.login_at,
loginIp: h.login_ip,
logoutAt: h.logout_at,
logoutIp: h.logout_ip,
// 현재 세션인지 여부
isCurrentSession: currentHistoryId && h.history_id === parseInt(currentHistoryId)
}))
}
})

View File

@@ -1,60 +0,0 @@
import { execute, queryOne } from '../../../utils/db'
import { getClientIp } from '../../../utils/ip'
import { getCurrentUserEmail } from '../../../utils/user'
// 빈 문자열을 null로 변환 (date 타입 등에서 필요)
const emptyToNull = (value: any) => (value === '' ? null : value)
interface UpdateEmployeeBody {
employeeName?: string
employeePhone?: string
employeePosition?: string
company?: string
joinDate?: string
isActive?: boolean
}
/**
* 직원 정보 수정
* PUT /api/employee/[id]/update
*/
export default defineEventHandler(async (event) => {
const employeeId = getRouterParam(event, 'id')
const body = await readBody<UpdateEmployeeBody>(event)
const clientIp = getClientIp(event)
const userEmail = await getCurrentUserEmail(event)
const existing = await queryOne<any>(`
SELECT * FROM wr_employee_info WHERE employee_id = $1
`, [employeeId])
if (!existing) {
throw createError({ statusCode: 404, message: '직원을 찾을 수 없습니다.' })
}
await execute(`
UPDATE wr_employee_info SET
employee_name = $1,
employee_phone = $2,
employee_position = $3,
company = $4,
join_date = $5,
is_active = $6,
updated_at = NOW(),
updated_ip = $7,
updated_email = $8
WHERE employee_id = $9
`, [
body.employeeName ?? existing.employee_name,
emptyToNull(body.employeePhone) ?? existing.employee_phone,
emptyToNull(body.employeePosition) ?? existing.employee_position,
body.company ?? existing.company,
emptyToNull(body.joinDate) ?? existing.join_date,
body.isActive ?? existing.is_active,
clientIp,
userEmail,
employeeId
])
return { success: true }
})

View File

@@ -1,62 +0,0 @@
import { insertReturning, queryOne } from '../../utils/db'
import { getClientIp } from '../../utils/ip'
import { getCurrentUserEmail } from '../../utils/user'
interface CreateEmployeeBody {
employeeName: string
employeeEmail: string
employeePhone?: string
employeePosition?: string
company?: string
joinDate?: string
}
/**
* 직원 등록
* POST /api/employee/create
*/
export default defineEventHandler(async (event) => {
const body = await readBody<CreateEmployeeBody>(event)
const clientIp = getClientIp(event)
const userEmail = await getCurrentUserEmail(event)
if (!body.employeeName || !body.employeeEmail) {
throw createError({ statusCode: 400, message: '이름과 이메일은 필수입니다.' })
}
// 이메일 중복 체크
const existing = await queryOne(`
SELECT employee_id FROM wr_employee_info WHERE employee_email = $1
`, [body.employeeEmail.toLowerCase()])
if (existing) {
throw createError({ statusCode: 409, message: '이미 등록된 이메일입니다.' })
}
const employee = await insertReturning(`
INSERT INTO wr_employee_info (
employee_name, employee_email, employee_phone,
employee_position, company, join_date,
created_ip, created_email, updated_ip, updated_email
) VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $7, $8)
RETURNING *
`, [
body.employeeName,
body.employeeEmail.toLowerCase(),
body.employeePhone || null,
body.employeePosition || null,
body.company || '(주)터보소프트',
body.joinDate || null,
clientIp,
userEmail
])
return {
success: true,
employee: {
employeeId: employee.employee_id,
employeeName: employee.employee_name,
employeeEmail: employee.employee_email
}
}
})

View File

@@ -1,32 +0,0 @@
import { query } from '../../utils/db'
/**
* 직원 목록 조회
* GET /api/employee/list
*/
export default defineEventHandler(async (event) => {
const queryParams = getQuery(event)
const activeOnly = queryParams.activeOnly !== 'false'
let sql = `
SELECT * FROM wr_employee_info
${activeOnly ? 'WHERE is_active = true' : ''}
ORDER BY employee_name
`
const employees = await query(sql)
return {
employees: employees.map((e: any) => ({
employeeId: e.employee_id,
employeeName: e.employee_name,
employeeEmail: e.employee_email,
employeePhone: e.employee_phone,
employeePosition: e.employee_position,
company: e.company,
joinDate: e.join_date,
isActive: e.is_active,
createdAt: e.created_at
}))
}
})

View File

@@ -1,29 +0,0 @@
import { query } from '../../utils/db'
/**
* 사원 검색 (이름/이메일)
* GET /api/employee/search?q=keyword
*/
export default defineEventHandler(async (event) => {
const queryParams = getQuery(event)
const keyword = queryParams.q as string
if (!keyword || keyword.length < 1) {
return []
}
const employees = await query(`
SELECT * FROM wr_employee_info
WHERE is_active = true
AND (employee_name ILIKE $1 OR employee_email ILIKE $1)
ORDER BY employee_name
LIMIT 20
`, [`%${keyword}%`])
return employees.map((e: any) => ({
employeeId: e.employee_id,
employeeName: e.employee_name,
employeeEmail: e.employee_email,
employeePosition: e.employee_position
}))
})