2025-08-21 09:41:46 +09:00
|
|
|
// 인증 컨트롤러
|
|
|
|
|
// 기존 Java ApiLoginController를 Node.js로 포팅
|
|
|
|
|
|
|
|
|
|
import { Request, Response } from "express";
|
|
|
|
|
import { AuthService } from "../services/authService";
|
|
|
|
|
import { JwtUtils } from "../utils/jwtUtils";
|
|
|
|
|
import { LoginRequest, UserInfo, ApiResponse, PersonBean } from "../types/auth";
|
|
|
|
|
import { logger } from "../utils/logger";
|
|
|
|
|
|
|
|
|
|
export class AuthController {
|
|
|
|
|
/**
|
|
|
|
|
* POST /api/auth/login
|
|
|
|
|
* 기존 Java ApiLoginController.login() 메서드 포팅
|
|
|
|
|
*/
|
|
|
|
|
static async login(req: Request, res: Response): Promise<void> {
|
|
|
|
|
try {
|
|
|
|
|
const { userId, password }: LoginRequest = req.body;
|
|
|
|
|
const remoteAddr = req.ip || req.connection.remoteAddress || "unknown";
|
|
|
|
|
|
|
|
|
|
logger.info(`=== API 로그인 호출됨 ===`);
|
|
|
|
|
logger.info(`userId: ${userId}`);
|
|
|
|
|
logger.info(`password: ${password ? "***" : "null"}`);
|
|
|
|
|
|
|
|
|
|
// 입력값 검증
|
|
|
|
|
if (!userId || !password) {
|
|
|
|
|
res.status(400).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: "사용자 ID와 비밀번호를 입력해주세요.",
|
|
|
|
|
error: {
|
|
|
|
|
code: "INVALID_INPUT",
|
|
|
|
|
details: "필수 입력값이 누락되었습니다.",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 로그인 프로세스 실행
|
|
|
|
|
const loginResult = await AuthService.processLogin(
|
|
|
|
|
userId,
|
|
|
|
|
password,
|
|
|
|
|
remoteAddr
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (loginResult.success && loginResult.userInfo && loginResult.token) {
|
|
|
|
|
// 로그인 성공
|
|
|
|
|
const userInfo: UserInfo = {
|
|
|
|
|
userId: loginResult.userInfo.userId,
|
|
|
|
|
userName: loginResult.userInfo.userName || "",
|
|
|
|
|
deptName: loginResult.userInfo.deptName || "",
|
|
|
|
|
companyCode: loginResult.userInfo.companyCode || "ILSHIN",
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
logger.info(`=== API 로그인 사용자 정보 디버그 ===`);
|
|
|
|
|
logger.info(
|
|
|
|
|
`PersonBean companyCode: ${loginResult.userInfo.companyCode}`
|
|
|
|
|
);
|
|
|
|
|
logger.info(`반환할 사용자 정보:`);
|
|
|
|
|
logger.info(`- userId: ${userInfo.userId}`);
|
|
|
|
|
logger.info(`- userName: ${userInfo.userName}`);
|
|
|
|
|
logger.info(`- companyCode: ${userInfo.companyCode}`);
|
|
|
|
|
|
2025-10-28 14:55:41 +09:00
|
|
|
// 사용자의 첫 번째 접근 가능한 메뉴 조회
|
|
|
|
|
let firstMenuPath: string | null = null;
|
|
|
|
|
try {
|
|
|
|
|
const { AdminService } = await import("../services/adminService");
|
|
|
|
|
const paramMap = {
|
|
|
|
|
userId: loginResult.userInfo.userId,
|
|
|
|
|
userCompanyCode: loginResult.userInfo.companyCode || "ILSHIN",
|
|
|
|
|
userType: loginResult.userInfo.userType,
|
|
|
|
|
userLang: "ko",
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const menuList = await AdminService.getUserMenuList(paramMap);
|
|
|
|
|
logger.info(`로그인 후 메뉴 조회: 총 ${menuList.length}개 메뉴`);
|
|
|
|
|
|
|
|
|
|
// 접근 가능한 첫 번째 메뉴 찾기
|
|
|
|
|
// 조건:
|
|
|
|
|
// 1. LEV (레벨)이 2 이상 (최상위 폴더 제외)
|
|
|
|
|
// 2. MENU_URL이 있고 비어있지 않음
|
|
|
|
|
// 3. 이미 PATH, SEQ로 정렬되어 있으므로 첫 번째로 찾은 것이 첫 번째 메뉴
|
|
|
|
|
const firstMenu = menuList.find((menu: any) => {
|
|
|
|
|
const level = menu.lev || menu.level;
|
|
|
|
|
const url = menu.menu_url || menu.url;
|
|
|
|
|
|
|
|
|
|
return level >= 2 && url && url.trim() !== "" && url !== "#";
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (firstMenu) {
|
|
|
|
|
firstMenuPath = firstMenu.menu_url || firstMenu.url;
|
|
|
|
|
logger.info(`✅ 첫 번째 접근 가능한 메뉴 발견:`, {
|
|
|
|
|
name: firstMenu.menu_name_kor || firstMenu.translated_name,
|
|
|
|
|
url: firstMenuPath,
|
|
|
|
|
level: firstMenu.lev || firstMenu.level,
|
|
|
|
|
seq: firstMenu.seq,
|
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
logger.info(
|
|
|
|
|
"⚠️ 접근 가능한 메뉴가 없습니다. 메인 페이지로 이동합니다."
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
} catch (menuError) {
|
|
|
|
|
logger.warn("메뉴 조회 중 오류 발생 (무시하고 계속):", menuError);
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-21 09:41:46 +09:00
|
|
|
res.status(200).json({
|
|
|
|
|
success: true,
|
|
|
|
|
message: "로그인 성공",
|
|
|
|
|
data: {
|
|
|
|
|
userInfo,
|
|
|
|
|
token: loginResult.token,
|
2025-10-28 14:55:41 +09:00
|
|
|
firstMenuPath, // 첫 번째 접근 가능한 메뉴 경로 추가
|
2025-08-21 09:41:46 +09:00
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
// 로그인 실패
|
|
|
|
|
res.status(401).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: "로그인 실패",
|
|
|
|
|
error: {
|
|
|
|
|
code: "LOGIN_FAILED",
|
|
|
|
|
details:
|
|
|
|
|
loginResult.errorReason || "알 수 없는 오류가 발생했습니다.",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
} catch (error) {
|
|
|
|
|
logger.error(
|
|
|
|
|
`로그인 API 오류: ${error instanceof Error ? error.message : error}`
|
|
|
|
|
);
|
|
|
|
|
res.status(500).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: "서버 오류가 발생했습니다.",
|
|
|
|
|
error: {
|
|
|
|
|
code: "SERVER_ERROR",
|
|
|
|
|
details:
|
|
|
|
|
error instanceof Error
|
|
|
|
|
? error.message
|
|
|
|
|
: "알 수 없는 오류가 발생했습니다.",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* POST /api/auth/logout
|
|
|
|
|
* 기존 Java ApiLoginController.logout() 메서드 포팅
|
|
|
|
|
*/
|
|
|
|
|
static async logout(req: Request, res: Response): Promise<void> {
|
|
|
|
|
try {
|
|
|
|
|
const remoteAddr = req.ip || req.connection.remoteAddress || "unknown";
|
|
|
|
|
|
|
|
|
|
// JWT 토큰에서 사용자 정보 추출
|
|
|
|
|
const authHeader = req.get("Authorization");
|
|
|
|
|
const token = authHeader && authHeader.split(" ")[1];
|
|
|
|
|
|
|
|
|
|
if (token) {
|
|
|
|
|
try {
|
|
|
|
|
const userInfo = JwtUtils.verifyToken(token);
|
|
|
|
|
await AuthService.processLogout(userInfo.userId, remoteAddr);
|
|
|
|
|
} catch (tokenError) {
|
|
|
|
|
logger.warn(
|
|
|
|
|
`로그아웃 시 토큰 검증 실패: ${tokenError instanceof Error ? tokenError.message : tokenError}`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
res.status(200).json({
|
|
|
|
|
success: true,
|
|
|
|
|
message: "로그아웃되었습니다.",
|
|
|
|
|
data: null,
|
|
|
|
|
});
|
|
|
|
|
} catch (error) {
|
|
|
|
|
logger.error(
|
|
|
|
|
`로그아웃 API 오류: ${error instanceof Error ? error.message : error}`
|
|
|
|
|
);
|
|
|
|
|
res.status(500).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: "로그아웃 처리 중 오류가 발생했습니다.",
|
|
|
|
|
error: {
|
|
|
|
|
code: "LOGOUT_ERROR",
|
|
|
|
|
details:
|
|
|
|
|
error instanceof Error
|
|
|
|
|
? error.message
|
|
|
|
|
: "알 수 없는 오류가 발생했습니다.",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* GET /api/auth/me
|
|
|
|
|
* 기존 Java ApiLoginController.getCurrentUser() 메서드 포팅
|
|
|
|
|
*/
|
|
|
|
|
static async getCurrentUser(req: Request, res: Response): Promise<void> {
|
|
|
|
|
try {
|
|
|
|
|
const authHeader = req.get("Authorization");
|
|
|
|
|
const token = authHeader && authHeader.split(" ")[1];
|
|
|
|
|
|
|
|
|
|
if (!token) {
|
|
|
|
|
res.status(401).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: "인증되지 않은 사용자입니다.",
|
|
|
|
|
error: {
|
|
|
|
|
code: "NOT_AUTHENTICATED",
|
|
|
|
|
details: "세션이 만료되었거나 로그인이 필요합니다.",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const userInfo = JwtUtils.verifyToken(token);
|
|
|
|
|
|
2025-08-28 10:05:06 +09:00
|
|
|
// DB에서 최신 사용자 정보 조회 (locale 포함)
|
|
|
|
|
const dbUserInfo = await AuthService.getUserInfo(userInfo.userId);
|
|
|
|
|
|
|
|
|
|
if (!dbUserInfo) {
|
|
|
|
|
res.status(401).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: "사용자 정보를 찾을 수 없습니다.",
|
|
|
|
|
error: {
|
|
|
|
|
code: "USER_NOT_FOUND",
|
|
|
|
|
details: "사용자 정보가 삭제되었거나 존재하지 않습니다.",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-05 21:52:19 +09:00
|
|
|
// 프론트엔드 호환성을 위해 더 많은 사용자 정보 반환
|
|
|
|
|
const userInfoResponse: any = {
|
2025-08-28 10:05:06 +09:00
|
|
|
userId: dbUserInfo.userId,
|
|
|
|
|
userName: dbUserInfo.userName || "",
|
|
|
|
|
deptName: dbUserInfo.deptName || "",
|
|
|
|
|
companyCode: dbUserInfo.companyCode || "ILSHIN",
|
2025-09-05 21:52:19 +09:00
|
|
|
company_code: dbUserInfo.companyCode || "ILSHIN", // 프론트엔드 호환성
|
2025-08-28 10:05:06 +09:00
|
|
|
userType: dbUserInfo.userType || "USER",
|
|
|
|
|
userTypeName: dbUserInfo.userTypeName || "일반사용자",
|
|
|
|
|
email: dbUserInfo.email || "",
|
|
|
|
|
photo: dbUserInfo.photo,
|
|
|
|
|
locale: dbUserInfo.locale || "KR", // locale 정보 추가
|
2025-09-05 21:52:19 +09:00
|
|
|
deptCode: dbUserInfo.deptCode, // 추가 필드
|
2025-08-21 09:41:46 +09:00
|
|
|
isAdmin:
|
2025-08-28 10:05:06 +09:00
|
|
|
dbUserInfo.userType === "ADMIN" || dbUserInfo.userId === "plm_admin",
|
2025-08-21 09:41:46 +09:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
res.status(200).json({
|
|
|
|
|
success: true,
|
|
|
|
|
message: "사용자 정보 조회 성공",
|
|
|
|
|
data: userInfoResponse,
|
|
|
|
|
});
|
|
|
|
|
} catch (error) {
|
|
|
|
|
logger.error(
|
|
|
|
|
`사용자 정보 조회 API 오류: ${error instanceof Error ? error.message : error}`
|
|
|
|
|
);
|
|
|
|
|
res.status(401).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: "인증되지 않은 사용자입니다.",
|
|
|
|
|
error: {
|
|
|
|
|
code: "NOT_AUTHENTICATED",
|
|
|
|
|
details: "세션이 만료되었거나 로그인이 필요합니다.",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* GET /api/auth/status
|
|
|
|
|
* 기존 Java ApiLoginController.checkAuthStatus() 메서드 포팅
|
|
|
|
|
*/
|
|
|
|
|
static async checkAuthStatus(req: Request, res: Response): Promise<void> {
|
|
|
|
|
try {
|
|
|
|
|
const authHeader = req.get("Authorization");
|
|
|
|
|
const token = authHeader && authHeader.split(" ")[1];
|
|
|
|
|
|
|
|
|
|
if (!token) {
|
|
|
|
|
res.status(200).json({
|
|
|
|
|
success: true,
|
|
|
|
|
message: "세션 상태 확인",
|
|
|
|
|
data: {
|
|
|
|
|
isLoggedIn: false,
|
|
|
|
|
isAdmin: false,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const validation = JwtUtils.validateToken(token);
|
|
|
|
|
|
2025-08-21 13:28:49 +09:00
|
|
|
if (!validation.isValid) {
|
|
|
|
|
res.status(200).json({
|
|
|
|
|
success: true,
|
|
|
|
|
message: "세션 상태 확인",
|
|
|
|
|
data: {
|
|
|
|
|
isLoggedIn: false,
|
|
|
|
|
isAdmin: false,
|
|
|
|
|
error: validation.error,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 토큰에서 사용자 정보 추출하여 관리자 권한 확인
|
|
|
|
|
let isAdmin = false;
|
|
|
|
|
try {
|
|
|
|
|
const userInfo = JwtUtils.verifyToken(token);
|
|
|
|
|
// 기존 Java 로직과 동일: plm_admin 사용자만 관리자로 인식
|
|
|
|
|
isAdmin =
|
|
|
|
|
userInfo.userId === "plm_admin" || userInfo.userType === "ADMIN";
|
|
|
|
|
|
|
|
|
|
logger.info(`인증 상태 확인: ${userInfo.userId}, 관리자: ${isAdmin}`);
|
|
|
|
|
} catch (error) {
|
|
|
|
|
logger.error(`토큰에서 사용자 정보 추출 실패: ${error}`);
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-21 09:41:46 +09:00
|
|
|
res.status(200).json({
|
|
|
|
|
success: true,
|
|
|
|
|
message: "세션 상태 확인",
|
|
|
|
|
data: {
|
2025-08-21 13:28:49 +09:00
|
|
|
isLoggedIn: true,
|
|
|
|
|
isAdmin: isAdmin,
|
2025-08-21 09:41:46 +09:00
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
} catch (error) {
|
|
|
|
|
logger.error(
|
|
|
|
|
`세션 상태 확인 API 오류: ${error instanceof Error ? error.message : error}`
|
|
|
|
|
);
|
|
|
|
|
res.status(500).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: "세션 상태 확인 중 오류가 발생했습니다.",
|
|
|
|
|
error: {
|
|
|
|
|
code: "SESSION_CHECK_ERROR",
|
|
|
|
|
details:
|
|
|
|
|
error instanceof Error
|
|
|
|
|
? error.message
|
|
|
|
|
: "알 수 없는 오류가 발생했습니다.",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* POST /api/auth/refresh
|
|
|
|
|
* JWT 토큰 갱신 API
|
|
|
|
|
*/
|
|
|
|
|
static async refreshToken(req: Request, res: Response): Promise<void> {
|
|
|
|
|
try {
|
|
|
|
|
const authHeader = req.get("Authorization");
|
|
|
|
|
const token = authHeader && authHeader.split(" ")[1];
|
|
|
|
|
|
|
|
|
|
if (!token) {
|
|
|
|
|
res.status(401).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: "토큰이 필요합니다.",
|
|
|
|
|
error: {
|
|
|
|
|
code: "TOKEN_MISSING",
|
|
|
|
|
details: "인증 토큰이 필요합니다.",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const newToken = JwtUtils.refreshToken(token);
|
|
|
|
|
|
|
|
|
|
res.status(200).json({
|
|
|
|
|
success: true,
|
|
|
|
|
message: "토큰 갱신 성공",
|
|
|
|
|
data: {
|
|
|
|
|
token: newToken,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
} catch (error) {
|
|
|
|
|
logger.error(
|
|
|
|
|
`토큰 갱신 API 오류: ${error instanceof Error ? error.message : error}`
|
|
|
|
|
);
|
|
|
|
|
res.status(401).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: "토큰 갱신에 실패했습니다.",
|
|
|
|
|
error: {
|
|
|
|
|
code: "TOKEN_REFRESH_ERROR",
|
|
|
|
|
details:
|
|
|
|
|
error instanceof Error
|
|
|
|
|
? error.message
|
|
|
|
|
: "알 수 없는 오류가 발생했습니다.",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-12-01 18:41:02 +09:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* POST /api/auth/signup
|
|
|
|
|
* 공차중계 회원가입 API
|
|
|
|
|
*/
|
|
|
|
|
static async signup(req: Request, res: Response): Promise<void> {
|
|
|
|
|
try {
|
|
|
|
|
const { userId, password, userName, phoneNumber, licenseNumber, vehicleNumber, vehicleType } = req.body;
|
|
|
|
|
|
|
|
|
|
logger.info(`=== 공차중계 회원가입 API 호출 ===`);
|
|
|
|
|
logger.info(`userId: ${userId}, vehicleNumber: ${vehicleNumber}`);
|
|
|
|
|
|
|
|
|
|
// 입력값 검증
|
|
|
|
|
if (!userId || !password || !userName || !phoneNumber || !licenseNumber || !vehicleNumber) {
|
|
|
|
|
res.status(400).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: "필수 입력값이 누락되었습니다.",
|
|
|
|
|
error: {
|
|
|
|
|
code: "INVALID_INPUT",
|
|
|
|
|
details: "아이디, 비밀번호, 이름, 연락처, 면허번호, 차량번호는 필수입니다.",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 회원가입 처리
|
|
|
|
|
const signupResult = await AuthService.signupDriver({
|
|
|
|
|
userId,
|
|
|
|
|
password,
|
|
|
|
|
userName,
|
|
|
|
|
phoneNumber,
|
|
|
|
|
licenseNumber,
|
|
|
|
|
vehicleNumber,
|
|
|
|
|
vehicleType,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (signupResult.success) {
|
|
|
|
|
logger.info(`공차중계 회원가입 성공: ${userId}`);
|
|
|
|
|
res.status(201).json({
|
|
|
|
|
success: true,
|
|
|
|
|
message: "회원가입이 완료되었습니다.",
|
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
logger.warn(`공차중계 회원가입 실패: ${userId} - ${signupResult.message}`);
|
|
|
|
|
res.status(400).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: signupResult.message || "회원가입에 실패했습니다.",
|
|
|
|
|
error: {
|
|
|
|
|
code: "SIGNUP_FAILED",
|
|
|
|
|
details: signupResult.message,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
} catch (error) {
|
|
|
|
|
logger.error("공차중계 회원가입 API 오류:", error);
|
|
|
|
|
res.status(500).json({
|
|
|
|
|
success: false,
|
|
|
|
|
message: "회원가입 처리 중 오류가 발생했습니다.",
|
|
|
|
|
error: {
|
|
|
|
|
code: "SIGNUP_ERROR",
|
|
|
|
|
details: error instanceof Error ? error.message : "알 수 없는 오류",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-08-21 09:41:46 +09:00
|
|
|
}
|