Compare commits
32 Commits
aacd2fe7db
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
1c54c1c477 | ||
|
|
8129f85071 | ||
| 933afb02b1 | |||
|
|
87459c8f44 | ||
|
|
4d98d9a48e | ||
|
|
1da75e4abd | ||
|
|
3e26420945 | ||
|
|
8e22c1d713 | ||
|
|
8747b3946f | ||
|
|
ed3d8812c2 | ||
|
|
5588fae6f9 | ||
|
|
e6afe2b6d3 | ||
|
|
9049b60ee5 | ||
|
|
a5c4a15fab | ||
|
|
1ab59bc9e1 | ||
|
|
7389ed2d82 | ||
|
|
a73dd76e70 | ||
|
|
cbfc1bcd1d | ||
|
|
6ed939c6bf | ||
|
|
1ecee53966 | ||
|
|
322a8ae882 | ||
|
|
8176180e52 | ||
|
|
2137ee364c | ||
|
|
afd89322bb | ||
|
|
1457bf277f | ||
|
|
0bfff08af6 | ||
| ae1fd4b121 | |||
|
|
1eca0ede91 | ||
|
|
f36e8e93e2 | ||
|
|
9f165faf13 | ||
|
|
577f138533 | ||
|
|
237ac9ee25 |
@@ -8,3 +8,6 @@ npm-debug.log
|
|||||||
uploads
|
uploads
|
||||||
*.xlsx
|
*.xlsx
|
||||||
*.log
|
*.log
|
||||||
|
mysql_data
|
||||||
|
scratch
|
||||||
|
*.sql
|
||||||
6
.env
Normal file
6
.env
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
DB_HOST=itam-mysql
|
||||||
|
DB_PORT=3306
|
||||||
|
DB_USER=itam
|
||||||
|
DB_PASS=itam1234
|
||||||
|
DB_NAME=itam
|
||||||
|
PORT=3000
|
||||||
@@ -85,7 +85,7 @@ jobs:
|
|||||||
|
|
||||||
ssh "${PROD_USER}@${PROD_HOST}" "if [ ! -d '${PROD_DEPLOY_PATH}/.git' ]; then git clone '${PROD_GIT_URL}' '${PROD_DEPLOY_PATH}'; else cd '${PROD_DEPLOY_PATH}' && git remote set-url origin '${PROD_GIT_URL}'; fi"
|
ssh "${PROD_USER}@${PROD_HOST}" "if [ ! -d '${PROD_DEPLOY_PATH}/.git' ]; then git clone '${PROD_GIT_URL}' '${PROD_DEPLOY_PATH}'; else cd '${PROD_DEPLOY_PATH}' && git remote set-url origin '${PROD_GIT_URL}'; fi"
|
||||||
|
|
||||||
ssh "${PROD_USER}@${PROD_HOST}" "cd '${PROD_DEPLOY_PATH}' && git fetch origin '${TARGET_BRANCH}' && git checkout -B '${TARGET_BRANCH}' FETCH_HEAD && git reset --hard FETCH_HEAD"
|
ssh "${PROD_USER}@${PROD_HOST}" "cd '${PROD_DEPLOY_PATH}' && git checkout -- .env || true && git fetch origin '${TARGET_BRANCH}' && git checkout -B '${TARGET_BRANCH}' FETCH_HEAD && git reset --hard FETCH_HEAD"
|
||||||
|
|
||||||
EFFECTIVE_BACKUP_ROOT="${PROD_BACKUP_ROOT:-/home/user/dachs_backups}"
|
EFFECTIVE_BACKUP_ROOT="${PROD_BACKUP_ROOT:-/home/user/dachs_backups}"
|
||||||
|
|
||||||
|
|||||||
1
.gitignore
vendored
1
.gitignore
vendored
@@ -6,3 +6,4 @@ dist/
|
|||||||
.DS_Store
|
.DS_Store
|
||||||
Thumbs.db
|
Thumbs.db
|
||||||
backups/
|
backups/
|
||||||
|
mysql_data/
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ RUN npm ci
|
|||||||
# Copy source code
|
# Copy source code
|
||||||
COPY src ./src
|
COPY src ./src
|
||||||
COPY public ./public
|
COPY public ./public
|
||||||
COPY index.html map_editor.html ./
|
COPY index.html map_editor.html mobile.html ./
|
||||||
|
|
||||||
# Build application
|
# Build application
|
||||||
RUN npm run build
|
RUN npm run build
|
||||||
|
|||||||
@@ -1,103 +0,0 @@
|
|||||||
# 자산관리 시스템 운영 오픈 보고 요약
|
|
||||||
|
|
||||||
## 1. 운영 방식
|
|
||||||
|
|
||||||
```mermaid
|
|
||||||
flowchart LR
|
|
||||||
A[개발 완료] --> B[개발 완료 검증]
|
|
||||||
B --> C[운영 서버 반영]
|
|
||||||
C --> D[최종 점검]
|
|
||||||
D --> E[서비스 오픈]
|
|
||||||
|
|
||||||
linkStyle default stroke:#d32f2f,stroke-width:2px;
|
|
||||||
```
|
|
||||||
|
|
||||||
| 구분 | 운영 방향 |
|
|
||||||
| --- | --- |
|
|
||||||
| 반영 기준 | 개발 완료 및 검증 후 운영에 반영 |
|
|
||||||
| 반영 방식 | 운영 담당자 기준 통제 절차 |
|
|
||||||
| 데이터 연계 | 기존 외부 DB와 연동 |
|
|
||||||
| 안정성 확보 | 반영 전 점검, 반영 전 백업, 반영 후 확인 |
|
|
||||||
|
|
||||||
임의 반영 배제, 개발 완료 및 검증 결과 기준 운영 반영 방식.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 2. 운영 배포 절차
|
|
||||||
|
|
||||||
```mermaid
|
|
||||||
flowchart TD
|
|
||||||
A[배포 준비 완료] --> B[사전 점검]
|
|
||||||
B --> C[백업 수행]
|
|
||||||
C --> D[운영 배포 실행]
|
|
||||||
D --> E[접속 및 기능 확인]
|
|
||||||
E --> F[오픈]
|
|
||||||
|
|
||||||
B1[환경 설정 확인]
|
|
||||||
B2[외부 DB 연결 확인]
|
|
||||||
B3[배포 파일 확인]
|
|
||||||
|
|
||||||
B --> B1
|
|
||||||
B --> B2
|
|
||||||
B --> B3
|
|
||||||
|
|
||||||
linkStyle default stroke:#d32f2f,stroke-width:2px;
|
|
||||||
```
|
|
||||||
|
|
||||||
| 단계 | 목적 | 담당 |
|
|
||||||
| --- | --- | --- |
|
|
||||||
| 사전 점검 | 운영 반영에 필요한 조건 확인 | 개발팀 + 운영 |
|
|
||||||
| 백업 수행 | 문제 발생 시 신속 복구 대비 | 운영 |
|
|
||||||
| 운영 배포 실행 | 운영 서버에 검증 완료 결과 반영 | 운영 |
|
|
||||||
| 접속 및 기능 확인 | 주요 화면과 서비스 상태 확인 | 개발팀 + 운영 |
|
|
||||||
| 오픈 | 사용자 대상 서비스 오픈 | 운영 |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 3. 예상 일정
|
|
||||||
|
|
||||||
```mermaid
|
|
||||||
gantt
|
|
||||||
title 자산관리 시스템 운영 오픈 일정
|
|
||||||
dateFormat YYYY-MM-DD
|
|
||||||
axisFormat %m/%d
|
|
||||||
|
|
||||||
section 준비 단계
|
|
||||||
배포 구성 정리 및 환경 보완 :done, a1, 2026-06-18, 2026-06-24
|
|
||||||
|
|
||||||
section 검증 단계
|
|
||||||
테스트 배포 및 점검 :a2, 2026-06-25, 2026-06-27
|
|
||||||
오픈 전 최종 확인 :a3, 2026-06-28, 2026-06-30
|
|
||||||
|
|
||||||
section 오픈
|
|
||||||
운영 오픈 :milestone, a4, 2026-07-01, 1d
|
|
||||||
```
|
|
||||||
|
|
||||||
| 구간 | 일정 | 주요 내용 |
|
|
||||||
| --- | --- | --- |
|
|
||||||
| 준비 단계 | 6월 18일 ~ 6월 24일 | 운영 환경 정리 및 배포 준비 완료 |
|
|
||||||
| 검증 단계 | 6월 25일 ~ 6월 30일 | 테스트 배포, 접속 확인, 최종 보완 |
|
|
||||||
| 오픈 시점 | 7월 1일 | 운영 서비스 시작 |
|
|
||||||
|
|
||||||
6월 30일까지 준비 및 검증 완료, 7월 1일 오픈.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 4. 보고 요약
|
|
||||||
|
|
||||||
| 항목 | 보고 내용 |
|
|
||||||
| --- | --- |
|
|
||||||
| 통제된 배포 | 개발 완료 및 검증 후 운영 반영 |
|
|
||||||
| 운영 안정성 | 점검과 백업 후 반영 |
|
|
||||||
| 데이터 연속성 | 기존 외부 DB와 연계 유지 |
|
|
||||||
| 오픈 목표 | 7월 1일 서비스 개시 |
|
|
||||||
|
|
||||||
1. 통제 절차 기반 운영 반영.
|
|
||||||
2. 오픈 전 점검 및 백업 기반 리스크 관리.
|
|
||||||
3. 6월 30일까지 준비 완료, 7월 1일 오픈.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 5. 결론
|
|
||||||
|
|
||||||
자산관리 시스템, 6월 30일까지 운영 배포 준비 및 검증 완료, 7월 1일 오픈.
|
|
||||||
108
TEST_LOCAL.md
108
TEST_LOCAL.md
@@ -1,108 +0,0 @@
|
|||||||
# 로컬 Docker 테스트 가이드
|
|
||||||
|
|
||||||
## 준비 사항
|
|
||||||
|
|
||||||
- Docker & Docker Compose 설치 (WSL2 Ubuntu 권장)
|
|
||||||
- Node.js 20 (로컬 빌드 테스트 시)
|
|
||||||
|
|
||||||
## 테스트 단계
|
|
||||||
|
|
||||||
### 1. 파일 구조 확인
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# docker-compose.test.yaml이 다음을 사용 확인
|
|
||||||
ls -la docker/nginx/default.conf
|
|
||||||
ls -la docker/frontend/default.conf
|
|
||||||
ls -la Dockerfile.backend.prod
|
|
||||||
ls -la Dockerfile.frontend.prod
|
|
||||||
ls -la package.json
|
|
||||||
ls -la src/
|
|
||||||
ls -la public/
|
|
||||||
ls -la index.html
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2. Compose 파일 검증
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker compose -f docker-compose.test.yaml config
|
|
||||||
```
|
|
||||||
|
|
||||||
### 3. 이미지 빌드 테스트
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# 개별 빌드 테스트
|
|
||||||
docker build -f Dockerfile.backend.prod -t itam-backend:test .
|
|
||||||
docker build -f Dockerfile.frontend.prod -t itam-frontend:test .
|
|
||||||
```
|
|
||||||
|
|
||||||
### 4. 컨테이너 시작
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# WSL 터미널에서
|
|
||||||
cd /mnt/c/Users/user/Desktop/안건\ 파일/itam
|
|
||||||
|
|
||||||
# 또는 docker-compose.test.yaml을 사용하여 전체 스택 시작
|
|
||||||
docker compose -f docker-compose.test.yaml up --build
|
|
||||||
|
|
||||||
# 백그라운드에서 실행
|
|
||||||
docker compose -f docker-compose.test.yaml up -d --build
|
|
||||||
```
|
|
||||||
|
|
||||||
### 5. 컨테이너 상태 확인
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker compose -f docker-compose.test.yaml ps
|
|
||||||
docker logs itam-backend-test
|
|
||||||
docker logs itam-frontend-test
|
|
||||||
docker logs itam-nginx-test
|
|
||||||
```
|
|
||||||
|
|
||||||
### 6. 브라우저 테스트
|
|
||||||
|
|
||||||
```
|
|
||||||
http://localhost:8080/ # Frontend 접근
|
|
||||||
http://localhost:8080/api/ # Backend API 테스트
|
|
||||||
http://localhost:3000/health # 직접 backend health check
|
|
||||||
```
|
|
||||||
|
|
||||||
### 7. 정리
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker compose -f docker-compose.test.yaml down
|
|
||||||
```
|
|
||||||
|
|
||||||
## 예상되는 포트 매핑
|
|
||||||
|
|
||||||
- 8080: Nginx reverse proxy (frontend route to static + /api to backend)
|
|
||||||
- 3000: Backend Express API (내부, frontend와 nginx를 통해 접근)
|
|
||||||
- 80: Frontend Nginx (내부, 정적 파일 서빙)
|
|
||||||
|
|
||||||
## 테스트 체크리스트
|
|
||||||
|
|
||||||
- [ ] docker compose config 성공
|
|
||||||
- [ ] docker build 성공 (backend)
|
|
||||||
- [ ] docker build 성공 (frontend)
|
|
||||||
- [ ] 컨테이너 모두 실행 중 (ps 확인)
|
|
||||||
- [ ] http://localhost:8080 접근 가능 (frontend 페이지 로드)
|
|
||||||
- [ ] http://localhost:8080/api 응답 확인 (backend proxy 동작)
|
|
||||||
- [ ] backend health check 성공 (docker logs에서 /health 요청 확인)
|
|
||||||
|
|
||||||
## 문제 해결
|
|
||||||
|
|
||||||
### npm run build 실패
|
|
||||||
- Dockerfile.frontend.prod에서 tsc && vite build 실행
|
|
||||||
- package.json과 tsconfig.json 확인
|
|
||||||
|
|
||||||
### nginx 포트 이미 사용 중
|
|
||||||
```bash
|
|
||||||
docker ps
|
|
||||||
docker stop container_name
|
|
||||||
```
|
|
||||||
|
|
||||||
### DB 연결 실패
|
|
||||||
- 정상 동작 (NODE_ENV 때문에 /health는 200 반환)
|
|
||||||
- 실제 API 호출 시 DB 오류 예상
|
|
||||||
|
|
||||||
### 권한 문제
|
|
||||||
- logs/ 디렉토리 소유권 확인
|
|
||||||
- 필요 시 mkdir -p logs/nginx && chmod 777 logs
|
|
||||||
@@ -1,30 +0,0 @@
|
|||||||
# 📝 작업 보고서 (2026-06-15)
|
|
||||||
|
|
||||||
## 1. 서버 및 개발 환경 설정
|
|
||||||
- **백엔드 서버 구동**: 3000번 포트(DB 서버) 정상 구동 완료.
|
|
||||||
- **프론트엔드 서버 구동**: 8080번 포트 정상 구동 완료.
|
|
||||||
- **브랜치 전환**: \`db_setting\` 브랜치로 전환 및 최신 코드 Pull 완료.
|
|
||||||
|
|
||||||
## 2. 데이터베이스 정제 및 보강 (Surgical Update)
|
|
||||||
- **사용자 정보(system_users) 업데이트**:
|
|
||||||
- 엑셀(\`system_User (20260615).xlsx\`) 기반 987건 신규 입력.
|
|
||||||
- 기존 백업 데이터(212건)와 병합하여 총 1,199건의 사용자 DB 구축.
|
|
||||||
- **PC 자산(asset_pc) 데이터 입력**:
|
|
||||||
- 엑셀(\`asset_pc (2026.06.15).xlsx\`) 기반 1,030건 입력 완료.
|
|
||||||
- **용량 정제**: 괄호 제거 및 4자리 GB 단위를 TB로 자동 변환 (예: 1863GB -> 1.86TB).
|
|
||||||
- **구매일 보강**: 연도 데이터에 월/일 추가 (\`YYYY-12-01\` 형식으로 통일).
|
|
||||||
- **자산번호 재매핑**: \`PC-YYYY12-NNNN\` 형식으로 전수 재부여 및 기존 번호와의 연속성 유지.
|
|
||||||
|
|
||||||
## 3. 부서 및 자산 유형 정상화
|
|
||||||
- **부서명 통합**: '총괄기획실', '기술개발센터', '한맥', '장헌', 'PTC', '현타' 등을 제외한 1,045건의 부서명을 **'삼안'**으로 일괄 통합.
|
|
||||||
- **자산 유형 교정 (핵심)**:
|
|
||||||
- 엑셀의 오기입과 상관없이 **사번(emp_no) 존재 여부**를 기준으로 자산 유형을 재분류.
|
|
||||||
- 사번이 있는 991건 -> **개인PC**로 정상화.
|
|
||||||
- 사번이 없는 39건 -> **공용PC**로 지정 및 사용자명 '공용'으로 정리.
|
|
||||||
|
|
||||||
## 4. 운영 규칙 업데이트
|
|
||||||
- **README.md 수정**: 'DB 삭제 및 초기화 절대 엄금 (Rule 5)' 항목 추가.
|
|
||||||
|
|
||||||
---
|
|
||||||
**보고자**: Gemini CLI
|
|
||||||
**상태**: 소스 코드 수정 없음, 데이터베이스 정제 완료.
|
|
||||||
Binary file not shown.
132
baron-sso_login_guide/BARONSSO_loginAPI_task.md
Normal file
132
baron-sso_login_guide/BARONSSO_loginAPI_task.md
Normal file
@@ -0,0 +1,132 @@
|
|||||||
|
# BARON-SSO 통합 로그인 연동 가이드 (자산관리 시스템)
|
||||||
|
|
||||||
|
이 문서는 Baron SSO(OpenID Connect IdP)를 자산관리 시스템에 통합 연동하기 위한 상세 가이드입니다. Headless 로그인 방식과 OIDC 표준을 기반으로 사용자 인증 및 정보 획득 과정을 설명하며, Baron SSO `devfront` 페이지에서의 애플리케이션 설정 방법과 연동 시 필요한 핵심 로직을 다룹니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. 개요
|
||||||
|
|
||||||
|
자산관리 시스템에 Baron SSO를 연동하여 통합 로그인을 구현합니다. 이를 통해 사용자는 Baron SSO에서 인증하고, 자산관리 시스템은 사용자의 신원을 확인하고 필요한 정보를 획득하게 됩니다. 본 가이드는 주로 **Headless 로그인** 방식을 사용하며, 이는 사용자에게 Baron SSO의 로그인 화면을 직접 노출하지 않고 자산관리 시스템 내에서 인증 과정을 처리하는 방식입니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. OIDC 핵심 개념
|
||||||
|
|
||||||
|
### 2.1. Headless Login
|
||||||
|
사용자가 IdP(Baron SSO)의 로그인 화면을 거치지 않고, RP(Relying Party, 자산관리 시스템) 내에서 직접 사용자 인증 정보를 입력받아 백채널(Back-channel)로 인증을 수행하는 방식입니다.
|
||||||
|
|
||||||
|
### 2.2. Client Assertion
|
||||||
|
보안 강화를 위해 RP가 자신이 신뢰할 수 있는 앱임을 증명하는 JWT(JSON Web Token)입니다. 이는 중요한 인증 요청 시 함께 전송됩니다.
|
||||||
|
|
||||||
|
### 2.3. 스코프 (Scopes)
|
||||||
|
사용자 정보에 접근하기 위한 권한 요청 단위입니다.
|
||||||
|
- `openid`: OIDC 인증 요청임을 명시하며 `id_token` 발급에 필수적입니다.
|
||||||
|
- `profile`: 사용자의 기본 프로필 정보(이름, 사번, 부서명 등) 접근 권한을 요청합니다.
|
||||||
|
- `email`: 사용자의 이메일 주소 정보 접근 권한을 요청합니다.
|
||||||
|
|
||||||
|
### 2.4. 자동 승인 (Auto-Consent)
|
||||||
|
Headless 환경에서 사용자의 정보 제공 동의 화면 없이, RP 서버가 백그라운드에서 동의 과정을 자동으로 처리하는 로직입니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. Baron SSO (devfront) 애플리케이션 생성 및 설정
|
||||||
|
|
||||||
|
자산관리 시스템을 Baron SSO에 연동하기 위해 `devfront` 관리자 도구에서 Headless RP를 설정해야 합니다. 이때 IdP 서버가 RP의 JWKS 엔드포인트에 접속할 수 있어야 함을 유의하십시오.
|
||||||
|
|
||||||
|
### 3.1. Step 1: 클라이언트 기본 정보 입력
|
||||||
|
|
||||||
|
1. `devfront`에 접속하여 **연동 앱 > 연동 앱 추가**를 클릭합니다.
|
||||||
|
2. **Name**: `asset-management-system` (또는 자산관리 시스템의 이름을 입력)
|
||||||
|
3. **Redirect URIs**: 자산관리 시스템의 콜백 경로를 입력합니다.
|
||||||
|
- 예: `http://[자산관리_시스템_IP_또는_도메인]:[포트]/callback`
|
||||||
|
- **주의**: 실제 자산관리 시스템에서 요청하는 Redirect URI와 여기서 등록한 값이 **완전히 일치**해야 인증 거부 에러가 발생하지 않습니다.
|
||||||
|
4. **Scopes**: `openid`, `profile`, `email`을 선택합니다.
|
||||||
|
5. **Type**: 반드시 `pkce`를 선택합니다.
|
||||||
|
|
||||||
|
### 3.2. Step 2: Headless 기능 및 보안 설정
|
||||||
|
|
||||||
|
1. `pkce` 하단의 **"Headless Login (자체 로그인 UI 사용)"** 토글을 활성화합니다.
|
||||||
|
2. **JWKS URI**: **Baron SSO 서버에서 접근 가능한** 자산관리 시스템의 공개키 주소를 입력합니다.
|
||||||
|
- 예: `http://[자산관리_시스템_IP_또는_도메인]:[포트]/.well-known/jwks.json`
|
||||||
|
- **주의**: Baron SSO 서버에서 자산관리 시스템의 포트로 네트워크가 열려있어야 합니다.
|
||||||
|
|
||||||
|
### 3.3. Step 3: 저장 및 확인 (JWKS 캐시 검증)
|
||||||
|
|
||||||
|
1. **앱 생성** 버튼을 클릭하여 설정을 저장합니다.
|
||||||
|
2. 연동 앱 목록에서 생성된 앱이 **PKCE (Headless Login)** 유형인지 확인합니다.
|
||||||
|
3. 앱 상세 페이지 하단이나 설정 탭에서 **JWKS 캐시 상태**가 `Success`인지 반드시 확인합니다. 캐시 상태가 비어있거나 실패인 경우, 자산관리 시스템이 켜져 있는지 확인하고 **새로고침(Refresh)** 버튼을 눌러 수동으로 캐시를 갱신해야 합니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 4. 자산관리 시스템 연동 로직
|
||||||
|
|
||||||
|
### 4.1. 사번 로그인 (Headless Employee ID Login Flow)
|
||||||
|
|
||||||
|
사용자가 자산관리 시스템에서 사번과 비밀번호를 입력하여 로그인하는 시퀀스입니다.
|
||||||
|
|
||||||
|
1. **신호 요청**: 자산관리 시스템은 Baron SSO의 Authorization Endpoint(`oauth2/auth`)에 요청하여 `login_challenge`를 획득합니다.
|
||||||
|
2. **본인 인증**: 자산관리 시스템은 Private Key로 서명된 `client_assertion`(JWT)과 사용자의 사번, 비밀번호, `login_challenge`를 포함하여 Baron SSO의 Headless 로그인 API(`POST /api/v1/auth/headless/password/login`)를 호출합니다.
|
||||||
|
3. **권한 획득 (자동 승인 포함)**: 인증 성공 후 Baron SSO가 반환하는 `redirectTo` 주소를 추적하며, 이 과정에서 `consent` 화면이 감지되면 자산관리 시스템은 백그라운드에서 동의 API(`POST /api/v1/auth/consent/accept`)를 호출하여 자동으로 승인합니다. 최종적으로 `Authorization Code`를 획득합니다.
|
||||||
|
4. **인증키 발급**: 획득한 `Authorization Code`와 새로 생성한 `client_assertion`을 사용하여 Baron SSO의 Token Endpoint에 요청, `id_token`과 `access_token`을 발급받습니다.
|
||||||
|
5. **로그인 완료**: `id_token`을 검증하고 사용자 정보를 추출하여 자산관리 시스템의 세션을 생성하고 로그인 완료 처리합니다.
|
||||||
|
|
||||||
|
### 4.2. 전화번호 인증 로그인 (Headless Phone Number Authentication Login Flow)
|
||||||
|
|
||||||
|
사용자가 자산관리 시스템에서 전화번호를 입력하여 인증 링크를 받고, 모바일에서 승인하여 로그인하는 시퀀스입니다.
|
||||||
|
|
||||||
|
1. **신호 요청**: 사번 로그인과 동일하게 `login_challenge`를 획득합니다.
|
||||||
|
2. **링크 발송 요청**: 자산관리 시스템은 `client_assertion`, `login_challenge`, 사용자의 전화번호(`loginId`)를 포함하여 Baron SSO의 인증 링크 발송 API(`POST /api/v1/auth/headless/link/init`)를 호출합니다. Baron SSO는 사용자에게 인증 링크를 발송하고 `pendingRef`를 반환합니다.
|
||||||
|
3. **인증 상태 폴링**: 자산관리 시스템은 `pendingRef`와 `client_assertion`을 포함하여 Baron SSO의 상태 확인 API(`POST /api/v1/auth/headless/link/poll`)를 주기적으로 호출합니다. 사용자가 모바일에서 링크를 클릭하여 인증을 완료하면, Baron SSO는 `redirectTo` 정보를 반환하며 폴링이 종료됩니다.
|
||||||
|
4. **이후 과정**: 폴링 결과로 받은 `redirectTo` 주소부터는 사번 로그인과 동일하게 표준 OIDC 흐름을 따릅니다 (자동 승인, `Authorization Code` 획득, `id_token` 발급, 세션 생성).
|
||||||
|
|
||||||
|
### 4.3. 자동 승인 및 스코프 처리 (Auto-Consent and Scope Handling)
|
||||||
|
|
||||||
|
자산관리 시스템은 `openid`, `profile`, `email`과 같은 스코프를 Baron SSO에 요청합니다. Headless 환경에서는 SSO 서버로부터 리다이렉트 주소에 `/consent`가 포함될 경우, 자산관리 시스템은 이를 감지하여 `consent_challenge`를 추출하고, 동의 상세 정보(`GET /api/v1/auth/consent`)를 요청하여 필요한 스코프를 확인합니다. 이후, 확인된 스코프를 모두 허용(`grant_scope`)한다고 Baron SSO의 동의 승인 API(`POST /api/v1/auth/consent/accept`)에 전송하여 자동으로 승인 과정을 처리합니다.
|
||||||
|
|
||||||
|
### 4.4. Tenant 접근 제한 예외 처리 (Tenant Access Restriction Exception Handling)
|
||||||
|
|
||||||
|
자산관리 시스템이 `tenant_access_restricted=true`로 설정되어 있고, 현재 사용자의 tenant가 허용된 `allowed_tenants`에 포함되지 않는 경우, Baron SSO는 `403` HTTP 상태 코드와 `code=tenant_not_allowed`를 응답합니다. 자산관리 시스템은 이 응답을 감지하여 사용자에게 적절한 에러 페이지(예: `/ko/error?error=tenant_not_allowed...`)로 리다이렉트하여 처리해야 합니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 5. 자산관리 시스템 로컬 설정 (.env)
|
||||||
|
|
||||||
|
자산관리 시스템 프로젝트 루트의 `.env` 파일에 다음과 같이 Baron SSO 연동 설정을 구성합니다.
|
||||||
|
|
||||||
|
```env
|
||||||
|
# 애플리케이션 실행 포트 (자산관리 시스템의 포트)
|
||||||
|
PORT=3000 # 예시
|
||||||
|
|
||||||
|
# OIDC IdP 설정 (Baron SSO 서버 주소)
|
||||||
|
CLIENT_ID=... # devfront에서 발급받은 클라이언트 ID
|
||||||
|
ISSUER=http://[BARON_SSO_IP_또는_도메인]/oidc
|
||||||
|
|
||||||
|
# 리다이렉트 및 보안 설정
|
||||||
|
# REDIRECT_URI는 DevFront에 등록한 Redirect URIs 중 하나와 정확히 일치해야 합니다.
|
||||||
|
REDIRECT_URI=http://[자산관리_시스템_IP_또는_도메인]:[포트]/callback
|
||||||
|
# JWKS_URI는 Baron SSO 서버가 자산관리 시스템으로 접속할 때 사용하는 주소와 일치해야 합니다.
|
||||||
|
JWKS_URI=http://[자산관리_시스템_IP_또는_도메인]:[포트]/.well-known/jwks.json
|
||||||
|
|
||||||
|
# tenant 제한 시 이동시킬 userfront 에러 경로
|
||||||
|
ERROR_LOCALE_PATH=/ko/error
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 6. 주의 사항 (네트워크 구성)
|
||||||
|
|
||||||
|
- **서버 간 통신 (Server-to-Server)**: Headless 로그인은 Baron SSO 서버가 직접 자산관리 시스템의 `JWKS_URI`에 접속하여 서명을 검증합니다. 따라서 IdP 서버에서 자산관리 시스템의 포트로의 인바운드 통신이 허용되어야 합니다. 방화벽 및 네트워크 설정을 확인하십시오.
|
||||||
|
- **Redirect URI 일치**: 브라우저를 통해 접속할 주소가 IP라면, Redirect URI와 JWKS URI 모두 IP 기반 주소로 통일하는 것이 문제 발생 소지를 줄이는 가장 좋은 방법입니다. 도메인을 사용한다면 일관되게 도메인으로 설정해야 합니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 7. 실행
|
||||||
|
|
||||||
|
자산관리 시스템 프로젝트에서 다음 명령어를 실행하여 필요한 패키지를 설치하고 서버를 시작합니다.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
npm install
|
||||||
|
npm start
|
||||||
|
```
|
||||||
|
|
||||||
|
서버 실행 시 `keys.json` 파일이 자동으로 생성되며, `.env`에 설정된 `JWKS_URI` 경로로 공개키가 서빙되어 Baron SSO가 이를 검증에 사용할 수 있게 됩니다.
|
||||||
119
baron-sso_login_guide/headless-consent-and-scope.md
Normal file
119
baron-sso_login_guide/headless-consent-and-scope.md
Normal file
@@ -0,0 +1,119 @@
|
|||||||
|
# Headless 스코프(Scope) 및 자동 승인(Auto-Consent) 가이드
|
||||||
|
|
||||||
|
이 문서는 Headless 로그인 환경에서 애플리케이션이 사용자 정보를 가져오기 위해 사용하는 **스코프(Scope)**의 개념과, IdP의 화면 없이 권한을 획득하는 **자동 승인(Auto-Consent)** 로직을 설명합니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. 요구 스코프 (Requested Scopes)
|
||||||
|
|
||||||
|
데모 앱은 OIDC 표준에 따라 다음 스코프를 IdP(Baron SSO)에 요청합니다.
|
||||||
|
|
||||||
|
| 스코프 | 필수 여부 | 역할 및 획득 데이터 |
|
||||||
|
| :-------- | :-------: | :------------------------------------------------------------------ |
|
||||||
|
| `openid` | **필수** | 해당 요청이 OIDC 인증임을 명시. `id_token` 발급을 위해 반드시 필요. |
|
||||||
|
| `profile` | 권장 | 사용자의 기본 프로필 정보(이름, 사번, 부서명 등) 접근 권한. |
|
||||||
|
| `email` | 권장 | 사용자의 이메일 주소 정보 접근 권한. |
|
||||||
|
|
||||||
|
데모 앱은 최초 인증 요청(`GET /oauth2/auth`) 시 `scope=openid profile` 형식으로 권한을 요구합니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. 자동 승인(Auto-Consent) 흐름도
|
||||||
|
|
||||||
|
일반적인 OIDC 환경에서는 사용자에게 '정보 제공 동의' 화면이 노출되지만, Headless 환경에서는 **RP 서버가 이 과정을 백그라운드에서 자동으로 수행**합니다.
|
||||||
|
|
||||||
|
```mermaid
|
||||||
|
sequenceDiagram
|
||||||
|
autonumber
|
||||||
|
participant Browser as 사용자 브라우저
|
||||||
|
participant RP as 데모 앱 서버 (RP)
|
||||||
|
participant SSO as Baron SSO (IdP)
|
||||||
|
|
||||||
|
RP->>SSO: 1. 본인 인증 요청 (전화번호)
|
||||||
|
SSO-->>RP: 2. 인증 성공 및 리다이렉트 (redirectTo: /consent...)
|
||||||
|
|
||||||
|
Note over RP: [자동 승인 로직 시작]
|
||||||
|
RP->>RP: 3. 리다이렉트 경로 중 '/consent' 감지
|
||||||
|
RP->>SSO: 4. 동의 상세 정보 요청 (GET /api/v1/auth/consent)
|
||||||
|
SSO-->>RP: 5. 요청된 스코프 정보 반환 (openid, profile 등)
|
||||||
|
|
||||||
|
RP->>SSO: 6. 동의 승인 API 호출 (POST /api/v1/auth/consent/accept)<br/>[grant_scope 포함]
|
||||||
|
SSO-->>RP: 7. 최종 리다이렉트 주소 반환 (code=...)
|
||||||
|
|
||||||
|
Note over RP: [표준 OIDC 흐름 복귀]
|
||||||
|
RP->>SSO: 8. Token Exchange (Code -> id_token)
|
||||||
|
SSO-->>RP: 9. 사용자 정보가 담긴 id_token 발급
|
||||||
|
RP-->>Browser: 10. 세션 생성 및 로그인 완료 안내
|
||||||
|
```
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. 핵심 구현 코드 (server.js)
|
||||||
|
|
||||||
|
데모 앱의 `server.js` 내 `resolveRedirects` 함수는 SSO 서버로부터 오는 리다이렉트 주소를 추적하다가, 동의 화면(`consent_challenge`)이 나타나면 이를 가로채서 처리합니다.
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
if (currentUrl.includes("/consent")) {
|
||||||
|
console.log(
|
||||||
|
" [자동 승인] 사용자의 정보 제공 동의 화면이 감지되어 시스템이 자동으로 승인 중입니다.",
|
||||||
|
);
|
||||||
|
|
||||||
|
// 1. URL에서 consent_challenge 추출
|
||||||
|
const consentUrl = new URL(currentUrl);
|
||||||
|
const consentChallenge = consentUrl.searchParams.get("consent_challenge");
|
||||||
|
|
||||||
|
// 2. SSO 서버에 현재 요청된 스코프가 무엇인지 확인
|
||||||
|
const detailsUrl = new URL("/api/v1/auth/consent", currentUrl);
|
||||||
|
detailsUrl.searchParams.set("consent_challenge", consentChallenge);
|
||||||
|
const detailsRes = await fetch(detailsUrl.toString(), {
|
||||||
|
headers: { Cookie: nextCookies },
|
||||||
|
});
|
||||||
|
const consentInfo = await detailsRes.json();
|
||||||
|
|
||||||
|
// 3. 확인된 스코프(openid, profile 등)를 모두 허용(accept)한다고 서버에 전송
|
||||||
|
const acceptUrl = new URL("/api/v1/auth/consent/accept", currentUrl);
|
||||||
|
const acceptRes = await fetch(acceptUrl.toString(), {
|
||||||
|
method: "POST",
|
||||||
|
headers: { "Content-Type": "application/json", Cookie: nextCookies },
|
||||||
|
body: JSON.stringify({
|
||||||
|
consent_challenge: consentChallenge,
|
||||||
|
grant_scope: consentInfo.requested_scope || ["openid", "profile"],
|
||||||
|
grant_access_token_audience:
|
||||||
|
consentInfo.requested_access_token_audience || [],
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
const acceptPayload = await acceptRes.json();
|
||||||
|
// 4. 승인 후 받은 최종 목적지(code 포함)로 이동 계속
|
||||||
|
return resolveRedirects(acceptPayload.redirectTo, nextCookies, depth + 1);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 4. Tenant 접근 제한 예외 흐름
|
||||||
|
|
||||||
|
Headless RP도 consent 단계에서 tenant 제한에 걸릴 수 있습니다. 이 경우는 일반적인 auto-consent 성공 흐름과 별도로 처리해야 합니다.
|
||||||
|
|
||||||
|
조건:
|
||||||
|
- RP metadata에 `tenant_access_restricted=true`
|
||||||
|
- 현재 사용자의 tenant가 `allowed_tenants`에 없음
|
||||||
|
|
||||||
|
이때 Baron SSO 응답:
|
||||||
|
- `GET /api/v1/auth/consent` 또는 `POST /api/v1/auth/consent/accept`
|
||||||
|
- HTTP `403`
|
||||||
|
- body에 `code=tenant_not_allowed`
|
||||||
|
|
||||||
|
데모 앱 처리 원칙:
|
||||||
|
- `redirectTo`가 없다고 가정하고 계속 진행하면 안 됩니다.
|
||||||
|
- `tenant_not_allowed`를 감지하면 `userfront` 에러 화면 URL로 변환해 브라우저를 이동시켜야 합니다.
|
||||||
|
- 기본 목적지는 `/ko/error?error=tenant_not_allowed&error_description=...&details=...` 입니다.
|
||||||
|
|
||||||
|
운영 관점 해석:
|
||||||
|
- backend 로그에 `tenant_not_allowed`가 보이면 Baron SSO 제한은 정상입니다.
|
||||||
|
- 브라우저에서 `Invalid URL`이 보이면 RP가 `403` 응답을 잘못 소비하고 있다는 뜻입니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 4. 특징 및 장점
|
||||||
|
|
||||||
|
- **심리스한 UX**: 사용자는 "로그인 중..."이라는 메시지만 보게 되며, 복잡한 권한 동의 절차를 신경 쓸 필요가 없습니다.
|
||||||
|
- **강력한 보안**: 자동 승인 과정은 서버 대 서버(Back-channel) 통신으로 이루어지며, 브라우저에는 동의 관련 데이터가 전혀 노출되지 않습니다.
|
||||||
|
- **유연한 확장**: 나중에 새로운 사용자 정보(`email` 등)가 필요해지더라도, 앱의 스코프 설정만 변경하면 자동으로 승인 로직에 반영됩니다.
|
||||||
95
baron-sso_login_guide/headless-employee-login-flow.md
Normal file
95
baron-sso_login_guide/headless-employee-login-flow.md
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
# Headless 사번 로그인 로직
|
||||||
|
|
||||||
|
이 문서는 사용자가 데모 애플리케이션에서 **사번과 비밀번호**를 이용해 로그인할 때 발생하는 내부 OIDC (OpenID Connect) Headless 인증 흐름을 설명합니다.
|
||||||
|
|
||||||
|
## 핵심 개념
|
||||||
|
|
||||||
|
- **Headless Login**: 사용자가 IdP(Identity Provider, Baron SSO)의 로그인 화면을 거치지 않고, RP(Relying Party, 데모 앱) 내에서 직접 아이디와 비밀번호를 입력받아 백채널(Back-channel)로 인증을 수행하는 방식입니다.
|
||||||
|
- **Client Assertion**: 보안을 위해 데모 앱은 사번/비밀번호를 전송할 때 자신이 신뢰할 수 있는 앱임을 증명하는 JWT(`client_assertion`)를 생성하여 함께 전송합니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. 사번 로그인 시퀀스 다이어그램
|
||||||
|
|
||||||
|
```mermaid
|
||||||
|
sequenceDiagram
|
||||||
|
autonumber
|
||||||
|
actor User as 사용자
|
||||||
|
participant RP as 데모 앱 (RP)
|
||||||
|
participant OIDC as Baron SSO (OIDC/Hydra)
|
||||||
|
participant Auth as Baron SSO (Headless API)
|
||||||
|
|
||||||
|
User->>RP: 1. 사번 / 비밀번호 입력 후 로그인 요청
|
||||||
|
|
||||||
|
note over RP: [1단계: 신호 요청]
|
||||||
|
RP->>OIDC: 2. OIDC Discovery 및 Authorization Request<br/>(GET /oauth2/auth?response_type=code...)
|
||||||
|
OIDC-->>RP: 3. login_challenge 발급 및 302 Redirect
|
||||||
|
|
||||||
|
note over RP: [2단계: 본인 인증]
|
||||||
|
RP->>RP: 4. Private Key로 client_assertion (JWT) 생성
|
||||||
|
RP->>Auth: 5. Headless 로그인 API 호출<br/>(POST /api/v1/auth/headless/password/login)<br/>[client_assertion, login_challenge, 사번, 비밀번호 포함]
|
||||||
|
Auth-->>RP: 6. 인증 성공 및 리다이렉트 URL 반환
|
||||||
|
|
||||||
|
note over RP: [3단계: 권한 획득]
|
||||||
|
RP->>OIDC: 7. 리다이렉트 URL 추적 (Cookie 유지)
|
||||||
|
OIDC-->>RP: 8. Consent 화면 (필요 시 자동 승인 API 호출)
|
||||||
|
OIDC-->>RP: 9. 최종 Authorization Code 발급 (code=...)
|
||||||
|
|
||||||
|
note over RP: [4단계: 인증키 발급]
|
||||||
|
RP->>OIDC: 10. Token Endpoint 호출 (Authorization Code Grant)<br/>[client_assertion 포함]
|
||||||
|
OIDC-->>RP: 11. id_token, access_token 발급
|
||||||
|
|
||||||
|
note over RP: [5단계: 로그인 완료]
|
||||||
|
RP->>RP: 12. id_token 검증 및 사용자 세션 생성
|
||||||
|
RP-->>User: 13. 홈 화면 리다이렉트 및 로그인 성공
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. 단계별 상세 로직 설명
|
||||||
|
|
||||||
|
위 다이어그램의 주요 단계를 데모 애플리케이션(`server.js`의 `runHeadlessSsoLogin` 함수) 로직을 중심으로 설명합니다.
|
||||||
|
|
||||||
|
### [1단계] 신호 요청 (Authorization Request)
|
||||||
|
- **목적**: SSO 서버와의 세션을 시작하고, 인증 컨텍스트를 식별할 수 있는 고유값(`login_challenge`)을 획득합니다.
|
||||||
|
- **동작**:
|
||||||
|
1. OIDC Discovery를 통해 엔드포인트들을 가져옵니다.
|
||||||
|
2. `response_type=code`, `client_id`, `state`, `nonce` 등을 포함하여 Authorization Endpoint(`/oauth2/auth`)로 `fetch` 요청(리다이렉트 수동 추적 모드)을 보냅니다.
|
||||||
|
3. 응답 헤더의 `Location`에 포함된 `login_challenge` 값을 추출합니다.
|
||||||
|
|
||||||
|
### [2단계] 본인 인증 (Headless Password Login)
|
||||||
|
- **목적**: 사용자가 입력한 자격 증명(사번, 비밀번호)을 SSO 서버에 직접 전달하여 인증을 승인받습니다.
|
||||||
|
- **동작**:
|
||||||
|
1. RP의 **Private Key(`keys.json`)**를 사용하여 `client_assertion` JWT를 서명합니다. (이때 `aud` 값은 SSO 서버의 Headless API URL이어야 합니다.)
|
||||||
|
2. `POST /api/v1/auth/headless/password/login` 엔드포인트로 JSON 페이로드를 전송합니다.
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"client_id": "...",
|
||||||
|
"login_challenge": "...",
|
||||||
|
"loginId": "사번",
|
||||||
|
"password": "비밀번호",
|
||||||
|
"client_assertion": "..."
|
||||||
|
}
|
||||||
|
```
|
||||||
|
3. 인증이 성공하면 서버는 OIDC 로그인 흐름을 계속할 수 있는 `redirectTo` 주소를 반환합니다.
|
||||||
|
|
||||||
|
### [3단계] 권한 획득 (Redirect Resolution & Consent)
|
||||||
|
- **목적**: 인증 성공 후, OIDC 프로토콜에 따라 최종적으로 `Authorization Code`를 획득합니다.
|
||||||
|
- **동작**:
|
||||||
|
1. 2단계에서 받은 `redirectTo` 주소를 따라가며 쿠키(Cookie)를 계속 유지시킵니다.
|
||||||
|
2. 만약 경로 중에 정보 제공 동의(`consent`) 화면이 감지되면, 데모 앱이 사용자 대신 백그라운드에서 동의 API(`POST /api/v1/auth/consent/accept`)를 호출하여 자동 승인합니다.
|
||||||
|
3. 최종적으로 URL에 `code=...`가 포함된 리다이렉트를 찾습니다.
|
||||||
|
|
||||||
|
### [4단계] 인증키 발급 (Token Exchange)
|
||||||
|
- **목적**: 획득한 `Authorization Code`를 안전한 토큰들로 교환합니다.
|
||||||
|
- **동작**:
|
||||||
|
1. Token Endpoint로 코드를 전송합니다.
|
||||||
|
2. 이때도 보안을 위해 `private_key_jwt` 방식이 사용되므로, 새로 생성한 `client_assertion`을 요청 본문에 포함시킵니다.
|
||||||
|
3. 검증이 완료되면 `access_token`과 사용자 정보가 담긴 `id_token`을 받게 됩니다.
|
||||||
|
|
||||||
|
### [5단계] 로그인 완료 (Session Creation)
|
||||||
|
- **목적**: 받은 토큰을 검증하고, 브라우저가 인식할 수 있는 로컬 세션을 만듭니다.
|
||||||
|
- **동작**:
|
||||||
|
1. `jose` 라이브러리를 통해 `id_token`의 서명을 검증하고 payload(사번, 이름, 부서 등)를 추출합니다.
|
||||||
|
2. 추출된 사용자 정보를 `express-session`의 `req.session.user`에 저장합니다.
|
||||||
|
3. 클라이언트 브라우저에게 세션 쿠키를 발급하며 홈 화면(`/home.html`)으로 리다이렉트 시킵니다.
|
||||||
84
baron-sso_login_guide/headless-phone-login-flow.md
Normal file
84
baron-sso_login_guide/headless-phone-login-flow.md
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
# Headless 전화번호 인증 로그인 로직
|
||||||
|
|
||||||
|
이 문서는 사용자가 데모 애플리케이션에서 **전화번호**를 입력하여 인증 링크를 받고, 모바일에서 승인하여 로그인하는 내부 흐름을 설명합니다.
|
||||||
|
|
||||||
|
## 핵심 개념
|
||||||
|
|
||||||
|
- **Link Init**: 사용자의 전화번호로 실제 인증 링크(카카오톡 또는 SMS)를 발송하도록 SSO 서버에 요청하는 단계입니다.
|
||||||
|
- **Polling**: 사용자가 모바일에서 링크를 클릭할 때까지 앱 서버가 주기적으로 SSO 서버에 "인증이 완료되었는지" 물어보는 과정입니다.
|
||||||
|
- **Security**: 이 과정에서도 모든 API 호출은 RP의 Private Key로 서명된 `client_assertion`을 포함하여 보안을 유지합니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. 전화번호 로그인 시퀀스 다이어그램
|
||||||
|
|
||||||
|
```mermaid
|
||||||
|
sequenceDiagram
|
||||||
|
autonumber
|
||||||
|
actor User as 사용자
|
||||||
|
participant RP as 데모 앱 (RP)
|
||||||
|
participant OIDC as Baron SSO (OIDC)
|
||||||
|
participant Auth as Baron SSO (Headless API)
|
||||||
|
|
||||||
|
User->>RP: 1. 전화번호 입력 후 인증 요청
|
||||||
|
|
||||||
|
note over RP: [1단계: 신호 요청]
|
||||||
|
RP->>OIDC: 2. Authorization Request (GET /oauth2/auth)<br/>[login_challenge 획득]
|
||||||
|
|
||||||
|
note over RP: [2단계: 링크 발송 요청]
|
||||||
|
RP->>Auth: 3. 인증 링크 발송 요청 (POST .../link/init)<br/>[client_assertion, login_challenge, 전화번호 포함]
|
||||||
|
Auth-->>User: 4. 인증 링크 발송 (카카오톡/SMS)
|
||||||
|
Auth-->>RP: 5. 대기 참조값(pendingRef) 반환
|
||||||
|
|
||||||
|
note over RP: [3단계: 인증 상태 폴링]
|
||||||
|
loop 인증 완료 시까지 반복 (최대 3분)
|
||||||
|
RP->>Auth: 6. 상태 확인 (POST .../link/poll)<br/>[client_assertion, pendingRef 포함]
|
||||||
|
alt 아직 대기 중
|
||||||
|
Auth-->>RP: 7. authorization_pending 응답
|
||||||
|
RP->>RP: 잠시 대기 (interval)
|
||||||
|
else 사용자 링크 클릭 완료
|
||||||
|
Auth-->>RP: 8. 인증 성공 및 리다이렉트 URL 반환
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
note over User: 사용자가 모바일에서 링크 클릭 및 승인
|
||||||
|
|
||||||
|
note over RP: [4단계: 이후 과정 (사번 로그인과 동일)]
|
||||||
|
RP->>OIDC: 9. 리다이렉트 추적 및 Authorization Code 획득
|
||||||
|
RP->>OIDC: 10. Token Endpoint 호출 및 id_token 발급
|
||||||
|
RP->>RP: 11. 세션 생성 및 홈 화면 이동
|
||||||
|
RP-->>User: 12. 로그인 완료 안내
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. 단계별 상세 로직 설명
|
||||||
|
|
||||||
|
### [1단계] 신호 요청 (Login Challenge)
|
||||||
|
- 사번 로그인과 동일하게 OIDC 표준 흐름을 시작하기 위해 `login_challenge`를 먼저 획득합니다. 이 값은 이후 인증 시도 시 "이 사용자가 어떤 OIDC 요청에 응답하고 있는지"를 SSO 서버에 알려주는 매개체가 됩니다.
|
||||||
|
|
||||||
|
### [2단계] 링크 발송 요청 (Link Init)
|
||||||
|
- **동작**: `POST /api/v1/auth/headless/link/init` 호출.
|
||||||
|
- **데이터**: `client_assertion`, `login_challenge`, 사용자의 `loginId`(전화번호).
|
||||||
|
- **결과**: SSO 서버는 사용자의 전화번호로 인증 가능한 고유 링크를 발송하고, RP에게는 해당 요청을 추적할 수 있는 `pendingRef`와 다음 폴링까지의 권장 대기 시간(`interval`)을 반환합니다.
|
||||||
|
|
||||||
|
### [3단계] 인증 상태 폴링 (Polling)
|
||||||
|
- **동작**: `POST /api/v1/auth/headless/link/poll`을 주기적으로 호출.
|
||||||
|
- **상태 처리**:
|
||||||
|
- `authorization_pending`: 사용자가 아직 링크를 클릭하지 않았으므로 지정된 시간만큼 기다린 후 다시 시도합니다.
|
||||||
|
- `slow_down`: 폴링 간격이 너무 잦다는 신호로, 대기 시간을 조금 더 늘립니다.
|
||||||
|
- `expired_token`: 3분이 경과하여 인증 요청이 만료된 경우입니다.
|
||||||
|
- **성공**: 사용자가 승인을 완료하면 서버는 `redirectTo` 정보를 반환하며 폴링이 종료됩니다.
|
||||||
|
|
||||||
|
### [4단계] 이후 과정 (Standard OIDC Flow)
|
||||||
|
- 폴링 결과로 받은 `redirectTo` 주소부터는 표준 OIDC 흐름을 따릅니다.
|
||||||
|
- 데모 앱은 해당 주소로 리다이렉트하며 발생하는 쿠키를 유지하고, 필요 시 사용자 동의(Consent) 과정을 자동 수행하여 최종적으로 `Authorization Code`를 얻어냅니다.
|
||||||
|
- 마지막으로 해당 코드를 `id_token`으로 교환하여 사용자 정보를 세션에 저장합니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. 특징 및 주의 사항
|
||||||
|
|
||||||
|
- **비동기 사용자 경험**: 사용자는 PC에서 전화번호만 입력하고, 실제 인증 행위는 모바일에서 수행합니다. PC 화면은 폴링이 완료될 때까지 "인증 대기 중" 상태를 유지합니다.
|
||||||
|
- **보안 검증**: 폴링 시에도 `client_assertion`이 필요합니다. 이는 제3자가 `pendingRef`를 가로채더라도 RP의 비밀키 없이는 인증 상태를 훔쳐볼 수 없음을 보장합니다.
|
||||||
|
- **제한 시간**: 보안상 폴링은 보통 3분(180초)으로 제한되며, 이후에는 요청을 처음부터 다시 시작해야 합니다.
|
||||||
115
baron-sso_login_guide/setup-guide.md
Normal file
115
baron-sso_login_guide/setup-guide.md
Normal file
@@ -0,0 +1,115 @@
|
|||||||
|
# Headless 로그인 데모 설정 가이드 (Baron SSO 연동)
|
||||||
|
|
||||||
|
이 문서는 `headless-login-demo` 프로젝트를 설치하고, **Baron SSO (OIDC IdP)** 관리자 도구(`devfront`)에서 **Headless RP**를 올바르게 생성 및 설정하는 방법을 단계별로 설명합니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. Baron SSO (devfront) 설정
|
||||||
|
|
||||||
|
Headless RP는 일반 OIDC 클라이언트와 설정 방식이 다릅니다. 특히 **IdP 서버가 RP(데모 앱)의 JWKS 엔드포인트에 접속할 수 있어야 함**을 유의하십시오.
|
||||||
|
|
||||||
|
### Step 1: 클라이언트 기본 정보 입력
|
||||||
|
|
||||||
|
1. `devfront` 접속 후 **연동 앱 > 연동 앱 추가**를 클릭합니다.
|
||||||
|
2. **Name**: `headless-login` (또는 자유롭게 입력)
|
||||||
|
3. **Redirect URIs**: 데모 앱에 접속할 주소의 콜백 경로를 입력합니다.
|
||||||
|
- **통합 환경 (IdP와 RP가 같은 서버일 때)**: `http://localhost:3000/callback`
|
||||||
|
- **분리 환경 (IdP는 원격 서버, RP는 내 PC일 때)**: `http://[내_PC_IP]:3000/callback`
|
||||||
|
- _주의: 실제 앱에서 요청하는 Redirect URI와 여기서 등록한 값이 **완전히 일치**해야 인증 거부 에러가 발생하지 않습니다._
|
||||||
|
4. **Scopes**: `openid`, `profile`, `email`을 선택합니다.
|
||||||
|
5. **Type**: 반드시 `pkce`를 선택해야 합니다.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
### Step 2: Headless 기능 및 보안 설정
|
||||||
|
|
||||||
|
1. `pkce` 하단의 **"Headless Login (자체 로그인 UI 사용)"** 토글을 활성화합니다.
|
||||||
|
2. **JWKS URI**: **Baron SSO 서버에서 접근 가능한** 데모 앱의 공개키 주소를 입력합니다.
|
||||||
|
- **통합 환경 (IdP와 RP가 같은 서버일 때)**: `http://localhost:3000/.well-known/jwks.json`
|
||||||
|
- **분리 환경 (IdP는 원격 서버, RP는 내 PC일 때)**: `http://[내_PC_IP]:3000/.well-known/jwks.json`
|
||||||
|
- _주의: Baron SSO 서버(`172.16.10.175`)에서 사용자님의 로컬 PC 포트로 네트워크가 열려있어야 합니다._
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
### Step 3: 저장 및 확인 (JWKS 캐시 검증)
|
||||||
|
|
||||||
|
1. **앱 생성** 버튼을 클릭해 저장합니다.
|
||||||
|
2. 연동 앱 목록에서 해당 앱이 **PKCE (Headless Login)** 유형으로 생성됐는지 확인합니다.
|
||||||
|
3. 앱 상세 페이지 하단이나 설정 탭에서 **JWKS 캐시 상태**가 `Success`인지 반드시 확인합니다.
|
||||||
|
- _팁: 캐시 상태가 비어있거나 실패인 경우, 데모 앱(내 PC)이 켜져 있는지 확인하고 **새로고침(Refresh)** 버튼을 눌러 수동으로 캐시를 갱신하세요. 캐싱이 정상적으로 이루어져야만 로그인이 작동합니다._
|
||||||
|
|
||||||
|
## 
|
||||||
|
|
||||||
|
## 2. Headless Login 애플리케이션 로컬 설정 (.env)
|
||||||
|
|
||||||
|
`devfront`에서 설정한 값을 바탕으로 프로젝트 루트의 `.env` 파일을 구성합니다.
|
||||||
|
|
||||||
|
```env
|
||||||
|
# 애플리케이션 실행 포트
|
||||||
|
PORT=3000
|
||||||
|
|
||||||
|
# OIDC IdP 설정 (원격 서버 주소 입력)
|
||||||
|
CLIENT_ID=a9b64539-7242-4aa5-ad3d-13c7f1ef00f2
|
||||||
|
ISSUER=http://172.16.10.175/oidc
|
||||||
|
|
||||||
|
# 리다이렉트 및 보안 설정
|
||||||
|
# REDIRECT_URI는 DevFront에 등록한 Redirect URIs 중 하나와 정확히 일치해야 합니다.
|
||||||
|
REDIRECT_URI=http://[내_PC_IP]:3000/callback
|
||||||
|
# JWKS_URI는 Baron SSO 서버가 내 PC로 접속할 때 사용하는 주소와 일치해야 합니다.
|
||||||
|
JWKS_URI=http://[내_PC_IP]:3000/.well-known/jwks.json
|
||||||
|
|
||||||
|
# tenant 제한 시 이동시킬 userfront 에러 경로
|
||||||
|
ERROR_LOCALE_PATH=/ko/error
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. 주의 사항 (네트워크 구성)
|
||||||
|
|
||||||
|
- **서버 간 통신 (Server-to-Server)**: Headless 로그인은 Baron SSO 서버가 직접 데모 앱의 `JWKS_URI`에 접속하여 서명을 검증합니다. 따라서 IdP 서버에서 내 로컬 PC의 포트(`3000`)로의 인바운드 통신이 허용되어야 합니다.
|
||||||
|
- **Redirect URI 일치**: 브라우저를 통해 접속할 주소가 IP라면, Redirect URI와 JWKS URI 모두 IP 기반 주소로 통일하는 것이 문제 발생 소지를 줄이는 가장 좋은 방법입니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 4. 실행
|
||||||
|
|
||||||
|
```bash
|
||||||
|
npm install
|
||||||
|
npm start
|
||||||
|
```
|
||||||
|
|
||||||
|
> **💡 참고: 자동 키 생성 및 JWKS 엔드포인트 활성화**
|
||||||
|
> `npm start` 명령어로 서버를 실행하면, 프로젝트 내에 `keys.json` 파일이 자동으로 생성됩니다. 이 파일에는 서버가 자체적으로 발급한 RSA 보안 키 쌍(공개키/개인키)이 저장됩니다.
|
||||||
|
> 동시에, `.env`에 설정된 `JWKS_URI` 경로(또는 기본 경로)로 공개키가 서빙되어 Baron SSO가 검증에 사용할 수 있게 됩니다.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 5. Tenant 제한 테스트 가이드
|
||||||
|
|
||||||
|
Headless RP는 login API 성공 이후 consent 단계에서 tenant 제한이 걸릴 수 있습니다. 따라서 단순히 `/api/v1/auth/headless/password/login` 성공만 보면 안 되고, 최종 redirect 해석까지 확인해야 합니다.
|
||||||
|
|
||||||
|
### 권장 검증 순서
|
||||||
|
|
||||||
|
1. 제한이 없는 상태에서 로그인 성공 여부를 먼저 확인합니다.
|
||||||
|
2. `devfront`에서 대상 RP에 `tenant_access_restricted`와 `allowed_tenants`를 설정합니다.
|
||||||
|
3. 허용된 tenant 사용자로 로그인해 정상 성공 여부를 확인합니다.
|
||||||
|
4. 허용되지 않은 tenant 사용자로 로그인해 `userfront`의 `/ko/error?error=tenant_not_allowed...` 화면으로 이동하는지 확인합니다.
|
||||||
|
|
||||||
|
### 기대 동작
|
||||||
|
|
||||||
|
- `tenant_not_allowed`는 Baron SSO backend가 `403`으로 반환합니다.
|
||||||
|
- headless demo는 이 응답을 감지해 `userfront` 에러 화면으로 브라우저를 이동시킵니다.
|
||||||
|
- 따라서 브라우저 팝업으로 `Invalid URL`이 보인다면 demo 쪽 에러 처리 누락을 먼저 의심해야 합니다.
|
||||||
|
|
||||||
|
### 로그 확인 포인트
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker logs --tail 100 headless-login-demo
|
||||||
|
docker logs --tail 100 baron_backend
|
||||||
|
```
|
||||||
|
|
||||||
|
- `headless-login-demo`
|
||||||
|
- consent 단계에서 `tenant_not_allowed`를 받아 `errorUrl`로 변환했는지 확인
|
||||||
|
- `baron_backend`
|
||||||
|
- `GET /api/v1/auth/consent` 또는 `POST /api/v1/auth/consent/accept`가 `403`인지 확인
|
||||||
|
- 해당 응답 body에 `code=tenant_not_allowed`가 포함되는지 확인
|
||||||
BIN
baron-sso_login_guide/setup-step1-example.png
Normal file
BIN
baron-sso_login_guide/setup-step1-example.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 191 KiB |
BIN
baron-sso_login_guide/setup-step2-example.png
Normal file
BIN
baron-sso_login_guide/setup-step2-example.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 186 KiB |
BIN
baron-sso_login_guide/setup-step3-example.png
Normal file
BIN
baron-sso_login_guide/setup-step3-example.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 95 KiB |
940
doc_readme.md
940
doc_readme.md
@@ -1,940 +0,0 @@
|
|||||||
# ITAM 도커라이징 실전 가이드
|
|
||||||
|
|
||||||
## 1. 문서 목적
|
|
||||||
|
|
||||||
이 문서는 Gitea에 올라가 있는 현재 저장소를 기준으로, 개발 PC에 WSL2와 Ubuntu만 설치되어 있는 상태에서 지금의 Docker 실행 구조를 재현하는 방법을 처음부터 끝까지 설명하는 실전 가이드다.
|
|
||||||
|
|
||||||
이 문서는 아래 상황을 가정한다.
|
|
||||||
|
|
||||||
1. 소스 코드는 아직 로컬에 없거나, Gitea에서 막 받아올 예정이다.
|
|
||||||
2. Windows에는 WSL2와 Ubuntu는 설치되어 있다.
|
|
||||||
3. 그 외 Docker 관련 세팅은 아직 안 되어 있을 수 있다.
|
|
||||||
4. 최종 목표는 현재 저장소 기준 `frontend + backend + external DB` 구조를 Docker로 재현하는 것이다.
|
|
||||||
|
|
||||||
이 문서의 목적은 아래 네 가지다.
|
|
||||||
|
|
||||||
1. 현재 시스템 구조와 Docker 구조를 먼저 이해하게 한다.
|
|
||||||
2. 기존 파일 중 무엇이 새로 추가되었고 무엇이 수정되었는지 정리한다.
|
|
||||||
3. 각 단계별로 정확히 어디에서 명령을 실행해야 하는지 명시한다.
|
|
||||||
4. Gitea 소스만 받은 상태에서 지금과 같은 Docker 실행 상태까지 도달하게 한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 2. 현재 시스템 구조 개요
|
|
||||||
|
|
||||||
## 2.1 애플리케이션 원래 구조
|
|
||||||
|
|
||||||
현재 저장소의 본래 실행 구조는 다음과 같다.
|
|
||||||
|
|
||||||
1. 프런트엔드: Vite 기반 TypeScript 앱
|
|
||||||
2. 백엔드: Express 기반 Node.js API 서버
|
|
||||||
3. 데이터베이스: 외부 MySQL 서버
|
|
||||||
|
|
||||||
즉, 원래부터 MySQL이 Docker 안에 들어 있던 구조가 아니다.
|
|
||||||
|
|
||||||
프런트와 백엔드는 각각 별도 프로세스로 실행되며, 프런트는 `/api` 상대 경로로 백엔드 API를 호출한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 2.2 현재 Docker 구조
|
|
||||||
|
|
||||||
현재 최종 Docker 구조는 아래와 같다.
|
|
||||||
|
|
||||||
1. `frontend` 컨테이너
|
|
||||||
2. `backend` 컨테이너
|
|
||||||
3. 외부 MySQL DB
|
|
||||||
|
|
||||||
즉, 지금은 내부 `db` 컨테이너가 없고, 내부 `db-bootstrap` 컨테이너도 없다.
|
|
||||||
|
|
||||||
현재 구조를 문장으로 풀면 다음과 같다.
|
|
||||||
|
|
||||||
1. 브라우저는 `http://localhost:8080`으로 `frontend` 컨테이너에 접속한다.
|
|
||||||
2. `frontend`는 `/api` 요청을 `backend:3000`으로 프록시한다.
|
|
||||||
3. `backend`는 `.env`에 적힌 외부 DB 정보로 외부 MySQL에 직접 접속한다.
|
|
||||||
4. 조회 결과 JSON을 프런트가 받아 화면에 렌더링한다.
|
|
||||||
|
|
||||||
간단한 흐름은 아래와 같다.
|
|
||||||
|
|
||||||
```text
|
|
||||||
Browser
|
|
||||||
-> frontend container :8080
|
|
||||||
-> Vite proxy (/api)
|
|
||||||
-> backend container :3000
|
|
||||||
-> external MySQL (.env)
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 2.3 왜 이 구조가 맞는가
|
|
||||||
|
|
||||||
현재 구조가 적절한 이유는 다음과 같다.
|
|
||||||
|
|
||||||
1. 원래 시스템도 외부 MySQL을 쓰는 구조였다.
|
|
||||||
2. 지금 목표는 운영형 단일 배포가 아니라 현재 개발형 구조를 Docker로 재현하는 것이다.
|
|
||||||
3. 프런트는 Vite dev server 기반이라 운영형 nginx 정적 배포 구조로 억지로 바꾸는 것보다, 현 구조를 유지하는 편이 안전하다.
|
|
||||||
4. 실무 표준 관점에서도 앱 컨테이너는 무상태로 유지하고, DB는 외부 인프라를 사용하는 구성이 더 일반적이다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 3. 이번 도커라이징에서 추가되거나 수정된 파일 정리
|
|
||||||
|
|
||||||
아래 파일들은 이번 Docker 재현 구조를 위해 새로 추가되었거나 수정된 핵심 파일이다.
|
|
||||||
|
|
||||||
## 3.1 새로 추가된 파일
|
|
||||||
|
|
||||||
1. `Dockerfile.frontend`
|
|
||||||
2. `Dockerfile.backend`
|
|
||||||
3. `.dockerignore`
|
|
||||||
4. `docker-compose.yaml`
|
|
||||||
5. `start_docker_wsl.ps1`
|
|
||||||
6. `stop_docker_wsl.ps1`
|
|
||||||
7. `start_docker_wsl.bat`
|
|
||||||
8. `stop_docker_wsl.bat`
|
|
||||||
9. `docker/mysql/init/README.md`
|
|
||||||
10. `docker_task_plan.md`
|
|
||||||
11. `doc_readme2.md`
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 3.2 기존 파일 중 수정된 핵심 파일
|
|
||||||
|
|
||||||
1. `server.js`
|
|
||||||
2. `vite.config.ts`
|
|
||||||
3. `doc_readme.md`
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 3.3 각 파일의 역할
|
|
||||||
|
|
||||||
### `Dockerfile.frontend`
|
|
||||||
|
|
||||||
역할:
|
|
||||||
|
|
||||||
1. 프런트 Vite 개발 서버 이미지를 만든다.
|
|
||||||
2. 컨테이너 내부에서 `npm run dev -- --host 0.0.0.0`를 실행한다.
|
|
||||||
|
|
||||||
### `Dockerfile.backend`
|
|
||||||
|
|
||||||
역할:
|
|
||||||
|
|
||||||
1. 백엔드 Express 서버 이미지를 만든다.
|
|
||||||
2. 컨테이너 내부에서 `npm run server`를 실행한다.
|
|
||||||
|
|
||||||
### `.dockerignore`
|
|
||||||
|
|
||||||
역할:
|
|
||||||
|
|
||||||
1. `node_modules`, `build`, `.git`, `.env`, `uploads` 같은 불필요한 파일을 Docker build context에서 제외한다.
|
|
||||||
|
|
||||||
### `docker-compose.yaml`
|
|
||||||
|
|
||||||
역할:
|
|
||||||
|
|
||||||
1. `frontend`, `backend` 두 컨테이너를 동시에 띄운다.
|
|
||||||
2. `backend`는 `.env`의 외부 DB를 사용한다.
|
|
||||||
3. `frontend`는 `backend:3000`으로 프록시한다.
|
|
||||||
|
|
||||||
### `start_docker_wsl.ps1`
|
|
||||||
|
|
||||||
역할:
|
|
||||||
|
|
||||||
1. Windows 경로를 WSL 경로로 안전하게 바꾼다.
|
|
||||||
2. WSL 내부 Docker를 사용해 `docker compose up --build -d`를 실행한다.
|
|
||||||
3. 한글 경로와 공백 경로에서도 안정적으로 실행되게 한다.
|
|
||||||
|
|
||||||
### `stop_docker_wsl.ps1`
|
|
||||||
|
|
||||||
역할:
|
|
||||||
|
|
||||||
1. 같은 방식으로 WSL 내부에서 `docker compose down`을 실행한다.
|
|
||||||
|
|
||||||
### `start_docker_wsl.bat`, `stop_docker_wsl.bat`
|
|
||||||
|
|
||||||
역할:
|
|
||||||
|
|
||||||
1. PowerShell 스크립트를 쉽게 실행하는 래퍼 역할을 한다.
|
|
||||||
|
|
||||||
### `server.js`
|
|
||||||
|
|
||||||
중요 수정 사항:
|
|
||||||
|
|
||||||
1. `dotenv.config({ override: true })`가 아니라 `dotenv.config()`를 사용한다.
|
|
||||||
|
|
||||||
이유:
|
|
||||||
|
|
||||||
1. Compose나 실행 환경이 주는 환경변수를 `.env`가 덮어써 버리면 안 된다.
|
|
||||||
2. 외부 DB 정보와 포트 설정 등 실행 환경 우선 구조를 유지해야 한다.
|
|
||||||
|
|
||||||
### `vite.config.ts`
|
|
||||||
|
|
||||||
중요 수정 사항:
|
|
||||||
|
|
||||||
1. 프록시 타깃을 고정 `localhost:3000`이 아니라 환경변수 기반으로 받도록 바꿨다.
|
|
||||||
|
|
||||||
현재 구조:
|
|
||||||
|
|
||||||
```ts
|
|
||||||
const proxyTarget = process.env.VITE_DEV_PROXY_TARGET || 'http://localhost:3000';
|
|
||||||
```
|
|
||||||
|
|
||||||
이유:
|
|
||||||
|
|
||||||
1. 로컬에서 직접 프런트를 띄울 때는 `localhost:3000`이 맞다.
|
|
||||||
2. Docker 안에서는 `frontend` 컨테이너에서 보는 `localhost`가 백엔드가 아니므로 `backend:3000`을 써야 한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 4. 현재 `docker-compose.yaml` 기준 실제 동작 구조
|
|
||||||
|
|
||||||
현재 `docker-compose.yaml`은 아래 구조다.
|
|
||||||
|
|
||||||
### `backend`
|
|
||||||
|
|
||||||
1. `Dockerfile.backend`로 이미지를 빌드한다.
|
|
||||||
2. `.env`를 읽는다.
|
|
||||||
3. DB 관련 변수는 `${DB_HOST}`, `${DB_PORT}`, `${DB_USER}`, `${DB_PASS}`, `${DB_NAME}`를 그대로 사용한다.
|
|
||||||
4. 포트 `3000:3000`으로 노출한다.
|
|
||||||
5. `uploads`, `map_config.json`을 마운트한다.
|
|
||||||
|
|
||||||
### `frontend`
|
|
||||||
|
|
||||||
1. `Dockerfile.frontend`로 이미지를 빌드한다.
|
|
||||||
2. `VITE_DEV_PROXY_TARGET: http://backend:3000` 환경변수를 사용한다.
|
|
||||||
3. 포트 `8080:8080`으로 노출한다.
|
|
||||||
4. 브라우저의 `/api` 요청을 `backend`로 프록시한다.
|
|
||||||
|
|
||||||
즉, 현재 Compose는 DB를 띄우지 않고 앱 두 개만 띄운다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 5. 사전 준비 사항
|
|
||||||
|
|
||||||
이 섹션은 Gitea에서 코드를 받기 전 또는 받은 직후에 확인해야 한다.
|
|
||||||
|
|
||||||
## 5.1 가정하는 기본 상태
|
|
||||||
|
|
||||||
이미 설치되어 있다고 가정하는 것:
|
|
||||||
|
|
||||||
1. Windows
|
|
||||||
2. WSL2
|
|
||||||
3. Ubuntu 배포판
|
|
||||||
|
|
||||||
아직 없을 수 있는 것:
|
|
||||||
|
|
||||||
1. Docker Desktop 또는 WSL 내부 Docker 사용 환경
|
|
||||||
2. Git 클라이언트
|
|
||||||
3. 프로젝트 `.env`
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 5.2 권장 Docker 실행 방식
|
|
||||||
|
|
||||||
현재 저장소 구조상 가장 권장하는 방식은 다음이다.
|
|
||||||
|
|
||||||
1. Windows에 Docker Desktop 설치
|
|
||||||
2. Docker Desktop에서 WSL2 통합 활성화
|
|
||||||
3. Ubuntu WSL 내부에서 `docker` 명령을 사용할 수 있게 한다.
|
|
||||||
|
|
||||||
이유:
|
|
||||||
|
|
||||||
1. 현재 `start_docker_wsl.ps1`가 WSL 내부의 `docker`를 호출하는 구조다.
|
|
||||||
2. 실제 검증도 WSL 내부 Docker 기준으로 이루어졌다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 5.3 외부 DB 정보 준비
|
|
||||||
|
|
||||||
현재 구조는 외부 MySQL을 사용하므로 `.env` 파일이 반드시 필요하다.
|
|
||||||
|
|
||||||
최소한 아래 값이 필요하다.
|
|
||||||
|
|
||||||
```env
|
|
||||||
DB_HOST=<외부 MySQL 호스트>
|
|
||||||
DB_PORT=3306
|
|
||||||
DB_USER=<외부 MySQL 계정>
|
|
||||||
DB_PASS=<외부 MySQL 비밀번호>
|
|
||||||
DB_NAME=itam
|
|
||||||
```
|
|
||||||
|
|
||||||
필요 시 추가 환경변수는 현재 백엔드 코드 기준으로 함께 넣을 수 있다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 6. Gitea에서 소스 받기
|
|
||||||
|
|
||||||
## 6.1 작업 실행 위치
|
|
||||||
|
|
||||||
이 단계는 **Windows PowerShell** 또는 **Windows 터미널의 PowerShell**에서 수행한다.
|
|
||||||
|
|
||||||
실행 위치 이유:
|
|
||||||
|
|
||||||
1. 이후 `start_docker_wsl.ps1`도 Windows PowerShell에서 실행하는 것이 가장 자연스럽다.
|
|
||||||
2. 로컬 작업 폴더를 Windows 경로 기준으로 준비할 수 있다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 6.2 소스 클론
|
|
||||||
|
|
||||||
예시:
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
git clone <Gitea 저장소 URL>
|
|
||||||
cd <클론된 저장소 경로>
|
|
||||||
```
|
|
||||||
|
|
||||||
현재 프로젝트처럼 한글 경로를 사용할 수도 있지만, 가능하면 너무 복잡한 경로는 피하는 것이 좋다.
|
|
||||||
|
|
||||||
현재 실제 프로젝트 경로 예시는 아래였다.
|
|
||||||
|
|
||||||
```text
|
|
||||||
c:\Users\user\Desktop\안건 파일\itam
|
|
||||||
```
|
|
||||||
|
|
||||||
이 경로도 현재 스크립트로는 동작 가능하다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 7. Docker 환경 준비
|
|
||||||
|
|
||||||
## 7.1 작업 실행 위치
|
|
||||||
|
|
||||||
이 단계는 **Windows PowerShell**과 **WSL Ubuntu 터미널**을 둘 다 사용한다.
|
|
||||||
|
|
||||||
1. 설치 확인은 Windows PowerShell에서 시작
|
|
||||||
2. 실제 Docker 동작 확인은 WSL Ubuntu에서 수행
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 7.2 Docker Desktop 설치 여부 확인
|
|
||||||
|
|
||||||
**실행 위치: Windows PowerShell**
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
docker version
|
|
||||||
```
|
|
||||||
|
|
||||||
만약 여기서 바로 안 잡혀도 현재 프로젝트는 WSL 내부 Docker를 쓰므로, 다음 단계로 넘어가 WSL 내부 확인을 한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 7.3 WSL 내부 Docker 확인
|
|
||||||
|
|
||||||
**실행 위치: Windows PowerShell**
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
wsl -l -v
|
|
||||||
wsl sh -lc "docker --version"
|
|
||||||
```
|
|
||||||
|
|
||||||
정상 기대 결과:
|
|
||||||
|
|
||||||
1. Ubuntu가 Running 상태
|
|
||||||
2. `docker --version`이 정상 출력
|
|
||||||
|
|
||||||
만약 `docker --version`이 실패하면, Docker Desktop 설치 및 WSL 통합을 먼저 완료해야 한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 8. `.env` 파일 준비
|
|
||||||
|
|
||||||
## 8.1 작업 실행 위치
|
|
||||||
|
|
||||||
이 단계는 **Windows PowerShell**, **VS Code**, 또는 아무 텍스트 편집기**에서 수행한다.
|
|
||||||
|
|
||||||
즉, 프로젝트 루트에 `.env` 파일을 만드는 작업이다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 8.2 `.env` 작성
|
|
||||||
|
|
||||||
프로젝트 루트에 `.env`를 만든다.
|
|
||||||
|
|
||||||
예시:
|
|
||||||
|
|
||||||
```env
|
|
||||||
DB_HOST=your-external-db-host
|
|
||||||
DB_PORT=3306
|
|
||||||
DB_USER=your-db-user
|
|
||||||
DB_PASS=your-db-password
|
|
||||||
DB_NAME=itam
|
|
||||||
```
|
|
||||||
|
|
||||||
주의:
|
|
||||||
|
|
||||||
1. 현재 Compose는 내부 DB를 만들지 않는다.
|
|
||||||
2. 따라서 이 값이 곧 실제 운영/개발 외부 DB 연결 정보다.
|
|
||||||
3. 이 정보가 틀리면 `backend`는 기동해도 API에서 DB 오류가 난다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 9. 현재 Docker 파일이 어떻게 동작하는지 이해하기
|
|
||||||
|
|
||||||
## 9.1 `Dockerfile.frontend`
|
|
||||||
|
|
||||||
**확인 위치: 프로젝트 루트 / VS Code**
|
|
||||||
|
|
||||||
현재 내용 핵심:
|
|
||||||
|
|
||||||
```dockerfile
|
|
||||||
FROM node:20-alpine
|
|
||||||
WORKDIR /app
|
|
||||||
COPY package*.json ./
|
|
||||||
RUN npm ci
|
|
||||||
COPY . .
|
|
||||||
EXPOSE 8080
|
|
||||||
CMD ["npm", "run", "dev", "--", "--host", "0.0.0.0"]
|
|
||||||
```
|
|
||||||
|
|
||||||
의미:
|
|
||||||
|
|
||||||
1. Node 20 Alpine 기반
|
|
||||||
2. 의존성 설치 후 전체 소스 복사
|
|
||||||
3. Vite 개발 서버 실행
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 9.2 `Dockerfile.backend`
|
|
||||||
|
|
||||||
**확인 위치: 프로젝트 루트 / VS Code**
|
|
||||||
|
|
||||||
현재 내용 핵심:
|
|
||||||
|
|
||||||
```dockerfile
|
|
||||||
FROM node:20-alpine
|
|
||||||
WORKDIR /app
|
|
||||||
COPY package*.json ./
|
|
||||||
RUN npm ci
|
|
||||||
COPY . .
|
|
||||||
EXPOSE 3000
|
|
||||||
CMD ["npm", "run", "server"]
|
|
||||||
```
|
|
||||||
|
|
||||||
의미:
|
|
||||||
|
|
||||||
1. Node 20 Alpine 기반
|
|
||||||
2. Express 서버 실행
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 9.3 `vite.config.ts`
|
|
||||||
|
|
||||||
**확인 위치: 프로젝트 루트 / VS Code**
|
|
||||||
|
|
||||||
현재 핵심:
|
|
||||||
|
|
||||||
```ts
|
|
||||||
const proxyTarget = process.env.VITE_DEV_PROXY_TARGET || 'http://localhost:3000';
|
|
||||||
```
|
|
||||||
|
|
||||||
그리고 `/api`, `/uploads`가 모두 `proxyTarget`으로 프록시된다.
|
|
||||||
|
|
||||||
의미:
|
|
||||||
|
|
||||||
1. 로컬 실행 시 기본값은 `localhost:3000`
|
|
||||||
2. Docker 실행 시 Compose가 `http://backend:3000`을 주입
|
|
||||||
|
|
||||||
이 수정이 있어야 Docker 안에서도 화면에 데이터가 표시된다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 10. Docker Compose 기동
|
|
||||||
|
|
||||||
## 10.1 작업 실행 위치
|
|
||||||
|
|
||||||
이 단계는 반드시 **Windows PowerShell**에서 수행하는 것을 권장한다.
|
|
||||||
|
|
||||||
이유:
|
|
||||||
|
|
||||||
1. `start_docker_wsl.ps1`가 Windows 경로를 받아 WSL 경로로 바꾸는 구조다.
|
|
||||||
2. 한글/공백 경로에서 가장 안전하다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 10.2 권장 기동 방법
|
|
||||||
|
|
||||||
**실행 위치: 프로젝트 루트의 Windows PowerShell**
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
.\start_docker_wsl.ps1
|
|
||||||
```
|
|
||||||
|
|
||||||
또는
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
.\start_docker_wsl.bat
|
|
||||||
```
|
|
||||||
|
|
||||||
이 스크립트는 내부적으로 아래를 수행한다.
|
|
||||||
|
|
||||||
1. PowerShell 출력 인코딩을 UTF-8로 설정
|
|
||||||
2. 현재 Windows 경로를 WSL 경로로 변환
|
|
||||||
3. WSL 동작 확인
|
|
||||||
4. WSL 내부 Docker 동작 확인
|
|
||||||
5. `docker compose up --build -d` 수행
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 10.3 직접 기동이 필요할 때
|
|
||||||
|
|
||||||
**실행 위치: WSL Ubuntu 터미널**
|
|
||||||
|
|
||||||
직접 실행 예시는 아래와 같다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
cd /mnt/c/Users/user/Desktop/안건\ 파일/itam
|
|
||||||
docker compose up --build -d
|
|
||||||
```
|
|
||||||
|
|
||||||
하지만 현재 프로젝트는 한글 경로 이슈가 있었기 때문에, 특별한 이유가 없으면 `start_docker_wsl.ps1`를 우선 사용한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 11. 컨테이너 기동 후 검증
|
|
||||||
|
|
||||||
## 11.1 컨테이너 상태 확인
|
|
||||||
|
|
||||||
**실행 위치: Windows PowerShell**
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
wsl sh -lc "docker ps -a --format 'table {{.Names}}\t{{.Status}}' | grep itam"
|
|
||||||
```
|
|
||||||
|
|
||||||
정상 기대 상태:
|
|
||||||
|
|
||||||
1. `itam-backend` -> `Up`
|
|
||||||
2. `itam-frontend` -> `Up`
|
|
||||||
|
|
||||||
현재는 `itam-db`, `itam-db-bootstrap`가 없어야 정상이다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 11.2 백엔드 API 확인
|
|
||||||
|
|
||||||
**실행 위치: Windows PowerShell**
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
Invoke-WebRequest -Uri http://localhost:3000/api/assets/master -UseBasicParsing | Select-Object -ExpandProperty StatusCode
|
|
||||||
```
|
|
||||||
|
|
||||||
정상 기대값:
|
|
||||||
|
|
||||||
1. `200`
|
|
||||||
|
|
||||||
이 검사는 `backend`가 외부 DB에 정상 연결됐는지 보는 가장 직접적인 검사다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 11.3 프런트 경유 API 확인
|
|
||||||
|
|
||||||
**실행 위치: Windows PowerShell**
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
Invoke-WebRequest -Uri http://localhost:8080/api/assets/master -UseBasicParsing | Select-Object -ExpandProperty StatusCode
|
|
||||||
```
|
|
||||||
|
|
||||||
정상 기대값:
|
|
||||||
|
|
||||||
1. `200`
|
|
||||||
|
|
||||||
이 검사는 프런트 프록시가 정상인지 확인한다.
|
|
||||||
|
|
||||||
예전에 화면에 데이터가 안 보였던 것은 외부 DB 자체가 아니라, 이 프록시 경로가 잘못돼 있었기 때문이다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 11.4 브라우저 화면 확인
|
|
||||||
|
|
||||||
**실행 위치: 브라우저**
|
|
||||||
|
|
||||||
```text
|
|
||||||
http://localhost:8080
|
|
||||||
```
|
|
||||||
|
|
||||||
확인 포인트:
|
|
||||||
|
|
||||||
1. 화면이 열리는지
|
|
||||||
2. 목록/대시보드/테이블 데이터가 비어 있지 않은지
|
|
||||||
3. 모달 진입 시 데이터가 정상적으로 보이는지
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 12. 지금 데이터가 표시되는 원리
|
|
||||||
|
|
||||||
현재는 내부 DB로 데이터를 옮겨 담지 않는다.
|
|
||||||
|
|
||||||
현재 실제 동작 원리는 다음과 같다.
|
|
||||||
|
|
||||||
1. 브라우저가 `frontend`에 접속한다.
|
|
||||||
2. 프런트가 `/api/...`로 요청한다.
|
|
||||||
3. Vite 프록시가 `backend:3000`으로 요청을 넘긴다.
|
|
||||||
4. `backend`가 `.env`의 외부 MySQL에 직접 접속한다.
|
|
||||||
5. 조회 결과 JSON을 프런트가 받아 화면에 렌더링한다.
|
|
||||||
|
|
||||||
즉, 현재는 아래 구조다.
|
|
||||||
|
|
||||||
```text
|
|
||||||
Browser -> frontend -> backend -> external MySQL
|
|
||||||
```
|
|
||||||
|
|
||||||
예전 외부 DB 구조에서 화면에 데이터가 안 보였던 이유는 외부 DB 때문이 아니라, 프런트 컨테이너가 `localhost:3000`을 잘못 바라보고 있었기 때문이다.
|
|
||||||
|
|
||||||
지금은 `VITE_DEV_PROXY_TARGET: http://backend:3000`으로 수정되어 있기 때문에 정상 표시된다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 13. 자주 헷갈리는 포인트
|
|
||||||
|
|
||||||
## 13.1 현재는 내부 DB 컨테이너가 없다
|
|
||||||
|
|
||||||
현재 `docker-compose.yaml`에는 아래가 없다.
|
|
||||||
|
|
||||||
1. `db` 서비스
|
|
||||||
2. `db-bootstrap` 서비스
|
|
||||||
3. `itam_mysql_data` 볼륨
|
|
||||||
|
|
||||||
즉, DB는 Docker 스택 밖에 있다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 13.2 현재는 `.env`가 곧 실제 DB 연결 정보다
|
|
||||||
|
|
||||||
현재 `backend`는 아래처럼 Compose에서 그대로 받는다.
|
|
||||||
|
|
||||||
1. `DB_HOST: ${DB_HOST}`
|
|
||||||
2. `DB_PORT: ${DB_PORT}`
|
|
||||||
3. `DB_USER: ${DB_USER}`
|
|
||||||
4. `DB_PASS: ${DB_PASS}`
|
|
||||||
5. `DB_NAME: ${DB_NAME}`
|
|
||||||
|
|
||||||
즉, `.env`를 틀리게 적으면 화면도 데이터가 안 뜬다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 13.3 `server.js`는 여전히 중요하게 수정된 상태다
|
|
||||||
|
|
||||||
현재 `server.js`는 `dotenv.config()`를 사용한다.
|
|
||||||
|
|
||||||
이 구조는 이후 Compose나 실행 환경에서 변수를 주입할 때, 애플리케이션이 그 값을 받아들일 수 있게 하기 위해 유지해야 한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 14. 스택 중지 방법
|
|
||||||
|
|
||||||
## 14.1 작업 실행 위치
|
|
||||||
|
|
||||||
**Windows PowerShell / 프로젝트 루트**
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 14.2 권장 종료 명령
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
.\stop_docker_wsl.ps1
|
|
||||||
```
|
|
||||||
|
|
||||||
또는
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
.\stop_docker_wsl.bat
|
|
||||||
```
|
|
||||||
|
|
||||||
이 스크립트는 내부적으로 WSL 경로 변환 후 `docker compose down`을 수행한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 15. 장애 발생 시 점검 순서
|
|
||||||
|
|
||||||
## 15.1 `frontend` 화면은 뜨는데 데이터가 없을 때
|
|
||||||
|
|
||||||
**실행 위치: Windows PowerShell**
|
|
||||||
|
|
||||||
먼저 아래 두 API를 분리해서 본다.
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
Invoke-WebRequest -Uri http://localhost:3000/api/assets/master -UseBasicParsing | Select-Object -ExpandProperty StatusCode
|
|
||||||
Invoke-WebRequest -Uri http://localhost:8080/api/assets/master -UseBasicParsing | Select-Object -ExpandProperty StatusCode
|
|
||||||
```
|
|
||||||
|
|
||||||
판단 기준:
|
|
||||||
|
|
||||||
1. `3000`은 200이고 `8080`만 실패 -> 프런트 프록시 문제
|
|
||||||
2. 둘 다 실패 -> 백엔드 또는 외부 DB 연결 문제
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 15.2 백엔드가 외부 DB에 연결되지 않을 때
|
|
||||||
|
|
||||||
**실행 위치: Windows PowerShell**
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
wsl sh -lc "docker logs --tail=200 itam-backend"
|
|
||||||
```
|
|
||||||
|
|
||||||
점검 항목:
|
|
||||||
|
|
||||||
1. `.env`의 DB 정보가 정확한지
|
|
||||||
2. 외부 DB 서버 접근이 가능한지
|
|
||||||
3. 계정/비밀번호가 맞는지
|
|
||||||
4. 방화벽 또는 네트워크 이슈가 없는지
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 16. 운영 수동 배포 플로우
|
|
||||||
|
|
||||||
이 섹션은 현재 ITAM 저장소 기준으로 운영 서버에 반영할 때의 전체 흐름을 설명한다.
|
|
||||||
|
|
||||||
중요한 전제는 아래와 같다.
|
|
||||||
|
|
||||||
1. 로컬 수정본을 서버에 직접 복사하지 않는다.
|
|
||||||
2. 반드시 Gitea에 올라간 커밋을 기준으로 배포한다.
|
|
||||||
3. 운영 반영은 자동 푸시 배포가 아니라 Gitea workflow 수동 실행으로 진행한다.
|
|
||||||
4. 현재 기준 운영 배포 workflow는 `.gitea/workflows/itam_production_deploy.yml`이다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 16.1 전체 운영/배포 분기 흐름
|
|
||||||
|
|
||||||
운영 반영은 크게 세 상황으로 나뉜다.
|
|
||||||
|
|
||||||
1. 최초 운영 서버 구축 후 첫 배포
|
|
||||||
2. 코드 수정 후 일반 재배포
|
|
||||||
3. 검증 실패 또는 배포 실패 후 수정 재배포
|
|
||||||
|
|
||||||
아래 분기 구조로 이해하면 된다.
|
|
||||||
|
|
||||||
```mermaid
|
|
||||||
flowchart TD
|
|
||||||
START["배포 필요 발생"] --> CASE{"어떤 상황인가?"}
|
|
||||||
|
|
||||||
CASE -->|초기 구축| INIT["초기 운영 배포 준비"]
|
|
||||||
CASE -->|수정 반영| CHANGE["수정 후 재배포 준비"]
|
|
||||||
CASE -->|실패 후 재시도| RETRY["실패 원인 분석 후 재배포 준비"]
|
|
||||||
|
|
||||||
INIT --> INIT1["운영 서버 Docker / compose 확인"]
|
|
||||||
INIT1 --> INIT2["Gitea Variables / Secrets 등록"]
|
|
||||||
INIT2 --> INIT3["map_config.json / uploads 초기 데이터 준비"]
|
|
||||||
INIT3 --> MANUAL["Gitea에서 수동 배포 workflow 실행"]
|
|
||||||
|
|
||||||
CHANGE --> CHANGE1["로컬 수정 및 테스트"]
|
|
||||||
CHANGE1 --> CHANGE2["Gitea 커밋 / push"]
|
|
||||||
CHANGE2 --> CHANGE3["Code Check / Docker Build Check 통과"]
|
|
||||||
CHANGE3 --> MANUAL
|
|
||||||
|
|
||||||
RETRY --> RETRY1{"어디서 실패했는가?"}
|
|
||||||
RETRY1 -->|코드 체크 실패| FIX1["코드 또는 설정 수정"]
|
|
||||||
RETRY1 -->|배포 단계 실패| FIX2["서버 / 변수 / 권한 / 네트워크 수정"]
|
|
||||||
RETRY1 -->|Smoke Check 실패| FIX3["앱 기동 상태 / 프록시 / DB 상태 수정"]
|
|
||||||
FIX1 --> CHANGE2
|
|
||||||
FIX2 --> MANUAL
|
|
||||||
FIX3 --> MANUAL
|
|
||||||
|
|
||||||
MANUAL --> DEPLOY["운영 서버 반영 수행"]
|
|
||||||
DEPLOY --> RESULT{"최종 검증 통과?"}
|
|
||||||
RESULT -->|예| DONE["운영 반영 완료"]
|
|
||||||
RESULT -->|아니오| RETRY
|
|
||||||
linkStyle default stroke:#d32f2f,stroke-width:2px;
|
|
||||||
```
|
|
||||||
|
|
||||||
핵심은 아래와 같다.
|
|
||||||
|
|
||||||
1. 초기 구축은 서버와 운영 데이터 준비가 먼저다.
|
|
||||||
2. 수정 반영은 반드시 커밋과 push가 먼저다.
|
|
||||||
3. 실패 후 재배포는 실패 지점에 따라 수정 위치가 달라진다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 16.2 최초 운영 배포 플로우
|
|
||||||
|
|
||||||
최초 배포에서는 코드보다 운영 환경 준비가 더 중요하다.
|
|
||||||
|
|
||||||
순서는 아래와 같다.
|
|
||||||
|
|
||||||
1. 운영 서버에 Docker Engine과 `docker compose`를 설치한다.
|
|
||||||
2. 운영 서버에서 Gitea 저장소에 접근 가능한 SSH 키를 준비한다.
|
|
||||||
3. Gitea repository Variables / Secrets를 등록한다.
|
|
||||||
4. `PROD_DEPLOY_PATH` 경로를 정한다.
|
|
||||||
5. `map_config.json`, `uploads/` 초기 데이터를 준비한다.
|
|
||||||
6. Gitea에서 `itam_production_deploy.yml`을 수동 실행한다.
|
|
||||||
7. 배포 후 `ps`, `/health`, `/`, `/ready`를 확인한다.
|
|
||||||
|
|
||||||
즉 최초 배포는 아래 조건이 먼저 충족되어야 한다.
|
|
||||||
|
|
||||||
```text
|
|
||||||
서버 준비 완료
|
|
||||||
-> Gitea 변수 / 시크릿 등록 완료
|
|
||||||
-> 초기 데이터 준비 완료
|
|
||||||
-> 수동 배포 실행
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 16.3 수정 후 일반 재배포 플로우
|
|
||||||
|
|
||||||
일반적인 수정 반영은 아래 흐름이다.
|
|
||||||
|
|
||||||
1. 개발자가 로컬에서 코드 또는 설정을 수정한다.
|
|
||||||
2. 로컬에서 필요한 테스트를 수행한다.
|
|
||||||
3. 변경사항을 Gitea에 커밋 후 push 한다.
|
|
||||||
4. `itam_code_check.yml`이 빌드와 compose 문법을 검사한다.
|
|
||||||
5. `itam_docker_build_check.yml`이 운영용 이미지 빌드 가능 여부를 검사한다.
|
|
||||||
6. 두 검증이 통과하면 운영자가 Gitea에서 `itam_production_deploy.yml`을 수동 실행한다.
|
|
||||||
7. 운영 서버가 최신 커밋으로 동기화되고 컨테이너가 다시 올라온다.
|
|
||||||
8. smoke check 통과 여부를 확인한다.
|
|
||||||
|
|
||||||
아래 다이어그램은 이 일반 재배포 흐름을 보여준다.
|
|
||||||
|
|
||||||
```mermaid
|
|
||||||
flowchart LR
|
|
||||||
DEV["로컬 수정"] --> TEST["로컬 확인"]
|
|
||||||
TEST --> PUSH["커밋 / push"]
|
|
||||||
PUSH --> CODE["ITAM Code Check"]
|
|
||||||
CODE --> BUILD["ITAM Docker Build Check"]
|
|
||||||
BUILD --> GATE{"검증 통과?"}
|
|
||||||
GATE -->|예| RUN["Gitea에서 수동 배포 실행"]
|
|
||||||
GATE -->|아니오| FIX["로컬 수정 후 재커밋"]
|
|
||||||
FIX --> PUSH
|
|
||||||
RUN --> PROD["운영 서버 배포"]
|
|
||||||
PROD --> SMOKE{"Smoke Check 통과?"}
|
|
||||||
SMOKE -->|예| OK["배포 완료"]
|
|
||||||
SMOKE -->|아니오| FIXDEPLOY["원인 수정 후 재배포"]
|
|
||||||
FIXDEPLOY --> RUN
|
|
||||||
linkStyle default stroke:#d32f2f,stroke-width:2px;
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 16.4 수동 배포 workflow 내부 실행 순서
|
|
||||||
|
|
||||||
Gitea에서 `itam_production_deploy.yml`을 수동 실행하면 내부적으로는 아래 순서로 진행된다.
|
|
||||||
|
|
||||||
1. SSH agent를 설정한다.
|
|
||||||
2. 필수 Variables / Secrets가 모두 있는지 확인한다.
|
|
||||||
3. 운영용 `.env.deploy` 파일을 생성한다.
|
|
||||||
4. 운영 서버에 접속한다.
|
|
||||||
5. `PROD_DEPLOY_PATH`를 생성한다.
|
|
||||||
6. 저장소를 clone 또는 fetch 한다.
|
|
||||||
7. 선택한 브랜치의 최신 커밋으로 checkout, reset, clean 한다.
|
|
||||||
8. `uploads`, `logs/nginx` 디렉토리를 준비한다.
|
|
||||||
9. `.env.deploy`를 서버의 `.env`로 복사한다.
|
|
||||||
10. `docker compose -f docker-compose.prod.yaml config`를 수행한다.
|
|
||||||
11. `docker compose -f docker-compose.prod.yaml up -d --build`를 수행한다.
|
|
||||||
12. `docker compose ps`를 확인한다.
|
|
||||||
13. `/health`, `/`, backend `/ready` smoke check를 수행한다.
|
|
||||||
|
|
||||||
아래 다이어그램은 workflow 내부 실행 순서다.
|
|
||||||
|
|
||||||
```mermaid
|
|
||||||
flowchart TD
|
|
||||||
A["수동 배포 시작"] --> B["SSH agent 설정"]
|
|
||||||
B --> C["Variables / Secrets 검증"]
|
|
||||||
C --> D{"필수 값 누락 여부"}
|
|
||||||
D -->|예| E["즉시 실패 후 설정 보완"]
|
|
||||||
D -->|아니오| F[".env.deploy 생성"]
|
|
||||||
F --> G["운영 서버 SSH 접속"]
|
|
||||||
G --> H["배포 경로 생성"]
|
|
||||||
H --> I["git clone 또는 fetch"]
|
|
||||||
I --> J["지정 브랜치 checkout / reset / clean"]
|
|
||||||
J --> K["uploads / logs/nginx 준비"]
|
|
||||||
K --> L[".env 업로드 및 권한 설정"]
|
|
||||||
L --> M["compose config 검증"]
|
|
||||||
M --> N{"compose config 성공?"}
|
|
||||||
N -->|아니오| O["설정 수정 후 재실행"]
|
|
||||||
N -->|예| P["compose up -d --build"]
|
|
||||||
P --> Q["docker compose ps 확인"]
|
|
||||||
Q --> R["/health, /, /ready smoke check"]
|
|
||||||
R --> S{"smoke check 성공?"}
|
|
||||||
S -->|예| T["운영 배포 완료"]
|
|
||||||
S -->|아니오| U["원인 분석 후 재배포"]
|
|
||||||
linkStyle default stroke:#d32f2f,stroke-width:2px;
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 16.5 실패 후 검증 및 재배포 플로우
|
|
||||||
|
|
||||||
실패가 났다고 해서 항상 같은 방식으로 다시 배포하면 안 된다.
|
|
||||||
|
|
||||||
실패 지점별 판단은 아래처럼 나눈다.
|
|
||||||
|
|
||||||
1. Code Check 실패: TypeScript, build, compose 문법 문제를 먼저 수정한다.
|
|
||||||
2. Docker Build Check 실패: Dockerfile, 정적 자산 복사, 운영 빌드 컨텍스트 문제를 수정한다.
|
|
||||||
3. Deploy 단계 실패: SSH, Gitea 변수, 서버 권한, 경로, git 접근, Docker 권한을 수정한다.
|
|
||||||
4. Smoke Check 실패: Nginx 프록시, backend readiness, 외부 DB 연결, 앱 런타임 오류를 수정한다.
|
|
||||||
|
|
||||||
즉 재배포 전 판단 기준은 아래와 같다.
|
|
||||||
|
|
||||||
```text
|
|
||||||
CI 실패 -> 로컬 코드 / 설정 수정 후 재커밋
|
|
||||||
배포 실패 -> 서버 환경 또는 배포 설정 수정 후 수동 재실행
|
|
||||||
Smoke Check 실패 -> 앱 / 프록시 / DB 상태 수정 후 수동 재실행
|
|
||||||
```
|
|
||||||
|
|
||||||
운영 관점에서는 아래 순서를 지키는 것이 안전하다.
|
|
||||||
|
|
||||||
1. 실패 지점 확인
|
|
||||||
2. 원인 수정
|
|
||||||
3. 같은 실패가 다시 나는지 좁은 범위로 재검증
|
|
||||||
4. 그 다음에만 수동 배포 재실행
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 16.6 문서 기준 요약
|
|
||||||
|
|
||||||
현재 ITAM 운영 배포는 아래 원칙으로 이해하면 된다.
|
|
||||||
|
|
||||||
1. 수정은 로컬에서 한다.
|
|
||||||
2. 배포 기준점은 Gitea에 올라간 커밋이다.
|
|
||||||
3. 운영 반영은 Gitea 수동 workflow 실행으로 한다.
|
|
||||||
4. 초기 배포, 일반 재배포, 실패 후 재배포는 분기 기준이 다르다.
|
|
||||||
5. 최종 성공 여부는 컨테이너 상태가 아니라 smoke check까지 통과했는지로 판단한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 15.3 프런트 프록시가 의심될 때
|
|
||||||
|
|
||||||
**확인 위치: `vite.config.ts`, `docker-compose.yaml`**
|
|
||||||
|
|
||||||
다음 두 설정이 유지되는지 확인한다.
|
|
||||||
|
|
||||||
`vite.config.ts`
|
|
||||||
|
|
||||||
```ts
|
|
||||||
const proxyTarget = process.env.VITE_DEV_PROXY_TARGET || 'http://localhost:3000';
|
|
||||||
```
|
|
||||||
|
|
||||||
`docker-compose.yaml`
|
|
||||||
|
|
||||||
```yaml
|
|
||||||
VITE_DEV_PROXY_TARGET: http://backend:3000
|
|
||||||
```
|
|
||||||
|
|
||||||
이 둘 중 하나라도 바뀌면 Docker 안에서 화면 데이터가 다시 안 보일 수 있다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 17. 현재 기준 재현 절차 요약
|
|
||||||
|
|
||||||
가장 짧게 정리하면 아래 순서다.
|
|
||||||
|
|
||||||
1. Gitea에서 소스를 클론한다.
|
|
||||||
2. Windows PowerShell에서 프로젝트 루트로 이동한다.
|
|
||||||
3. `.env`에 외부 MySQL 정보를 작성한다.
|
|
||||||
4. Docker Desktop + WSL 통합 또는 WSL 내부 Docker 사용 가능 상태를 만든다.
|
|
||||||
5. `start_docker_wsl.ps1`를 실행한다.
|
|
||||||
6. `http://localhost:3000/api/assets/master`가 200인지 확인한다.
|
|
||||||
7. `http://localhost:8080/api/assets/master`가 200인지 확인한다.
|
|
||||||
8. 브라우저에서 `http://localhost:8080`을 열어 실제 데이터 표시를 확인한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 18. 현재 최종 결론
|
|
||||||
|
|
||||||
현재 저장소의 도커라이징 구조는 실무 표준에 맞는 `무상태 앱 컨테이너 + 외부 DB` 구조다.
|
|
||||||
|
|
||||||
현재 핵심은 아래 세 가지다.
|
|
||||||
|
|
||||||
1. `backend`는 외부 MySQL에 직접 연결한다.
|
|
||||||
2. `frontend`는 `backend:3000`으로 API 프록시한다.
|
|
||||||
3. WSL 경로 변환 스크립트를 통해 Windows 한글 경로에서도 안정적으로 실행한다.
|
|
||||||
|
|
||||||
즉, 이 문서대로 진행하면 Gitea 소스만 받은 상태에서 지금과 같은 Docker 실행 구조를 재현할 수 있다.
|
|
||||||
730
doc_readme2.md
730
doc_readme2.md
@@ -1,730 +0,0 @@
|
|||||||
# ITAM 도커라이징 최종 재현 가이드
|
|
||||||
|
|
||||||
## 1. 문서 목적
|
|
||||||
|
|
||||||
이 문서는 현재 Git 저장소에 올라간 파일만 가지고, 지금과 동일한 수준으로 ITAM 시스템을 도커라이징하고 실행하는 절차를 처음부터 끝까지 정리한 최종 가이드다.
|
|
||||||
|
|
||||||
이 문서만 읽어도 아래 목표를 달성할 수 있게 작성한다.
|
|
||||||
|
|
||||||
1. 현재 저장소 구조를 이해한다.
|
|
||||||
2. 왜 이렇게 도커라이징했는지 판단 근거를 안다.
|
|
||||||
3. WSL2 기반으로 실제 스택을 기동한다.
|
|
||||||
4. 외부 MySQL에서 내부 MySQL 컨테이너로 초기 데이터를 bootstrap 한다.
|
|
||||||
5. 프런트 8080과 백엔드 3000이 모두 정상 동작하는지 검증한다.
|
|
||||||
6. 재초기화, 재기동, 장애 확인까지 수행한다.
|
|
||||||
|
|
||||||
이 문서는 최종 성공 구조 기준이다. 실패 기록은 `doc_readme.md`를 본다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 2. 최종 목표 구조
|
|
||||||
|
|
||||||
현재 최종 구조는 아래 4개 서비스/역할로 나뉜다.
|
|
||||||
|
|
||||||
1. `frontend`: Vite 개발 서버 컨테이너, 포트 8080
|
|
||||||
2. `backend`: Express API 서버 컨테이너, 포트 3000
|
|
||||||
3. `db`: MySQL 8 컨테이너, 포트 3306
|
|
||||||
4. `db-bootstrap`: 외부 MySQL -> 내부 MySQL로 1회성 복제 수행 후 종료되는 도우미 컨테이너
|
|
||||||
|
|
||||||
논리 흐름은 다음과 같다.
|
|
||||||
|
|
||||||
```text
|
|
||||||
브라우저 -> frontend:8080 -> Vite proxy -> backend:3000 -> db:3306
|
|
||||||
\
|
|
||||||
-> /uploads -> backend 정적 경로
|
|
||||||
|
|
||||||
초기 1회 기동 시
|
|
||||||
외부 MySQL(.env) -> db-bootstrap -> 내부 MySQL(db)
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 3. 왜 이 구조를 선택했는가
|
|
||||||
|
|
||||||
이 저장소는 처음부터 운영형 정적 배포 앱이 아니었다. 실제 구조는 다음과 같았다.
|
|
||||||
|
|
||||||
1. 프런트는 Vite 개발 서버가 따로 돈다.
|
|
||||||
2. 백엔드는 Express API가 따로 돈다.
|
|
||||||
3. 프런트는 상대 경로 `/api`를 호출한다.
|
|
||||||
4. 백엔드는 프런트의 `dist`를 서빙하지 않는다.
|
|
||||||
|
|
||||||
따라서 내일 바로 시연 가능한 수준까지 빠르게 안정화하려면 아래 전략이 가장 맞다.
|
|
||||||
|
|
||||||
1. 프런트를 Vite dev server 그대로 컨테이너화한다.
|
|
||||||
2. 백엔드를 별도 컨테이너로 유지한다.
|
|
||||||
3. DB는 MySQL 8 컨테이너로 묶되, 초기 데이터는 외부 DB에서 복제한다.
|
|
||||||
4. 프런트 프록시는 컨테이너 네트워크 서비스명 `backend`로 붙게 한다.
|
|
||||||
|
|
||||||
즉, 현재 구조는 "개발형 구조를 Docker로 재현한 시연/개발용 Compose"다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 4. 저장소 내 최종 관련 파일 목록
|
|
||||||
|
|
||||||
현재 도커라이징과 직접 관련된 핵심 파일은 아래와 같다.
|
|
||||||
|
|
||||||
1. `.dockerignore`
|
|
||||||
2. `Dockerfile.frontend`
|
|
||||||
3. `Dockerfile.backend`
|
|
||||||
4. `docker-compose.yaml`
|
|
||||||
5. `start_docker_wsl.ps1`
|
|
||||||
6. `stop_docker_wsl.ps1`
|
|
||||||
7. `start_docker_wsl.bat`
|
|
||||||
8. `stop_docker_wsl.bat`
|
|
||||||
9. `docker/mysql/init/README.md`
|
|
||||||
10. `server.js`
|
|
||||||
11. `vite.config.ts`
|
|
||||||
|
|
||||||
각 파일 역할은 다음과 같다.
|
|
||||||
|
|
||||||
### 4.1 `.dockerignore`
|
|
||||||
|
|
||||||
Docker build context에서 제외할 파일을 정의한다.
|
|
||||||
|
|
||||||
주요 제외 대상은 다음과 같다.
|
|
||||||
|
|
||||||
1. `node_modules`
|
|
||||||
2. `dist`
|
|
||||||
3. `build`
|
|
||||||
4. `.git`
|
|
||||||
5. `.env`
|
|
||||||
6. `uploads`
|
|
||||||
7. `*.xlsx`
|
|
||||||
8. `*.log`
|
|
||||||
|
|
||||||
### 4.2 `Dockerfile.frontend`
|
|
||||||
|
|
||||||
프런트 컨테이너 이미지 정의다.
|
|
||||||
|
|
||||||
```dockerfile
|
|
||||||
FROM node:20-alpine
|
|
||||||
|
|
||||||
WORKDIR /app
|
|
||||||
|
|
||||||
COPY package*.json ./
|
|
||||||
RUN npm ci
|
|
||||||
|
|
||||||
COPY . .
|
|
||||||
|
|
||||||
EXPOSE 8080
|
|
||||||
|
|
||||||
CMD ["npm", "run", "dev", "--", "--host", "0.0.0.0"]
|
|
||||||
```
|
|
||||||
|
|
||||||
이 이미지는 Vite dev server를 컨테이너에서 띄우기 위한 것이다.
|
|
||||||
|
|
||||||
### 4.3 `Dockerfile.backend`
|
|
||||||
|
|
||||||
백엔드 컨테이너 이미지 정의다.
|
|
||||||
|
|
||||||
```dockerfile
|
|
||||||
FROM node:20-alpine
|
|
||||||
|
|
||||||
WORKDIR /app
|
|
||||||
|
|
||||||
COPY package*.json ./
|
|
||||||
RUN npm ci
|
|
||||||
|
|
||||||
COPY . .
|
|
||||||
|
|
||||||
EXPOSE 3000
|
|
||||||
|
|
||||||
CMD ["npm", "run", "server"]
|
|
||||||
```
|
|
||||||
|
|
||||||
### 4.4 `docker-compose.yaml`
|
|
||||||
|
|
||||||
전체 스택의 핵심 파일이다.
|
|
||||||
|
|
||||||
현재 최종 구성은 다음 논리를 가진다.
|
|
||||||
|
|
||||||
1. `db`는 MySQL 8 내부 DB다.
|
|
||||||
2. `db-bootstrap`은 외부 DB 데이터를 내부 DB로 1회 복제한다.
|
|
||||||
3. `backend`는 내부 `db`에 붙는다.
|
|
||||||
4. `frontend`는 `backend` 서비스명으로 프록시한다.
|
|
||||||
|
|
||||||
### 4.5 `start_docker_wsl.ps1`
|
|
||||||
|
|
||||||
Windows에서 WSL 경유로 Docker Compose를 안전하게 기동하는 진입점이다.
|
|
||||||
|
|
||||||
핵심은 다음 두 가지다.
|
|
||||||
|
|
||||||
1. 프로젝트 Windows 경로를 `wslpath`로 WSL 경로로 바꾼다.
|
|
||||||
2. 그 경로로 이동한 뒤 `docker compose up --build -d`를 수행한다.
|
|
||||||
|
|
||||||
### 4.6 `stop_docker_wsl.ps1`
|
|
||||||
|
|
||||||
같은 방식으로 WSL 내부에서 `docker compose down`을 수행해 스택을 안전하게 내린다.
|
|
||||||
|
|
||||||
### 4.7 `start_docker_wsl.bat`, `stop_docker_wsl.bat`
|
|
||||||
|
|
||||||
더블클릭 또는 간단 실행용 래퍼다. 내부적으로 PowerShell 스크립트를 호출한다.
|
|
||||||
|
|
||||||
### 4.8 `server.js`
|
|
||||||
|
|
||||||
중요 포인트는 다음 두 가지다.
|
|
||||||
|
|
||||||
1. `dotenv.config();`를 사용한다.
|
|
||||||
2. `dotenv.config({ override: true })`를 사용하지 않는다.
|
|
||||||
|
|
||||||
이 차이로 Compose 환경변수 `DB_HOST=db`가 `.env`보다 우선하도록 보장한다.
|
|
||||||
|
|
||||||
### 4.9 `vite.config.ts`
|
|
||||||
|
|
||||||
현재 프록시는 환경변수 기반으로 동작한다.
|
|
||||||
|
|
||||||
```ts
|
|
||||||
const proxyTarget = process.env.VITE_DEV_PROXY_TARGET || 'http://localhost:3000';
|
|
||||||
```
|
|
||||||
|
|
||||||
로컬 PC에서 직접 Vite를 띄우면 기본값 `http://localhost:3000`을 쓴다.
|
|
||||||
컨테이너에서는 Compose가 `http://backend:3000`을 주입한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 5. 현재 최종 `docker-compose.yaml` 구조 설명
|
|
||||||
|
|
||||||
아래는 실제 동작 관점에서 읽어야 할 핵심 내용이다.
|
|
||||||
|
|
||||||
### 5.1 `db` 서비스
|
|
||||||
|
|
||||||
역할:
|
|
||||||
|
|
||||||
1. 내부 MySQL 데이터 저장소
|
|
||||||
2. 앱이 최종적으로 붙는 DB
|
|
||||||
|
|
||||||
핵심 설정:
|
|
||||||
|
|
||||||
1. 이미지: `mysql:8.0`
|
|
||||||
2. DB 이름: `itam`
|
|
||||||
3. 앱 계정: `itam_admin`
|
|
||||||
4. 데이터 볼륨: `itam_mysql_data`
|
|
||||||
5. healthcheck 사용
|
|
||||||
|
|
||||||
healthcheck는 `mysqladmin ping`으로 동작하며, `backend`와 `db-bootstrap`은 이 상태를 기다린다.
|
|
||||||
|
|
||||||
### 5.2 `db-bootstrap` 서비스
|
|
||||||
|
|
||||||
역할:
|
|
||||||
|
|
||||||
1. 외부 원본 DB에서 내부 `db`로 초기 데이터 복제
|
|
||||||
2. 1회성 작업 후 종료
|
|
||||||
|
|
||||||
핵심 포인트:
|
|
||||||
|
|
||||||
1. `.env`를 읽어 외부 DB 접속 정보를 가져온다.
|
|
||||||
2. 내부 `db`에 `asset_core` 테이블이 이미 존재하면 아무 것도 하지 않고 종료한다.
|
|
||||||
3. 그렇지 않으면 `mysqldump | mysql` 파이프라인으로 복제한다.
|
|
||||||
4. `restart: "no"` 이므로 정상 종료 후 반복 실행하지 않는다.
|
|
||||||
|
|
||||||
또한 source DB와 target DB 변수는 분리돼 있다.
|
|
||||||
|
|
||||||
1. source: `SOURCE_DB_*`
|
|
||||||
2. target: `TARGET_DB_*`
|
|
||||||
|
|
||||||
이 구조로 외부 원본 DB 자격증명과 내부 컨테이너 DB 자격증명이 섞이지 않는다.
|
|
||||||
|
|
||||||
### 5.3 `backend` 서비스
|
|
||||||
|
|
||||||
역할:
|
|
||||||
|
|
||||||
1. Express API 제공
|
|
||||||
2. 내부 `db`에 연결
|
|
||||||
3. `/uploads` 정적 제공
|
|
||||||
|
|
||||||
핵심 포인트:
|
|
||||||
|
|
||||||
1. `env_file: .env`를 유지하지만,
|
|
||||||
2. Compose `environment`에서 `DB_HOST=db`, `DB_PORT=3306`, `DB_USER=itam_admin`, `DB_PASS=itam1234`, `DB_NAME=itam`를 다시 지정한다.
|
|
||||||
3. `depends_on`은 `db` healthy와 `db-bootstrap` 성공 종료를 모두 기다린다.
|
|
||||||
|
|
||||||
즉, 백엔드는 DB bootstrap이 끝난 뒤 시작한다.
|
|
||||||
|
|
||||||
### 5.4 `frontend` 서비스
|
|
||||||
|
|
||||||
역할:
|
|
||||||
|
|
||||||
1. Vite dev server 제공
|
|
||||||
2. 브라우저 요청 `/api`, `/uploads`를 `backend`로 프록시
|
|
||||||
|
|
||||||
핵심 포인트:
|
|
||||||
|
|
||||||
1. `VITE_DEV_PROXY_TARGET: http://backend:3000`
|
|
||||||
2. `CHOKIDAR_USEPOLLING: "true"`
|
|
||||||
3. `npm run dev -- --host 0.0.0.0`
|
|
||||||
|
|
||||||
중요한 이유는 컨테이너 안의 `localhost`가 호스트의 `localhost`가 아니기 때문이다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 6. 사전 준비 조건
|
|
||||||
|
|
||||||
이 저장소를 지금처럼 기동하려면 다음 전제가 필요하다.
|
|
||||||
|
|
||||||
### 6.1 운영체제와 런타임
|
|
||||||
|
|
||||||
1. Windows
|
|
||||||
2. WSL2 Ubuntu 설치 및 실행 중
|
|
||||||
3. Docker CLI가 WSL 내부에서 동작 가능
|
|
||||||
|
|
||||||
권장 확인 명령:
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
wsl -l -v
|
|
||||||
wsl sh -lc "docker --version"
|
|
||||||
```
|
|
||||||
|
|
||||||
### 6.2 `.env` 파일
|
|
||||||
|
|
||||||
현재 최종 구조는 "첫 기동 시 외부 DB에서 내부 DB로 bootstrap" 하는 방식이므로 `.env`가 반드시 필요하다.
|
|
||||||
|
|
||||||
최소한 다음 값은 외부 원본 DB를 가리켜야 한다.
|
|
||||||
|
|
||||||
```env
|
|
||||||
DB_HOST=<external-mysql-host>
|
|
||||||
DB_PORT=3306
|
|
||||||
DB_USER=<external-db-user>
|
|
||||||
DB_PASS=<external-db-password>
|
|
||||||
DB_NAME=itam
|
|
||||||
```
|
|
||||||
|
|
||||||
주의:
|
|
||||||
|
|
||||||
1. `.env`는 `db-bootstrap`이 외부 원본 DB에 접속할 때 사용한다.
|
|
||||||
2. `backend`는 최종적으로 내부 `db` 컨테이너를 쓰므로, 런타임에서는 Compose `environment`가 우선한다.
|
|
||||||
|
|
||||||
### 6.3 한글 경로 주의
|
|
||||||
|
|
||||||
현재 프로젝트 경로는 한글과 공백을 포함한다.
|
|
||||||
|
|
||||||
```text
|
|
||||||
c:\Users\user\Desktop\안건 파일\itam
|
|
||||||
```
|
|
||||||
|
|
||||||
이 때문에 Docker 관련 명령은 수동으로 경로를 조립하지 말고, `start_docker_wsl.ps1` / `stop_docker_wsl.ps1`을 우선 사용해야 한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 7. 첫 기동 절차
|
|
||||||
|
|
||||||
이 절차는 "Git에서 소스를 받은 뒤 처음 올리는 경우" 기준이다.
|
|
||||||
|
|
||||||
### 7.1 저장소 준비
|
|
||||||
|
|
||||||
1. 저장소를 받는다.
|
|
||||||
2. `.env`가 올바른 외부 원본 DB를 가리키는지 확인한다.
|
|
||||||
3. WSL이 켜져 있는지 확인한다.
|
|
||||||
|
|
||||||
### 7.2 권장 실행 방법
|
|
||||||
|
|
||||||
Windows PowerShell에서 프로젝트 루트로 이동한 뒤 아래 중 하나를 사용한다.
|
|
||||||
|
|
||||||
방법 A:
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
.\start_docker_wsl.ps1
|
|
||||||
```
|
|
||||||
|
|
||||||
방법 B:
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
.\start_docker_wsl.bat
|
|
||||||
```
|
|
||||||
|
|
||||||
### 7.3 내부 실행 순서
|
|
||||||
|
|
||||||
스크립트는 내부적으로 다음 순서로 동작한다.
|
|
||||||
|
|
||||||
1. 현재 Windows 경로를 WSL 경로로 변환한다.
|
|
||||||
2. WSL 동작 여부를 확인한다.
|
|
||||||
3. WSL 내부 Docker 사용 가능 여부를 확인한다.
|
|
||||||
4. `docker compose up --build -d`를 수행한다.
|
|
||||||
|
|
||||||
### 7.4 기대되는 컨테이너 순서
|
|
||||||
|
|
||||||
정상이라면 다음 순서로 올라온다.
|
|
||||||
|
|
||||||
1. `itam-db`
|
|
||||||
2. `itam-db-bootstrap`
|
|
||||||
3. `itam-backend`
|
|
||||||
4. `itam-frontend`
|
|
||||||
|
|
||||||
`itam-db-bootstrap`은 정상이라면 최종 상태가 `Exited (0)`이어야 한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 8. 첫 기동 후 검증 절차
|
|
||||||
|
|
||||||
기동 후에는 반드시 아래 검증을 수행한다.
|
|
||||||
|
|
||||||
### 8.1 컨테이너 상태 확인
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
wsl sh -lc "docker ps -a --format 'table {{.Names}}\t{{.Status}}' | grep itam"
|
|
||||||
```
|
|
||||||
|
|
||||||
정상 기대 상태:
|
|
||||||
|
|
||||||
1. `itam-db` -> `Up ... (healthy)`
|
|
||||||
2. `itam-db-bootstrap` -> `Exited (0)`
|
|
||||||
3. `itam-backend` -> `Up`
|
|
||||||
4. `itam-frontend` -> `Up`
|
|
||||||
|
|
||||||
### 8.2 백엔드 API 직접 확인
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
Invoke-WebRequest -Uri http://localhost:3000/api/assets/master -UseBasicParsing | Select-Object -ExpandProperty StatusCode
|
|
||||||
```
|
|
||||||
|
|
||||||
정상 기대값:
|
|
||||||
|
|
||||||
1. `200`
|
|
||||||
|
|
||||||
### 8.3 프런트 경유 API 확인
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
Invoke-WebRequest -Uri http://localhost:8080/api/assets/master -UseBasicParsing | Select-Object -ExpandProperty StatusCode
|
|
||||||
```
|
|
||||||
|
|
||||||
정상 기대값:
|
|
||||||
|
|
||||||
1. `200`
|
|
||||||
|
|
||||||
### 8.4 데이터가 실제로 들어왔는지 확인
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
wsl sh -lc "docker exec itam-db mysql -uitam_admin -pitam1234 -D itam -e 'SHOW TABLES' | head -n 20"
|
|
||||||
```
|
|
||||||
|
|
||||||
정상이라면 아래와 같은 테이블들이 보여야 한다.
|
|
||||||
|
|
||||||
1. `asset_core`
|
|
||||||
2. `asset_remote`
|
|
||||||
3. `asset_spec`
|
|
||||||
4. `asset_location`
|
|
||||||
5. `asset_history`
|
|
||||||
6. `asset_software_perpetual`
|
|
||||||
7. `asset_software_subscription`
|
|
||||||
8. `hardware_components_master`
|
|
||||||
9. `job_spec_standards`
|
|
||||||
|
|
||||||
### 8.5 브라우저 화면 확인
|
|
||||||
|
|
||||||
브라우저에서 아래 주소를 연다.
|
|
||||||
|
|
||||||
```text
|
|
||||||
http://localhost:8080
|
|
||||||
```
|
|
||||||
|
|
||||||
목록/대시보드 데이터가 보이면 화면까지 정상 연결된 것이다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 9. 재기동 절차
|
|
||||||
|
|
||||||
코드만 수정됐고 DB는 유지하고 싶다면 다음처럼 하면 된다.
|
|
||||||
|
|
||||||
### 9.1 스택 종료
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
.\stop_docker_wsl.ps1
|
|
||||||
```
|
|
||||||
|
|
||||||
또는
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
.\stop_docker_wsl.bat
|
|
||||||
```
|
|
||||||
|
|
||||||
### 9.2 스택 재기동
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
.\start_docker_wsl.ps1
|
|
||||||
```
|
|
||||||
|
|
||||||
이 경우 `itam_mysql_data` 볼륨이 유지되므로, `db-bootstrap`은 내부 DB에 `asset_core`가 이미 있음을 감지하고 빠르게 종료한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 10. DB를 완전히 다시 초기화하는 절차
|
|
||||||
|
|
||||||
외부 원본 DB에서 다시 처음부터 내부 DB를 복제하고 싶다면, MySQL 볼륨을 제거해야 한다.
|
|
||||||
|
|
||||||
### 10.1 스택 중지
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
.\stop_docker_wsl.ps1
|
|
||||||
```
|
|
||||||
|
|
||||||
### 10.2 MySQL 데이터 볼륨 삭제
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
wsl sh -lc "docker volume rm -f itam_itam_mysql_data"
|
|
||||||
```
|
|
||||||
|
|
||||||
### 10.3 다시 시작
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
.\start_docker_wsl.ps1
|
|
||||||
```
|
|
||||||
|
|
||||||
이때 `db-bootstrap`이 외부 DB에서 내부 DB로 전체를 다시 복제한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 11. 현재 구조에서 꼭 알아야 할 설계 포인트
|
|
||||||
|
|
||||||
### 11.1 `server.js`의 `dotenv.config()` 변경 이유
|
|
||||||
|
|
||||||
백엔드가 내부 DB로 붙게 하려면 Compose가 준 환경변수가 `.env`보다 우선해야 한다.
|
|
||||||
|
|
||||||
만약 아래처럼 `override: true`를 쓰면 안 된다.
|
|
||||||
|
|
||||||
```js
|
|
||||||
dotenv.config({ override: true });
|
|
||||||
```
|
|
||||||
|
|
||||||
이렇게 되면 내부 `db`가 아니라 `.env`의 외부 DB로 다시 붙을 수 있다.
|
|
||||||
|
|
||||||
현재는 아래가 맞다.
|
|
||||||
|
|
||||||
```js
|
|
||||||
dotenv.config();
|
|
||||||
```
|
|
||||||
|
|
||||||
### 11.2 왜 `docker-entrypoint-initdb.d` 기반 dump 파일을 안 쓰는가
|
|
||||||
|
|
||||||
처음에는 이 방식을 시도했지만, 실제 데이터의 긴 문자열/깨진 텍스트 때문에 import가 line 97에서 중단됐다.
|
|
||||||
|
|
||||||
그래서 현재는 더 안정적인 아래 방식을 쓴다.
|
|
||||||
|
|
||||||
1. 외부 DB에서 `mysqldump`
|
|
||||||
2. 파이프로 내부 `db`에 즉시 `mysql` import
|
|
||||||
|
|
||||||
즉, 파일 중간 생성물을 신뢰하지 않는 구조다.
|
|
||||||
|
|
||||||
### 11.3 왜 프런트 프록시 타깃을 환경변수화했는가
|
|
||||||
|
|
||||||
로컬 직접 실행과 컨테이너 실행의 네트워크 기준이 다르기 때문이다.
|
|
||||||
|
|
||||||
1. 로컬 직접 실행: `localhost:3000`이 맞다.
|
|
||||||
2. 컨테이너 내부 실행: `backend:3000`이 맞다.
|
|
||||||
|
|
||||||
그래서 `vite.config.ts`는 둘 다 수용할 수 있게 작성됐다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 12. 문제 발생 시 진단 순서
|
|
||||||
|
|
||||||
이 프로젝트에서는 문제를 아래 순서로 자르면 가장 빠르다.
|
|
||||||
|
|
||||||
### 12.1 브라우저 화면에 데이터가 없을 때
|
|
||||||
|
|
||||||
먼저 다음 둘을 분리해서 본다.
|
|
||||||
|
|
||||||
1. `http://localhost:3000/api/assets/master`
|
|
||||||
2. `http://localhost:8080/api/assets/master`
|
|
||||||
|
|
||||||
판단 기준:
|
|
||||||
|
|
||||||
1. `3000`은 200이고 `8080`만 실패면 프런트 프록시 문제다.
|
|
||||||
2. 둘 다 실패면 백엔드 또는 DB 문제다.
|
|
||||||
|
|
||||||
### 12.2 DB bootstrap이 성공했는지 확인할 때
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
wsl sh -lc "docker ps -a --format 'table {{.Names}}\t{{.Status}}' | grep itam"
|
|
||||||
```
|
|
||||||
|
|
||||||
여기서 `itam-db-bootstrap`이 `Exited (0)`인지 본다.
|
|
||||||
|
|
||||||
### 12.3 내부 DB에 실제 데이터가 있는지 확인할 때
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
wsl sh -lc "docker exec itam-db mysql -uitam_admin -pitam1234 -D itam -e 'SHOW TABLES'"
|
|
||||||
```
|
|
||||||
|
|
||||||
### 12.4 백엔드 로그 확인
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
wsl sh -lc "docker logs --tail=200 itam-backend"
|
|
||||||
```
|
|
||||||
|
|
||||||
### 12.5 DB 로그 확인
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
wsl sh -lc "docker logs --tail=200 itam-db"
|
|
||||||
```
|
|
||||||
|
|
||||||
### 12.6 프런트 로그 확인
|
|
||||||
|
|
||||||
```powershell
|
|
||||||
wsl sh -lc "docker logs --tail=200 itam-frontend"
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 13. 자주 나올 수 있는 장애와 해석
|
|
||||||
|
|
||||||
### 13.1 `docker` 명령이 PowerShell에서 안 보임
|
|
||||||
|
|
||||||
의미:
|
|
||||||
|
|
||||||
1. Windows 셸이 아니라 WSL에서 Docker를 쓰는 환경이다.
|
|
||||||
|
|
||||||
대응:
|
|
||||||
|
|
||||||
1. `start_docker_wsl.ps1` 사용
|
|
||||||
|
|
||||||
### 13.2 `asset_core` 테이블 없음
|
|
||||||
|
|
||||||
의미:
|
|
||||||
|
|
||||||
1. 내부 DB 초기화가 안 됐거나 bootstrap이 안 끝났다.
|
|
||||||
|
|
||||||
대응:
|
|
||||||
|
|
||||||
1. `db-bootstrap` 상태 확인
|
|
||||||
2. `.env` 외부 DB 접속 정보 확인
|
|
||||||
3. 필요하면 볼륨 삭제 후 재초기화
|
|
||||||
|
|
||||||
### 13.3 `3000` API는 되는데 화면은 비어 있음
|
|
||||||
|
|
||||||
의미:
|
|
||||||
|
|
||||||
1. DB는 정상이고 프런트 프록시 또는 화면 렌더링 문제다.
|
|
||||||
|
|
||||||
대응:
|
|
||||||
|
|
||||||
1. `8080/api/assets/master` 상태 먼저 확인
|
|
||||||
|
|
||||||
### 13.4 `db-bootstrap`가 실패 종료함
|
|
||||||
|
|
||||||
의미 후보:
|
|
||||||
|
|
||||||
1. `.env` 외부 DB 접속 정보 오류
|
|
||||||
2. 외부 DB 네트워크 접근 불가
|
|
||||||
3. 외부 계정 권한 문제
|
|
||||||
|
|
||||||
대응:
|
|
||||||
|
|
||||||
1. `docker logs itam-db-bootstrap` 확인
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 14. 현재 최종 검증 완료 상태
|
|
||||||
|
|
||||||
이 저장소는 아래 상태까지 검증이 완료됐다.
|
|
||||||
|
|
||||||
1. WSL2 Ubuntu에서 Docker 실행 가능
|
|
||||||
2. `start_docker_wsl.ps1`로 전체 스택 기동 가능
|
|
||||||
3. `db` 컨테이너 healthcheck 통과
|
|
||||||
4. `db-bootstrap`가 외부 DB에서 내부 DB로 데이터 복제 후 `Exited (0)` 종료
|
|
||||||
5. `backend`가 내부 `db`를 사용해 API 응답 가능
|
|
||||||
6. `frontend`가 `backend`를 프록시해 8080 기준 화면/API 동작 가능
|
|
||||||
7. 내부 MySQL에 실데이터 적재 확인
|
|
||||||
|
|
||||||
즉, 현재 Git에 올라간 상태만으로도 WSL2와 외부 원본 DB 정보만 있으면 지금과 같은 수준의 Docker 실행 재현이 가능하다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 15. 현재 구조의 한계와 다음 단계
|
|
||||||
|
|
||||||
현재 구조는 충분히 시연 가능하고 개발 재현도 가능하지만, 다음은 아직 별도 작업이 필요하다.
|
|
||||||
|
|
||||||
1. 운영형 정적 배포 구조 전환
|
|
||||||
2. 외부 DB 없이도 완전 독립 실행 가능한 정식 dump/backup 체계
|
|
||||||
3. `.env.example` 정리
|
|
||||||
4. DB bootstrap 전용 계정/권한 최소화
|
|
||||||
5. 장기적으로 `map_config.json` 파일 저장 정책 정리
|
|
||||||
|
|
||||||
하지만 "현재 저장소만으로 지금과 같은 Docker 실행 상태 재현"이라는 목표는 이미 충족한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 16. 빠른 실행 요약
|
|
||||||
|
|
||||||
가장 짧게 요약하면 다음 순서다.
|
|
||||||
|
|
||||||
1. `.env`에 외부 원본 MySQL 접속 정보를 넣는다.
|
|
||||||
2. WSL2 Ubuntu와 WSL 내부 Docker가 살아 있는지 확인한다.
|
|
||||||
3. `start_docker_wsl.ps1`를 실행한다.
|
|
||||||
4. `itam-db-bootstrap`가 `Exited (0)`인지 확인한다.
|
|
||||||
5. `http://localhost:3000/api/assets/master`와 `http://localhost:8080/api/assets/master`가 모두 200인지 확인한다.
|
|
||||||
6. 브라우저에서 `http://localhost:8080`을 열어 데이터가 보이는지 확인한다.
|
|
||||||
|
|
||||||
이 순서대로 진행하면 현재 저장소 기준 Dockerized ITAM 시연 환경을 재현할 수 있다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 17. 2026-06-16 최신 정정
|
|
||||||
|
|
||||||
이 문서의 상단 본문은 한동안 사용했던 `내부 db + db-bootstrap` 구조를 기준으로 작성됐다. 하지만 오늘 기준 현재 저장소의 실제 `docker-compose.yaml`은 다시 `무상태 앱 컨테이너 + 외부 DB` 구조로 되돌아가 있다.
|
|
||||||
|
|
||||||
따라서 현재 시점의 정답 아키텍처는 아래다.
|
|
||||||
|
|
||||||
1. `backend` 컨테이너
|
|
||||||
2. `frontend` 컨테이너
|
|
||||||
3. 외부 MySQL DB
|
|
||||||
|
|
||||||
현재는 더 이상 아래 항목이 없다.
|
|
||||||
|
|
||||||
1. `db` 서비스 없음
|
|
||||||
2. `db-bootstrap` 서비스 없음
|
|
||||||
3. `itam_mysql_data` 볼륨 없음
|
|
||||||
|
|
||||||
### 17.1 현재 실제 `docker-compose.yaml` 기준 backend 동작
|
|
||||||
|
|
||||||
현재 backend는 `.env`의 외부 DB 접속 정보를 그대로 사용한다.
|
|
||||||
|
|
||||||
즉, 아래 환경변수 매핑이 현재 기준이다.
|
|
||||||
|
|
||||||
1. `DB_HOST: ${DB_HOST}`
|
|
||||||
2. `DB_PORT: ${DB_PORT}`
|
|
||||||
3. `DB_USER: ${DB_USER}`
|
|
||||||
4. `DB_PASS: ${DB_PASS}`
|
|
||||||
5. `DB_NAME: ${DB_NAME}`
|
|
||||||
|
|
||||||
`PORT: 3000`만 Compose에서 고정한다.
|
|
||||||
|
|
||||||
### 17.2 현재 실제 기동 구조
|
|
||||||
|
|
||||||
현재 스택 기동 순서는 단순하다.
|
|
||||||
|
|
||||||
1. `backend` 기동
|
|
||||||
2. `frontend` 기동
|
|
||||||
3. backend는 외부 DB에 직접 접속
|
|
||||||
4. frontend는 `http://backend:3000`으로 프록시
|
|
||||||
|
|
||||||
즉, 현재는 DB 컨테이너 초기화 단계나 bootstrap 단계가 존재하지 않는다.
|
|
||||||
|
|
||||||
### 17.3 현재 기준 첫 실행 체크리스트
|
|
||||||
|
|
||||||
오늘 기준으로는 아래 순서가 맞다.
|
|
||||||
|
|
||||||
1. `.env`에 외부 DB 접속 정보 입력
|
|
||||||
2. `start_docker_wsl.ps1` 또는 `start_docker_wsl.bat` 실행
|
|
||||||
3. `http://localhost:3000/api/assets/master`가 200인지 확인
|
|
||||||
4. `http://localhost:8080/api/assets/master`가 200인지 확인
|
|
||||||
5. 브라우저에서 `http://localhost:8080` 접속 후 데이터 표시 확인
|
|
||||||
|
|
||||||
### 17.4 이 문서에서 현재 유효한 부분과 과거 이력 부분
|
|
||||||
|
|
||||||
현재도 그대로 유효한 내용은 아래다.
|
|
||||||
|
|
||||||
1. WSL2 기반 실행 방식
|
|
||||||
2. `start_docker_wsl.ps1` / `stop_docker_wsl.ps1` 사용 방식
|
|
||||||
3. `server.js`에서 Compose 환경변수가 `.env`보다 우선되도록 `dotenv.config()`를 유지해야 한다는 점
|
|
||||||
4. `vite.config.ts`에서 프록시 타깃을 환경변수화해야 한다는 점
|
|
||||||
|
|
||||||
현재는 과거 이력으로만 읽어야 하는 내용은 아래다.
|
|
||||||
|
|
||||||
1. 내부 `db` 서비스 설명
|
|
||||||
2. `db-bootstrap` 설명
|
|
||||||
3. `itam_mysql_data` 볼륨 설명
|
|
||||||
4. 내부 DB 재초기화 절차
|
|
||||||
5. 내부 테이블 확인 절차
|
|
||||||
|
|
||||||
### 17.5 현재 최종 한 줄 요약
|
|
||||||
|
|
||||||
오늘 날짜 기준 현재 저장소의 실사용 Compose 구조는 `frontend + backend + external DB`이며, 이전의 내부 DB/bootstrap 구조는 역사적으로 한 번 사용했던 임시 해결책으로만 남아 있다.
|
|
||||||
730
doc_readme3.md
730
doc_readme3.md
@@ -1,730 +0,0 @@
|
|||||||
# ITAM Linux 운영 배포 가이드
|
|
||||||
|
|
||||||
## 1. 문서 목적
|
|
||||||
|
|
||||||
이 문서는 현재 ITAM 저장소를 기준으로 Linux 환경에서 운영 배포하는 방법을 정리한 가이드다.
|
|
||||||
|
|
||||||
핵심 전제는 아래와 같다.
|
|
||||||
|
|
||||||
1. 저장소 구조를 크게 재편하지 않는다.
|
|
||||||
2. 현재 워크스페이스 기준 파일 구조를 그대로 활용한다.
|
|
||||||
3. `docker-compose.test.yaml`과 `docker-compose.prod.yaml` 모두 현재 저장소 루트 기준으로 동작한다.
|
|
||||||
4. DB는 Docker 내부가 아니라 외부 MySQL을 사용한다.
|
|
||||||
|
|
||||||
즉, 이 문서는 `/srv/itam` 같은 별도 운영 디렉터리 구조를 강제하는 문서가 아니라, 현재 저장소 구조를 기준으로 운영 전환하는 방법을 설명한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 2. 현재 운영 관련 파일
|
|
||||||
|
|
||||||
현재 저장소에서 운영 전환과 직접 관련된 파일은 아래와 같다.
|
|
||||||
|
|
||||||
1. `docker-compose.yaml`
|
|
||||||
2. `docker-compose.test.yaml`
|
|
||||||
3. `docker-compose.prod.yaml`
|
|
||||||
4. `Dockerfile.frontend.prod`
|
|
||||||
5. `Dockerfile.backend.prod`
|
|
||||||
6. `docker/nginx/default.conf`
|
|
||||||
7. `docker/frontend/default.conf`
|
|
||||||
8. `.env.example`
|
|
||||||
9. `server.js`
|
|
||||||
|
|
||||||
각 파일의 역할은 아래와 같다.
|
|
||||||
|
|
||||||
1. `docker-compose.yaml`: 개발 재현용 구성
|
|
||||||
2. `docker-compose.test.yaml`: 운영형 Dockerfile과 reverse proxy 구조를 로컬에서 검증하는 테스트용 구성
|
|
||||||
3. `docker-compose.prod.yaml`: 현재 저장소 기준 운영용 구성
|
|
||||||
4. `Dockerfile.frontend.prod`: 프런트 정적 빌드 및 Nginx 서빙 이미지 정의
|
|
||||||
5. `Dockerfile.backend.prod`: 백엔드 API 운영 이미지 정의
|
|
||||||
6. `docker/nginx/default.conf`: reverse proxy 설정
|
|
||||||
7. `docker/frontend/default.conf`: frontend 컨테이너 내부 정적 파일 서빙 설정
|
|
||||||
8. `.env.example`: 운영/테스트 환경변수 템플릿
|
|
||||||
9. `server.js`: `/health`, `/ready` 엔드포인트 포함
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 3. 현재 기준 운영 아키텍처
|
|
||||||
|
|
||||||
현재 구조에서의 요청 흐름은 아래와 같다.
|
|
||||||
|
|
||||||
1. 외부 요청은 Nginx 컨테이너로 들어온다.
|
|
||||||
2. `/` 요청은 frontend 컨테이너로 전달된다.
|
|
||||||
3. `/api/`와 `/uploads/` 요청은 backend 컨테이너로 전달된다.
|
|
||||||
4. backend는 외부 MySQL에 연결한다.
|
|
||||||
5. `uploads`, `map_config.json`, `.env`, 로그는 현재 저장소 기준 상대 경로를 사용한다.
|
|
||||||
|
|
||||||
```mermaid
|
|
||||||
flowchart LR
|
|
||||||
U["User Browser"] --> RP["Reverse Proxy Nginx 80"]
|
|
||||||
RP -->|root| FE["Frontend Container Nginx Static 80"]
|
|
||||||
RP -->|api| BE["Backend Container Node 3000"]
|
|
||||||
RP -->|uploads| BE
|
|
||||||
BE --> DB["External MySQL 3306"]
|
|
||||||
BE --> UP["./uploads"]
|
|
||||||
BE --> CFG["./map_config.json"]
|
|
||||||
BE --> ENV["./.env"]
|
|
||||||
linkStyle default stroke:#d32f2f,stroke-width:2px;
|
|
||||||
```
|
|
||||||
|
|
||||||
현재 저장소 기준 파일/컨테이너 관계는 아래와 같다.
|
|
||||||
|
|
||||||
```mermaid
|
|
||||||
flowchart TB
|
|
||||||
subgraph REPO["Current Repository Root"]
|
|
||||||
ENV[".env"]
|
|
||||||
UP["uploads/"]
|
|
||||||
MAP["map_config.json"]
|
|
||||||
LOGS["logs/nginx/"]
|
|
||||||
CONF["docker/nginx/default.conf"]
|
|
||||||
end
|
|
||||||
|
|
||||||
subgraph CTR["Containers"]
|
|
||||||
NGINX["itam-nginx"]
|
|
||||||
FRONT["itam-frontend"]
|
|
||||||
BACK["itam-backend"]
|
|
||||||
end
|
|
||||||
|
|
||||||
DB["External MySQL"]
|
|
||||||
|
|
||||||
CONF --> NGINX
|
|
||||||
LOGS --> NGINX
|
|
||||||
ENV --> BACK
|
|
||||||
UP --> BACK
|
|
||||||
MAP --> BACK
|
|
||||||
NGINX --> FRONT
|
|
||||||
NGINX --> BACK
|
|
||||||
BACK --> DB
|
|
||||||
linkStyle default stroke:#d32f2f,stroke-width:2px;
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 4. 개발용, 테스트용, 운영용 차이
|
|
||||||
|
|
||||||
### 4.1 `docker-compose.yaml`
|
|
||||||
|
|
||||||
용도:
|
|
||||||
|
|
||||||
1. 개발 재현
|
|
||||||
2. 소스 수정과 빠른 확인
|
|
||||||
3. Vite dev server 기반 실행
|
|
||||||
|
|
||||||
특징:
|
|
||||||
|
|
||||||
1. bind mount 중심
|
|
||||||
2. 프런트는 개발 서버 기반
|
|
||||||
3. 운영 배포보다는 개발 생산성에 초점
|
|
||||||
|
|
||||||
### 4.2 `docker-compose.test.yaml`
|
|
||||||
|
|
||||||
용도:
|
|
||||||
|
|
||||||
1. 운영형 Dockerfile 테스트
|
|
||||||
2. reverse proxy 동작 테스트
|
|
||||||
3. 로컬/WSL에서 8080 포트 검증
|
|
||||||
|
|
||||||
특징:
|
|
||||||
|
|
||||||
1. frontend/backend를 `build`로 생성
|
|
||||||
2. nginx는 8080 포트로 노출
|
|
||||||
3. 현재 저장소 상대 경로를 그대로 사용
|
|
||||||
|
|
||||||
### 4.3 `docker-compose.prod.yaml`
|
|
||||||
|
|
||||||
용도:
|
|
||||||
|
|
||||||
1. 현재 저장소 구조 기준 운영 배포
|
|
||||||
2. 운영 모드 환경변수와 health check 사용
|
|
||||||
3. 현재 구조를 바꾸지 않고 운영 전환
|
|
||||||
|
|
||||||
특징:
|
|
||||||
|
|
||||||
1. frontend/backend를 `build + image` 방식으로 정의
|
|
||||||
2. `.env`, `uploads`, `map_config.json`, `logs/nginx`를 현재 저장소 기준 상대 경로로 사용
|
|
||||||
3. nginx는 80 포트를 사용
|
|
||||||
4. backend는 `NODE_ENV=production`으로 실행
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 5. 운영 파일 구조 기준
|
|
||||||
|
|
||||||
현재 운영 배포는 저장소 루트를 기준으로 아래 구조를 전제로 한다.
|
|
||||||
|
|
||||||
```text
|
|
||||||
itam/
|
|
||||||
.env
|
|
||||||
.env.example
|
|
||||||
docker-compose.prod.yaml
|
|
||||||
docker-compose.test.yaml
|
|
||||||
Dockerfile.frontend.prod
|
|
||||||
Dockerfile.backend.prod
|
|
||||||
map_config.json
|
|
||||||
uploads/
|
|
||||||
logs/
|
|
||||||
nginx/
|
|
||||||
docker/
|
|
||||||
nginx/
|
|
||||||
default.conf
|
|
||||||
frontend/
|
|
||||||
default.conf
|
|
||||||
```
|
|
||||||
|
|
||||||
운영에서 실제로 중요한 경로는 아래 네 가지다.
|
|
||||||
|
|
||||||
1. `./.env`
|
|
||||||
2. `./uploads`
|
|
||||||
3. `./map_config.json`
|
|
||||||
4. `./logs/nginx`
|
|
||||||
|
|
||||||
즉, 현재 구조를 유지하려면 이 경로들이 항상 함께 관리되어야 한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 6. 운영 환경변수 정책
|
|
||||||
|
|
||||||
현재 기준 운영 환경변수는 저장소 루트의 `.env`를 사용한다.
|
|
||||||
|
|
||||||
`.env.example` 기준 예시는 아래와 같다.
|
|
||||||
|
|
||||||
```env
|
|
||||||
DB_HOST=172.16.8.151
|
|
||||||
DB_PORT=3306
|
|
||||||
DB_USER=itam_admin
|
|
||||||
DB_PASS=change-this
|
|
||||||
DB_NAME=itam
|
|
||||||
|
|
||||||
NODE_ENV=production
|
|
||||||
PORT=3000
|
|
||||||
LOG_LEVEL=info
|
|
||||||
```
|
|
||||||
|
|
||||||
운영 원칙은 아래와 같다.
|
|
||||||
|
|
||||||
1. 실제 운영 비밀번호가 들어간 `.env`는 Git에 올리지 않는다.
|
|
||||||
2. `.env.example`은 템플릿으로만 사용한다.
|
|
||||||
3. 운영 서버에서는 `.env` 파일 권한을 제한한다.
|
|
||||||
4. 운영 DB 계정과 개발 DB 계정은 분리한다.
|
|
||||||
|
|
||||||
권장 권한 예시는 아래와 같다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
chmod 600 .env
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 7. Frontend 운영 이미지 기준
|
|
||||||
|
|
||||||
`Dockerfile.frontend.prod`는 multi-stage build를 사용한다.
|
|
||||||
|
|
||||||
구성은 아래와 같다.
|
|
||||||
|
|
||||||
1. builder 단계에서 `npm ci` 수행
|
|
||||||
2. `npm run build` 수행
|
|
||||||
3. 결과물을 Nginx 이미지에 복사
|
|
||||||
4. `docker/frontend/default.conf`로 정적 파일 서빙
|
|
||||||
|
|
||||||
운영 관점에서의 장점은 아래와 같다.
|
|
||||||
|
|
||||||
1. runtime 이미지에 build 결과물만 포함된다.
|
|
||||||
2. dev server 없이 정적 파일만 제공한다.
|
|
||||||
3. frontend 컨테이너 자체도 health check 가능하다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 8. Backend 운영 이미지 기준
|
|
||||||
|
|
||||||
`Dockerfile.backend.prod`는 아래 기준으로 작성되어 있다.
|
|
||||||
|
|
||||||
1. `NODE_ENV=production`
|
|
||||||
2. production dependency만 설치
|
|
||||||
3. `appuser` 비루트 사용자 사용
|
|
||||||
4. `dumb-init` 사용
|
|
||||||
5. `/health` health check 사용
|
|
||||||
|
|
||||||
backend 컨테이너는 아래 자원을 사용한다.
|
|
||||||
|
|
||||||
1. `./.env`
|
|
||||||
2. `./uploads`
|
|
||||||
3. `./map_config.json`
|
|
||||||
4. 외부 MySQL
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 9. Reverse Proxy 기준
|
|
||||||
|
|
||||||
`docker/nginx/default.conf`는 현재 아래처럼 동작한다.
|
|
||||||
|
|
||||||
1. `/` -> `frontend:80`
|
|
||||||
2. `/api/` -> `backend:3000`
|
|
||||||
3. `/uploads/` -> `backend:3000`
|
|
||||||
|
|
||||||
추가로 아래 설정을 포함한다.
|
|
||||||
|
|
||||||
1. 기본 보안 헤더
|
|
||||||
2. access/error 로그
|
|
||||||
3. gzip 설정
|
|
||||||
4. health endpoint
|
|
||||||
|
|
||||||
중요한 점은, 현재 운영 기준에서 외부 사용자가 직접 frontend 컨테이너에 붙는 것이 아니라 반드시 nginx를 통해 들어간다는 점이다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 10. 현재 기준 운영 배포 절차
|
|
||||||
|
|
||||||
### 10.1 사전 점검
|
|
||||||
|
|
||||||
아래 항목을 먼저 확인한다.
|
|
||||||
|
|
||||||
1. `.env` 파일 존재 여부
|
|
||||||
2. `uploads/` 디렉터리 존재 여부
|
|
||||||
3. `logs/nginx/` 디렉터리 존재 여부
|
|
||||||
4. `map_config.json` 존재 여부
|
|
||||||
5. 외부 DB 접근 가능 여부
|
|
||||||
|
|
||||||
예시:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
ls -la .env
|
|
||||||
ls -la uploads
|
|
||||||
ls -la logs/nginx
|
|
||||||
ls -la map_config.json
|
|
||||||
```
|
|
||||||
|
|
||||||
### 10.2 Compose 검증
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker compose -f docker-compose.prod.yaml config
|
|
||||||
```
|
|
||||||
|
|
||||||
### 10.3 운영 기동
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker compose -f docker-compose.prod.yaml up -d --build
|
|
||||||
docker compose -f docker-compose.prod.yaml ps
|
|
||||||
```
|
|
||||||
|
|
||||||
### 10.4 운영 중지
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker compose -f docker-compose.prod.yaml down
|
|
||||||
```
|
|
||||||
|
|
||||||
### 10.5 운영/배포 분기 흐름
|
|
||||||
|
|
||||||
현재 운영 반영은 자동 push 배포가 아니라, Gitea에 올라간 커밋을 기준으로 수동 workflow를 실행하는 방식이다.
|
|
||||||
|
|
||||||
즉 아래 원칙으로 이해하면 된다.
|
|
||||||
|
|
||||||
1. 로컬 수정본을 서버에 직접 복사하지 않는다.
|
|
||||||
2. 반드시 Gitea에 올라간 커밋을 기준으로 배포한다.
|
|
||||||
3. 운영 반영은 `.gitea/workflows/itam_production_deploy.yml` 수동 실행으로 진행한다.
|
|
||||||
4. 실패 후 재배포는 실패 지점에 따라 수정 위치가 달라진다.
|
|
||||||
|
|
||||||
운영 반영은 크게 세 상황으로 나뉜다.
|
|
||||||
|
|
||||||
1. 최초 운영 서버 구축 후 첫 배포
|
|
||||||
2. 코드 수정 후 일반 재배포
|
|
||||||
3. 배포 실패 또는 검증 실패 후 재배포
|
|
||||||
|
|
||||||
```mermaid
|
|
||||||
flowchart TD
|
|
||||||
START["배포 필요 발생"] --> CASE{"어떤 상황인가?"}
|
|
||||||
|
|
||||||
CASE -->|초기 구축| INIT["초기 운영 배포 준비"]
|
|
||||||
CASE -->|수정 반영| CHANGE["수정 후 재배포 준비"]
|
|
||||||
CASE -->|실패 후 재시도| RETRY["실패 원인 분석 후 재배포 준비"]
|
|
||||||
|
|
||||||
INIT --> INIT1["운영 서버 Docker / compose 확인"]
|
|
||||||
INIT1 --> INIT2["Gitea Variables / Secrets 등록"]
|
|
||||||
INIT2 --> INIT3["map_config.json / uploads 초기 데이터 준비"]
|
|
||||||
INIT3 --> MANUAL["Gitea에서 수동 배포 workflow 실행"]
|
|
||||||
|
|
||||||
CHANGE --> CHANGE1["로컬 수정 및 테스트"]
|
|
||||||
CHANGE1 --> CHANGE2["Gitea 커밋 / push"]
|
|
||||||
CHANGE2 --> CHANGE3["Code Check / Docker Build Check 통과"]
|
|
||||||
CHANGE3 --> MANUAL
|
|
||||||
|
|
||||||
RETRY --> RETRY1{"어디서 실패했는가?"}
|
|
||||||
RETRY1 -->|코드 체크 실패| FIX1["코드 또는 설정 수정"]
|
|
||||||
RETRY1 -->|배포 단계 실패| FIX2["서버 / 변수 / 권한 / 네트워크 수정"]
|
|
||||||
RETRY1 -->|Smoke Check 실패| FIX3["앱 기동 상태 / 프록시 / DB 상태 수정"]
|
|
||||||
FIX1 --> CHANGE2
|
|
||||||
FIX2 --> MANUAL
|
|
||||||
FIX3 --> MANUAL
|
|
||||||
|
|
||||||
MANUAL --> BACKUP["기존 운영 상태가 있으면 배포 전 백업"]
|
|
||||||
BACKUP --> DEPLOY["운영 서버 반영 수행"]
|
|
||||||
DEPLOY --> RESULT{"최종 검증 통과?"}
|
|
||||||
RESULT -->|예| DONE["운영 반영 완료"]
|
|
||||||
RESULT -->|아니오| RETRY
|
|
||||||
linkStyle default stroke:#d32f2f,stroke-width:2px;
|
|
||||||
```
|
|
||||||
|
|
||||||
### 10.6 최초 운영 배포 플로우
|
|
||||||
|
|
||||||
최초 배포에서는 코드보다 운영 환경 준비가 먼저다.
|
|
||||||
|
|
||||||
순서는 아래와 같다.
|
|
||||||
|
|
||||||
1. 운영 서버에 Docker Engine과 `docker compose`를 설치한다.
|
|
||||||
2. 운영 서버에서 Gitea 저장소에 접근 가능한 SSH 키를 준비한다.
|
|
||||||
3. Gitea repository Variables / Secrets를 등록한다.
|
|
||||||
4. `PROD_DEPLOY_PATH` 경로를 확정한다.
|
|
||||||
5. `PROD_BACKUP_ROOT` 경로를 `PROD_DEPLOY_PATH` 바깥으로 확정한다.
|
|
||||||
6. `map_config.json`, `uploads/` 초기 데이터를 준비한다.
|
|
||||||
7. Gitea에서 `itam_production_deploy.yml`을 수동 실행한다.
|
|
||||||
8. 배포 후 `docker compose ps`, `/health`, `/`, `/ready`를 확인한다.
|
|
||||||
|
|
||||||
즉 최초 배포는 아래 순서다.
|
|
||||||
|
|
||||||
```text
|
|
||||||
서버 준비 완료
|
|
||||||
-> Gitea 변수 / 시크릿 등록 완료
|
|
||||||
-> 백업 경로 확정 완료
|
|
||||||
-> 초기 데이터 준비 완료
|
|
||||||
-> 수동 배포 실행
|
|
||||||
```
|
|
||||||
|
|
||||||
### 10.7 수정 후 일반 재배포 플로우
|
|
||||||
|
|
||||||
일반적인 수정 반영은 아래 흐름이다.
|
|
||||||
|
|
||||||
1. 개발자가 로컬에서 코드 또는 설정을 수정한다.
|
|
||||||
2. 로컬에서 필요한 테스트를 수행한다.
|
|
||||||
3. 변경사항을 Gitea에 커밋 후 push 한다.
|
|
||||||
4. `itam_code_check.yml`이 빌드와 compose 문법을 검사한다.
|
|
||||||
5. `itam_docker_build_check.yml`이 운영용 이미지 빌드 가능 여부를 검사한다.
|
|
||||||
6. 두 검증이 통과하면 운영자가 Gitea에서 `itam_production_deploy.yml`을 수동 실행한다.
|
|
||||||
7. 기존 운영 상태가 있으면 배포 전 백업을 먼저 수행한다.
|
|
||||||
8. 운영 서버가 최신 커밋으로 동기화되고 컨테이너가 다시 올라온다.
|
|
||||||
9. smoke check 통과 여부를 확인한다.
|
|
||||||
|
|
||||||
```mermaid
|
|
||||||
flowchart LR
|
|
||||||
DEV["로컬 수정"] --> TEST["로컬 확인"]
|
|
||||||
TEST --> PUSH["커밋 / push"]
|
|
||||||
PUSH --> CODE["ITAM Code Check"]
|
|
||||||
CODE --> BUILD["ITAM Docker Build Check"]
|
|
||||||
BUILD --> GATE{"검증 통과?"}
|
|
||||||
GATE -->|예| RUN["Gitea에서 수동 배포 실행"]
|
|
||||||
GATE -->|아니오| FIX["로컬 수정 후 재커밋"]
|
|
||||||
FIX --> PUSH
|
|
||||||
RUN --> BACKUP["배포 전 백업"]
|
|
||||||
BACKUP --> PROD["운영 서버 배포"]
|
|
||||||
PROD --> SMOKE{"Smoke Check 통과?"}
|
|
||||||
SMOKE -->|예| OK["배포 완료"]
|
|
||||||
SMOKE -->|아니오| FIXDEPLOY["원인 수정 후 재배포"]
|
|
||||||
FIXDEPLOY --> RUN
|
|
||||||
linkStyle default stroke:#d32f2f,stroke-width:2px;
|
|
||||||
```
|
|
||||||
|
|
||||||
### 10.8 수동 배포 workflow 내부 실행 순서
|
|
||||||
|
|
||||||
Gitea에서 `itam_production_deploy.yml`을 수동 실행하면 내부적으로는 아래 순서로 진행된다.
|
|
||||||
|
|
||||||
1. SSH agent를 설정한다.
|
|
||||||
2. 필수 Variables / Secrets가 모두 있는지 확인한다.
|
|
||||||
3. 운영용 `.env.deploy` 파일을 생성한다.
|
|
||||||
4. 운영 서버에 접속한다.
|
|
||||||
5. `PROD_DEPLOY_PATH`를 생성한다.
|
|
||||||
6. 기존 운영 상태가 있으면 `make predeploy-backup`을 실행한다.
|
|
||||||
7. 저장소를 clone 또는 fetch 한다.
|
|
||||||
8. 선택한 브랜치의 최신 커밋으로 checkout, reset, clean 한다.
|
|
||||||
9. `uploads`, `logs/nginx` 디렉토리를 준비한다.
|
|
||||||
10. `.env.deploy`를 서버의 `.env`로 복사한다.
|
|
||||||
11. `docker compose -f docker-compose.prod.yaml config`를 수행한다.
|
|
||||||
12. `docker compose -f docker-compose.prod.yaml up -d --build`를 수행한다.
|
|
||||||
13. `docker compose ps`를 확인한다.
|
|
||||||
14. `/health`, `/`, backend `/ready` smoke check를 수행한다.
|
|
||||||
|
|
||||||
```mermaid
|
|
||||||
flowchart TD
|
|
||||||
A["수동 배포 시작"] --> B["SSH agent 설정"]
|
|
||||||
B --> C["Variables / Secrets 검증"]
|
|
||||||
C --> D{"필수 값 누락 여부"}
|
|
||||||
D -->|예| E["즉시 실패 후 설정 보완"]
|
|
||||||
D -->|아니오| F[".env.deploy 생성"]
|
|
||||||
F --> G["운영 서버 SSH 접속"]
|
|
||||||
G --> H["배포 경로 생성"]
|
|
||||||
H --> I["기존 운영 상태가 있으면 make predeploy-backup"]
|
|
||||||
I --> J["git clone 또는 fetch"]
|
|
||||||
J --> K["지정 브랜치 checkout / reset / clean"]
|
|
||||||
K --> L["uploads / logs/nginx 준비"]
|
|
||||||
L --> M[".env 업로드 및 권한 설정"]
|
|
||||||
M --> N["compose config 검증"]
|
|
||||||
N --> O{"compose config 성공?"}
|
|
||||||
O -->|아니오| P["설정 수정 후 재실행"]
|
|
||||||
O -->|예| Q["compose up -d --build"]
|
|
||||||
Q --> R["docker compose ps 확인"]
|
|
||||||
R --> S["/health, /, /ready smoke check"]
|
|
||||||
S --> T{"smoke check 성공?"}
|
|
||||||
T -->|예| U["운영 배포 완료"]
|
|
||||||
T -->|아니오| V["원인 분석 후 재배포"]
|
|
||||||
linkStyle default stroke:#d32f2f,stroke-width:2px;
|
|
||||||
```
|
|
||||||
|
|
||||||
### 10.9 실패 후 검증 및 재배포 플로우
|
|
||||||
|
|
||||||
실패가 났다고 해서 같은 방식으로 바로 다시 배포하면 안 된다.
|
|
||||||
|
|
||||||
실패 지점별 판단은 아래처럼 나눈다.
|
|
||||||
|
|
||||||
1. Code Check 실패: TypeScript, build, compose 문법 문제를 먼저 수정한다.
|
|
||||||
2. Docker Build Check 실패: Dockerfile, 정적 자산 복사, 운영 빌드 컨텍스트 문제를 수정한다.
|
|
||||||
3. Deploy 단계 실패: SSH, Gitea 변수, 서버 권한, 경로, 백업 경로, git 접근, Docker 권한을 수정한다.
|
|
||||||
4. Smoke Check 실패: Nginx 프록시, backend readiness, 외부 DB 연결, 앱 런타임 오류를 수정한다.
|
|
||||||
|
|
||||||
즉 재배포 전 판단 기준은 아래와 같다.
|
|
||||||
|
|
||||||
```text
|
|
||||||
CI 실패 -> 로컬 코드 / 설정 수정 후 재커밋
|
|
||||||
배포 실패 -> 서버 환경 또는 배포 설정 수정 후 수동 재실행
|
|
||||||
Smoke Check 실패 -> 앱 / 프록시 / DB 상태 수정 후 수동 재실행
|
|
||||||
```
|
|
||||||
|
|
||||||
운영 관점에서는 아래 순서를 지키는 것이 안전하다.
|
|
||||||
|
|
||||||
1. 실패 지점 확인
|
|
||||||
2. 원인 수정
|
|
||||||
3. 같은 실패가 다시 나는지 좁은 범위로 재검증
|
|
||||||
4. 그 다음에만 수동 배포 재실행
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 11. 테스트 배포 절차
|
|
||||||
|
|
||||||
운영형 구성을 먼저 검증하려면 아래처럼 진행한다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker compose -f docker-compose.test.yaml up -d --build
|
|
||||||
docker compose -f docker-compose.test.yaml ps
|
|
||||||
```
|
|
||||||
|
|
||||||
접속 기준은 아래와 같다.
|
|
||||||
|
|
||||||
1. `http://localhost:8080` -> nginx reverse proxy
|
|
||||||
2. `http://localhost:3000/health` -> backend health 확인
|
|
||||||
|
|
||||||
테스트용은 운영과 매우 유사하지만, 외부 노출 포트와 일부 실행 목적이 다르다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 12. Health Check 및 상태 판정
|
|
||||||
|
|
||||||
backend에는 아래 두 엔드포인트가 있다.
|
|
||||||
|
|
||||||
1. `/health`
|
|
||||||
2. `/ready`
|
|
||||||
|
|
||||||
판정 기준은 아래와 같다.
|
|
||||||
|
|
||||||
1. `/health = 200`, `/ready = 200`: 정상 서비스 가능 상태
|
|
||||||
2. `/health = 200`, `/ready = 503`: 프로세스는 살아 있으나 DB 또는 외부 의존성 미준비
|
|
||||||
|
|
||||||
확인 예시는 아래와 같다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
curl http://localhost:3000/health
|
|
||||||
curl http://localhost:3000/ready
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 13. 운영 점검 체크리스트
|
|
||||||
|
|
||||||
### 13.1 애플리케이션
|
|
||||||
|
|
||||||
1. `docker compose -f docker-compose.prod.yaml config` 통과 여부
|
|
||||||
2. frontend 이미지 빌드 성공 여부
|
|
||||||
3. backend 이미지 빌드 성공 여부
|
|
||||||
4. 모든 컨테이너가 `Up` 상태인지
|
|
||||||
5. 메인 화면이 정상 렌더링되는지
|
|
||||||
6. 데이터 조회가 정상 동작하는지
|
|
||||||
|
|
||||||
### 13.2 데이터 및 파일
|
|
||||||
|
|
||||||
1. `uploads/`에 쓰기 가능한지
|
|
||||||
2. `map_config.json`을 backend가 읽을 수 있는지
|
|
||||||
3. `.env` 파일 권한이 적절한지
|
|
||||||
4. `logs/nginx/`에 로그가 쌓이는지
|
|
||||||
|
|
||||||
### 13.3 네트워크
|
|
||||||
|
|
||||||
1. 외부 DB 접근 가능한지
|
|
||||||
2. nginx에서 backend upstream 연결이 되는지
|
|
||||||
3. `/api` 요청이 정상 응답하는지
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 14. 로그 및 장애 대응
|
|
||||||
|
|
||||||
기본 점검 명령은 아래와 같다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker compose -f docker-compose.prod.yaml ps
|
|
||||||
docker compose -f docker-compose.prod.yaml logs --tail=200 nginx
|
|
||||||
docker compose -f docker-compose.prod.yaml logs --tail=200 backend
|
|
||||||
docker compose -f docker-compose.prod.yaml logs --tail=200 frontend
|
|
||||||
```
|
|
||||||
|
|
||||||
장애 확인 순서는 아래가 좋다.
|
|
||||||
|
|
||||||
1. 컨테이너가 살아 있는지
|
|
||||||
2. nginx가 frontend/backend로 프록시하는지
|
|
||||||
3. backend가 DB에 붙는지
|
|
||||||
4. 업로드/설정 파일 권한 문제는 없는지
|
|
||||||
|
|
||||||
추가 확인 예시는 아래와 같다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
curl -I http://localhost/
|
|
||||||
curl http://localhost:3000/health
|
|
||||||
curl http://localhost:3000/ready
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 15. 백업 기준
|
|
||||||
|
|
||||||
현재 구조 기준 최소 백업 대상은 아래와 같다.
|
|
||||||
|
|
||||||
1. `.env`
|
|
||||||
2. `uploads/`
|
|
||||||
3. `map_config.json`
|
|
||||||
4. 외부 MySQL 데이터베이스
|
|
||||||
|
|
||||||
현재 저장소에는 운영 백업을 직접 실행할 수 있도록 `Makefile`과 `scripts/backup.sh`가 추가되어 있다.
|
|
||||||
|
|
||||||
기본 원칙은 아래와 같다.
|
|
||||||
|
|
||||||
1. DB dump와 런타임 파일 백업을 분리해서 실행할 수 있어야 한다.
|
|
||||||
2. 기본 백업 산출물은 `backups/` 디렉터리에 쌓는다.
|
|
||||||
3. DB 접속 정보는 `.env`를 기준으로 읽는다.
|
|
||||||
4. 오래된 백업 파일은 보존 주기에 따라 정리한다.
|
|
||||||
|
|
||||||
백업 실행 흐름은 아래와 같다.
|
|
||||||
|
|
||||||
```mermaid
|
|
||||||
flowchart TD
|
|
||||||
START["운영 백업 실행"] --> TARGET{"무엇을 백업할 것인가?"}
|
|
||||||
TARGET -->|DB| DB["make db-dump"]
|
|
||||||
TARGET -->|운영 파일| FILES["make files-backup"]
|
|
||||||
TARGET -->|전체| FULL["make full-backup"]
|
|
||||||
DB --> OUT1["backups/db/*.sql.gz"]
|
|
||||||
FILES --> OUT2["backups/files/*.tar.gz"]
|
|
||||||
FULL --> OUT1
|
|
||||||
FULL --> OUT2
|
|
||||||
OUT1 --> CLEAN["make cleanup-backups"]
|
|
||||||
OUT2 --> CLEAN
|
|
||||||
linkStyle default stroke:#d32f2f,stroke-width:2px;
|
|
||||||
```
|
|
||||||
|
|
||||||
### 15.1 Make 명령 기준
|
|
||||||
|
|
||||||
사용 가능한 기본 명령은 아래와 같다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
make db-dump
|
|
||||||
make files-backup
|
|
||||||
make full-backup
|
|
||||||
make cleanup-backups
|
|
||||||
```
|
|
||||||
|
|
||||||
각 명령의 역할은 아래와 같다.
|
|
||||||
|
|
||||||
1. `make db-dump`: `.env` 기준 MySQL dump를 `backups/db/`에 `.sql.gz`로 저장
|
|
||||||
2. `make files-backup`: `.env`, `uploads/`, `map_config.json`을 `backups/files/`에 `.tar.gz`로 저장
|
|
||||||
3. `make full-backup`: DB dump와 파일 백업을 한 번에 수행
|
|
||||||
4. `make cleanup-backups`: 기본 14일이 지난 백업 파일 정리
|
|
||||||
|
|
||||||
### 15.2 실행 예시
|
|
||||||
|
|
||||||
가장 단순한 전체 백업 예시는 아래와 같다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
make full-backup
|
|
||||||
```
|
|
||||||
|
|
||||||
DB dump만 별도로 실행하려면 아래처럼 사용한다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
make db-dump
|
|
||||||
```
|
|
||||||
|
|
||||||
운영 파일만 묶으려면 아래처럼 사용한다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
make files-backup
|
|
||||||
```
|
|
||||||
|
|
||||||
보존 주기를 30일로 바꿔 정리하려면 아래처럼 사용한다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
make cleanup-backups RETENTION_DAYS=30
|
|
||||||
```
|
|
||||||
|
|
||||||
백업 경로를 별도 디스크나 마운트 경로로 바꾸려면 아래처럼 사용한다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
make full-backup BACKUP_ROOT=/opt/itam-backups
|
|
||||||
```
|
|
||||||
|
|
||||||
### 15.3 현재 스크립트가 실제로 백업하는 대상
|
|
||||||
|
|
||||||
현재 `scripts/backup.sh`는 아래 규칙으로 동작한다.
|
|
||||||
|
|
||||||
1. `.env` 파일에서 `DB_HOST`, `DB_PORT`, `DB_USER`, `DB_PASS`, `DB_NAME`을 읽는다.
|
|
||||||
2. `mysqldump --single-transaction --quick --routines --triggers` 옵션으로 dump를 생성한다.
|
|
||||||
3. DB dump는 gzip 압축본으로 저장한다.
|
|
||||||
4. 파일 백업은 `.env`, `uploads/`, `map_config.json` 중 실제로 존재하는 항목만 묶는다.
|
|
||||||
5. 백업 정리는 `find ... -mtime` 기준으로 수행한다.
|
|
||||||
|
|
||||||
즉 현재 스크립트는 운영 서버 또는 백업 서버에서 바로 실행 가능한 최소 백업 도구로 보면 된다.
|
|
||||||
|
|
||||||
### 15.4 운영 사용 권장 방식
|
|
||||||
|
|
||||||
운영에서는 아래 방식이 가장 현실적이다.
|
|
||||||
|
|
||||||
1. 매일 새벽 cron 또는 systemd timer로 `make full-backup` 실행
|
|
||||||
2. 백업 완료 후 `make cleanup-backups` 실행
|
|
||||||
3. `backups/` 또는 별도 `BACKUP_ROOT` 경로를 NAS 또는 외부 백업 스토리지로 추가 복사
|
|
||||||
4. 최소 월 1회 restore 테스트 수행
|
|
||||||
|
|
||||||
가장 단순한 예시는 아래와 같다.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
make full-backup BACKUP_ROOT=/opt/itam-backups
|
|
||||||
make cleanup-backups BACKUP_ROOT=/opt/itam-backups RETENTION_DAYS=30
|
|
||||||
```
|
|
||||||
|
|
||||||
DB 백업 자체는 여전히 DB 서버 정책과 함께 관리하는 것이 가장 안전하지만, 현재 저장소 기준 운영 자동화 진입점은 위 `make` 명령으로 통일해도 된다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 16. 롤백 기준
|
|
||||||
|
|
||||||
현재 구조에서는 가장 단순한 롤백 방식이 아래와 같다.
|
|
||||||
|
|
||||||
1. 이전 정상 커밋 또는 파일 상태 확보
|
|
||||||
2. 이미지 재빌드 또는 이전 이미지 재사용
|
|
||||||
3. `docker compose -f docker-compose.prod.yaml up -d --build` 재실행
|
|
||||||
4. `/health`, `/ready`, 메인 화면, 핵심 API 재검증
|
|
||||||
|
|
||||||
즉, 현재 구조에서는 별도 디렉터리 재배치보다 현재 저장소 상태 관리와 compose 재기동이 롤백의 중심이 된다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 17. 결론
|
|
||||||
|
|
||||||
현재 ITAM 저장소는 별도 `/srv/itam` 구조로 옮기지 않아도, 지금 파일 구조를 유지한 채 운영형 배포 흐름으로 전환할 수 있다.
|
|
||||||
|
|
||||||
정리하면 아래와 같다.
|
|
||||||
|
|
||||||
1. test와 prod 모두 현재 저장소 구조 기준으로 통일한다.
|
|
||||||
2. `.env`, `uploads`, `map_config.json`, `logs/nginx`를 운영 핵심 경로로 본다.
|
|
||||||
3. reverse proxy는 현재 `docker/nginx/default.conf`를 기준으로 운영한다.
|
|
||||||
4. backend는 production 모드, health check, 외부 DB 연결 구조를 유지한다.
|
|
||||||
5. 큰 구조 변경 없이도 운영 전환이 가능하다.
|
|
||||||
|
|
||||||
남은 작업은 TLS, 로그 로테이션, CI/CD, 보안 점검을 현재 구조 기준으로 계속 보강하는 것이다.
|
|
||||||
@@ -40,7 +40,7 @@ services:
|
|||||||
ports:
|
ports:
|
||||||
- "9090:80"
|
- "9090:80"
|
||||||
volumes:
|
volumes:
|
||||||
- ./docker/nginx/default.conf:/etc/nginx/conf.d/default.conf:ro
|
- ./docker/nginx/default.conf:/etc/nginx/conf.d/default.conf
|
||||||
- ./logs/nginx:/var/log/nginx
|
- ./logs/nginx:/var/log/nginx
|
||||||
depends_on:
|
depends_on:
|
||||||
backend:
|
backend:
|
||||||
@@ -55,3 +55,19 @@ services:
|
|||||||
retries: 3
|
retries: 3
|
||||||
start_period: 20s
|
start_period: 20s
|
||||||
|
|
||||||
|
database:
|
||||||
|
image: mysql:latest
|
||||||
|
container_name: itam-mysql
|
||||||
|
ports:
|
||||||
|
- "3306:3306"
|
||||||
|
environment:
|
||||||
|
- MYSQL_ROOT_PASSWORD=itam1234 # 여기 직접 기입
|
||||||
|
- MYSQL_DATABASE=itam
|
||||||
|
- MYSQL_USER=itam
|
||||||
|
- MYSQL_PASSWORD=itam1234
|
||||||
|
volumes:
|
||||||
|
- ./mysql_data:/var/lib/mysql
|
||||||
|
restart: always
|
||||||
|
command:
|
||||||
|
- --character-set-server=utf8mb4
|
||||||
|
- --collation-server=utf8mb4_unicode_ci
|
||||||
@@ -1,330 +0,0 @@
|
|||||||
# ITAM 도커라이징 작업 태스크 정리
|
|
||||||
|
|
||||||
## 1. 문서 목적
|
|
||||||
|
|
||||||
이 문서는 ITAM 자산관리 시스템의 도커라이징 작업을 실제 실행 단위로 쪼개서 정리한 태스크 문서다.
|
|
||||||
|
|
||||||
이 문서의 목표는 아래와 같다.
|
|
||||||
|
|
||||||
1. 내일까지 보여줄 시연 범위를 기준으로 우선순위를 정한다.
|
|
||||||
2. 시연용 작업과 운영형 전환 작업을 분리한다.
|
|
||||||
3. 개발 담당자가 바로 실행할 수 있는 체크리스트를 제공한다.
|
|
||||||
|
|
||||||
관련 배경과 구조 분석은 [doc_readme.md](c:/Users/user/Desktop/안건%20파일/itam/doc_readme.md) 문서를 기준으로 한다.
|
|
||||||
|
|
||||||
현재 구현/검증 상태:
|
|
||||||
|
|
||||||
- `Dockerfile.frontend` 생성 완료
|
|
||||||
- `Dockerfile.backend` 생성 완료
|
|
||||||
- `docker-compose.yaml` 생성 완료
|
|
||||||
- `.dockerignore` 생성 완료
|
|
||||||
- WSL2 Ubuntu에서 `docker compose up --build -d` 검증 완료
|
|
||||||
- frontend 8080 응답 확인 완료
|
|
||||||
- backend `/api/assets/master` 응답 확인 완료
|
|
||||||
- 현재 DB는 external MySQL 기준이며, DB 컨테이너 추가 작업은 다음 단계로 남아 있음
|
|
||||||
|
|
||||||
## 2. 이번 작업의 최우선 목표
|
|
||||||
|
|
||||||
이번 도커라이징의 1차 목표는 "운영 배포 완료"가 아니라 아래 상태를 재현하는 것이다.
|
|
||||||
|
|
||||||
1. frontend 컨테이너가 정상 기동한다.
|
|
||||||
2. backend 컨테이너가 정상 기동한다.
|
|
||||||
3. backend가 기존 외부 MySQL 또는 MySQL 컨테이너에 정상 연결된다.
|
|
||||||
4. 브라우저에서 화면이 열린다.
|
|
||||||
5. 핵심 API 호출이 정상 동작한다.
|
|
||||||
6. 업로드 저장 경로가 유지된다.
|
|
||||||
7. 필요 시 DB까지 함께 포함된 재현 가능한 스택을 제공한다.
|
|
||||||
|
|
||||||
## 3. 작업 범위 구분
|
|
||||||
|
|
||||||
### 3.1 이번 시연 범위에 포함
|
|
||||||
|
|
||||||
- Dockerfile.frontend 초안 작성
|
|
||||||
- Dockerfile.backend 초안 작성
|
|
||||||
- docker-compose.yaml 작성
|
|
||||||
- `.dockerignore` 작성
|
|
||||||
- MySQL 컨테이너 추가 설계
|
|
||||||
- 초기 SQL dump 또는 init SQL 적재 방식 정의
|
|
||||||
- `uploads` 볼륨 처리
|
|
||||||
- `map_config.json` 영속성 처리 방식 반영
|
|
||||||
- 컨테이너 기동 및 접속 확인
|
|
||||||
- 핵심 API 및 화면 확인
|
|
||||||
|
|
||||||
### 3.2 이번 시연 범위에서 제외
|
|
||||||
|
|
||||||
- DB 전체 마이그레이션 자동화
|
|
||||||
- nginx 기반 운영 배포 구조
|
|
||||||
- 단일 이미지 운영 구조 전환
|
|
||||||
- CI/CD 연계
|
|
||||||
|
|
||||||
## 4. 선행 확인 태스크
|
|
||||||
|
|
||||||
아래 태스크는 실제 Docker 파일 작성 전에 먼저 확인해야 한다.
|
|
||||||
|
|
||||||
### Task 1. 외부 MySQL 접근 가능 여부 확인
|
|
||||||
|
|
||||||
- 목적: 컨테이너에서 외부 DB 접속이 가능한지 확인
|
|
||||||
- 확인 항목:
|
|
||||||
- DB_HOST 접근 가능 여부
|
|
||||||
- DB_PORT 3306 접속 가능 여부
|
|
||||||
- 계정 권한 정상 여부
|
|
||||||
- 완료 기준:
|
|
||||||
- backend 컨테이너 기준 DB 연결 에러가 발생하지 않음
|
|
||||||
|
|
||||||
### Task 2. 기준 스키마 상태 확인
|
|
||||||
|
|
||||||
- 목적: 현재 앱이 요구하는 테이블 구조가 실제 DB와 맞는지 확인
|
|
||||||
- 확인 항목:
|
|
||||||
- `asset_core`
|
|
||||||
- `asset_spec`
|
|
||||||
- `asset_location`
|
|
||||||
- `asset_remote`
|
|
||||||
- `asset_history`
|
|
||||||
- `hardware_components_master`
|
|
||||||
- `job_spec_standards`
|
|
||||||
- 완료 기준:
|
|
||||||
- `/api/assets/master` 호출 시 쿼리 에러가 발생하지 않음
|
|
||||||
|
|
||||||
### Task 3. 파일 영속성 대상 확인
|
|
||||||
|
|
||||||
- 목적: 컨테이너 재시작 이후에도 유지되어야 할 파일/폴더 식별
|
|
||||||
- 대상:
|
|
||||||
- `uploads`
|
|
||||||
- `map_config.json`
|
|
||||||
- 완료 기준:
|
|
||||||
- 볼륨 설계 대상이 명확하게 문서화됨
|
|
||||||
|
|
||||||
### Task 4. DB 기준 데이터 소스 확정
|
|
||||||
|
|
||||||
- 목적: MySQL 컨테이너 최초 기동 시 어떤 데이터로 초기화할지 결정
|
|
||||||
- 선택지:
|
|
||||||
- 기존 사내 DB에서 추출한 SQL dump 사용
|
|
||||||
- 정리된 스키마 SQL + seed SQL 사용
|
|
||||||
- 수동 import 절차 사용
|
|
||||||
- 완료 기준:
|
|
||||||
- `docker/mysql/init` 기준 적재 전략 또는 수동 복원 절차가 확정됨
|
|
||||||
|
|
||||||
## 5. 시연용 도커라이징 태스크
|
|
||||||
|
|
||||||
### Task 5. 프런트 Dockerfile 작성
|
|
||||||
|
|
||||||
- 목적: Vite 개발 서버를 컨테이너에서 구동
|
|
||||||
- 작업 내용:
|
|
||||||
- Node 20 계열 이미지 사용
|
|
||||||
- `package*.json` 복사 후 `npm install`
|
|
||||||
- 8080 포트 노출
|
|
||||||
- `npm run dev -- --host 0.0.0.0` 실행
|
|
||||||
- 산출물:
|
|
||||||
- `Dockerfile.frontend`
|
|
||||||
- 완료 기준:
|
|
||||||
- 컨테이너에서 8080 포트가 정상 listen 상태가 됨
|
|
||||||
|
|
||||||
### Task 6. 백엔드 Dockerfile 작성
|
|
||||||
|
|
||||||
- 목적: Express API 서버를 컨테이너에서 구동
|
|
||||||
- 작업 내용:
|
|
||||||
- Node 20 계열 이미지 사용
|
|
||||||
- `package*.json` 복사 후 `npm install`
|
|
||||||
- 3000 포트 노출
|
|
||||||
- `npm run server` 실행
|
|
||||||
- 산출물:
|
|
||||||
- `Dockerfile.backend`
|
|
||||||
- 완료 기준:
|
|
||||||
- 컨테이너에서 3000 포트가 정상 listen 상태가 됨
|
|
||||||
|
|
||||||
### Task 7. MySQL Docker 구성 추가
|
|
||||||
|
|
||||||
- 목적: DB까지 포함한 재현 가능한 스택 구성
|
|
||||||
- 작업 내용:
|
|
||||||
- `mysql:8.0` 서비스 정의
|
|
||||||
- `MYSQL_DATABASE`, `MYSQL_USER`, `MYSQL_PASSWORD` 설정
|
|
||||||
- utf8mb4 문자셋 옵션 반영
|
|
||||||
- MySQL 데이터 volume 연결
|
|
||||||
- 초기 SQL 적재용 `docker/mysql/init` 디렉터리 설계
|
|
||||||
- 산출물:
|
|
||||||
- `docker-compose.yaml` 내 `db` 서비스 또는 별도 DB compose 확장안
|
|
||||||
- 완료 기준:
|
|
||||||
- MySQL 컨테이너가 정상 기동하고 3306 포트에서 응답 가능
|
|
||||||
|
|
||||||
### Task 8. backend DB 연결 전환
|
|
||||||
|
|
||||||
- 목적: backend가 external MySQL 대신 DB 컨테이너를 바라보도록 변경
|
|
||||||
- 작업 내용:
|
|
||||||
- `DB_HOST`를 `db`로 전환
|
|
||||||
- 필요 시 `.env.docker` 또는 compose 내부 환경변수 사용
|
|
||||||
- backend `depends_on`에 db 추가
|
|
||||||
- 산출물:
|
|
||||||
- DB 컨테이너용 backend 환경 정의
|
|
||||||
- 완료 기준:
|
|
||||||
- backend 로그에서 DB 연결 성공 확인
|
|
||||||
|
|
||||||
### Task 9. docker-compose.yaml 확장
|
|
||||||
|
|
||||||
- 목적: frontend/backend를 함께 기동
|
|
||||||
- 작업 내용:
|
|
||||||
- frontend 서비스 정의
|
|
||||||
- backend 서비스 정의
|
|
||||||
- db 서비스 정의
|
|
||||||
- 포트 매핑 추가
|
|
||||||
- `.env` 또는 docker 전용 환경변수 연결
|
|
||||||
- MySQL 데이터 볼륨 연결
|
|
||||||
- `uploads` 볼륨 연결
|
|
||||||
- `map_config.json` 처리 방식 반영
|
|
||||||
- 산출물:
|
|
||||||
- `docker-compose.yaml`
|
|
||||||
- 완료 기준:
|
|
||||||
- `docker compose up --build` 한 번으로 세 서비스가 모두 올라옴
|
|
||||||
|
|
||||||
### Task 10. `.dockerignore` 작성
|
|
||||||
|
|
||||||
- 목적: 불필요한 빌드 컨텍스트 제외
|
|
||||||
- 제외 권장 항목:
|
|
||||||
- `node_modules`
|
|
||||||
- `dist`
|
|
||||||
- `build`
|
|
||||||
- `.git`
|
|
||||||
- `uploads`
|
|
||||||
- `*.xlsx`
|
|
||||||
- 산출물:
|
|
||||||
- `.dockerignore`
|
|
||||||
- 완료 기준:
|
|
||||||
- 이미지 빌드 컨텍스트가 과도하게 커지지 않음
|
|
||||||
|
|
||||||
## 6. 시연 검증 태스크
|
|
||||||
|
|
||||||
### Task 11. WSL 컨테이너 기동 검증
|
|
||||||
|
|
||||||
- 실행 명령:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
powershell -ExecutionPolicy Bypass -File .\start_docker_wsl.ps1
|
|
||||||
```
|
|
||||||
|
|
||||||
- 확인 항목:
|
|
||||||
- frontend 로그 에러 여부
|
|
||||||
- backend 로그 에러 여부
|
|
||||||
- db 로그 에러 여부
|
|
||||||
- backend와 db 연결 성공 여부
|
|
||||||
- 완료 기준:
|
|
||||||
- 세 컨테이너 모두 종료 없이 유지됨
|
|
||||||
|
|
||||||
### Task 12. 웹 접속 검증
|
|
||||||
|
|
||||||
- 확인 항목:
|
|
||||||
- `http://localhost:8080` 접속 가능 여부
|
|
||||||
- 첫 화면 로딩 여부
|
|
||||||
- 콘솔 에러 여부
|
|
||||||
- 완료 기준:
|
|
||||||
- 브라우저에서 초기 화면이 정상 표시됨
|
|
||||||
|
|
||||||
### Task 13. API 검증
|
|
||||||
|
|
||||||
- 확인 항목:
|
|
||||||
- `http://localhost:3000/api/assets/master`
|
|
||||||
- 프런트에서 `/api/assets/master` 호출 정상 여부
|
|
||||||
- 완료 기준:
|
|
||||||
- 200 응답 또는 정상 데이터 응답 확인
|
|
||||||
|
|
||||||
### Task 14. DB 초기 데이터 검증
|
|
||||||
|
|
||||||
- 확인 항목:
|
|
||||||
- MySQL 컨테이너 내부에 목표 DB가 생성되었는지
|
|
||||||
- 기준 테이블이 존재하는지
|
|
||||||
- 샘플 데이터 또는 실데이터가 적재되었는지
|
|
||||||
- 완료 기준:
|
|
||||||
- backend가 기대하는 최소 테이블과 데이터가 실제로 조회됨
|
|
||||||
|
|
||||||
### Task 15. 업로드/파일 저장 검증
|
|
||||||
|
|
||||||
- 확인 항목:
|
|
||||||
- `/api/upload` 호출 정상 여부
|
|
||||||
- 업로드 파일이 `uploads`에 실제 저장되는지
|
|
||||||
- `map_config.json` 수정 내용이 유지되는지
|
|
||||||
- 완료 기준:
|
|
||||||
- 컨테이너 재시작 후에도 저장 데이터가 유지됨
|
|
||||||
|
|
||||||
## 7. 시연 후 후속 태스크
|
|
||||||
|
|
||||||
### Task 16. 운영형 프런트 배포 구조 전환
|
|
||||||
|
|
||||||
- 목표: Vite dev server 대신 정적 빌드 기반 구조로 전환
|
|
||||||
- 후보:
|
|
||||||
- nginx 정적 서빙
|
|
||||||
- Express 정적 서빙
|
|
||||||
|
|
||||||
### Task 17. DB 초기화/마이그레이션 전략 통합
|
|
||||||
|
|
||||||
- 목표: 기준 스키마와 실행 순서를 단일 정책으로 통일
|
|
||||||
- 필요 작업:
|
|
||||||
- 기준 스키마 선정
|
|
||||||
- 초기화 스크립트 확정
|
|
||||||
- 마이그레이션 순서 정의
|
|
||||||
|
|
||||||
### Task 18. `.env.example` 및 배포 환경 분리
|
|
||||||
|
|
||||||
- 목표: 민감정보를 저장소에서 분리하고 배포별 설정 체계화
|
|
||||||
|
|
||||||
### Task 19. 운영 볼륨 및 백업 전략 정리
|
|
||||||
|
|
||||||
- 목표: 업로드 파일과 설정 파일, MySQL 데이터의 장기 보존 정책 정리
|
|
||||||
|
|
||||||
### Task 20. DB 백업/복원 절차 문서화
|
|
||||||
|
|
||||||
- 목표: 컨테이너 DB를 기준으로 dump/restore 절차를 문서화
|
|
||||||
|
|
||||||
## 8. 우선순위 정리
|
|
||||||
|
|
||||||
### P0: 내일까지 반드시 필요한 작업
|
|
||||||
|
|
||||||
1. Task 1. 외부 MySQL 접근 가능 여부 확인
|
|
||||||
2. Task 2. 기준 스키마 상태 확인
|
|
||||||
3. Task 4. DB 기준 데이터 소스 확정
|
|
||||||
4. Task 7. MySQL Docker 구성 추가
|
|
||||||
5. Task 8. backend DB 연결 전환
|
|
||||||
6. Task 9. docker-compose.yaml 확장
|
|
||||||
7. Task 11. WSL 컨테이너 기동 검증
|
|
||||||
8. Task 12. 웹 접속 검증
|
|
||||||
9. Task 13. API 검증
|
|
||||||
10. Task 14. DB 초기 데이터 검증
|
|
||||||
|
|
||||||
### P1: 시연 안정화를 위해 권장되는 작업
|
|
||||||
|
|
||||||
1. Task 3. 파일 영속성 대상 확인
|
|
||||||
2. Task 10. `.dockerignore` 작성
|
|
||||||
3. Task 15. 업로드/파일 저장 검증
|
|
||||||
|
|
||||||
### P2: 시연 이후 진행할 작업
|
|
||||||
|
|
||||||
1. Task 16. 운영형 프런트 배포 구조 전환
|
|
||||||
2. Task 17. DB 초기화/마이그레이션 전략 통합
|
|
||||||
3. Task 18. `.env.example` 및 배포 환경 분리
|
|
||||||
4. Task 19. 운영 볼륨 및 백업 전략 정리
|
|
||||||
5. Task 20. DB 백업/복원 절차 문서화
|
|
||||||
|
|
||||||
## 9. 개발자용 최종 작업 순서 제안
|
|
||||||
|
|
||||||
개발 담당자에게는 아래 순서로 진행하라고 전달하면 된다.
|
|
||||||
|
|
||||||
1. 외부 DB 연결 가능 여부부터 확인
|
|
||||||
2. 현재 DB 스키마가 앱 요구사항과 맞는지 확인
|
|
||||||
3. DB 기준 dump 또는 init SQL 확보
|
|
||||||
4. MySQL 컨테이너 구성 추가
|
|
||||||
5. backend의 DB 연결 대상을 `db`로 전환
|
|
||||||
6. WSL에서 `docker compose config` 확인
|
|
||||||
7. WSL에서 컨테이너 기동 테스트
|
|
||||||
8. 웹 접속 및 API 확인
|
|
||||||
9. 업로드 및 파일 영속성 확인
|
|
||||||
10. 시연 완료 후 운영형 구조로 분리 작업 진행
|
|
||||||
|
|
||||||
## 10. 완료 판단 기준
|
|
||||||
|
|
||||||
이번 도커라이징 1차 작업은 아래 조건을 만족하면 완료로 본다.
|
|
||||||
|
|
||||||
1. `docker compose up --build`로 프런트, 백엔드, DB가 모두 기동한다.
|
|
||||||
2. 브라우저에서 8080 화면이 열린다.
|
|
||||||
3. `/api/assets/master`가 정상 응답한다.
|
|
||||||
4. backend가 DB 컨테이너와 정상 연결된다.
|
|
||||||
5. DB 초기 테이블과 데이터가 기대 상태로 적재된다.
|
|
||||||
6. `uploads`, `map_config.json`, MySQL 데이터가 재시작 후에도 유지된다.
|
|
||||||
|
|
||||||
이 문서는 실제 구현 작업의 체크리스트로 사용한다.
|
|
||||||
@@ -3,12 +3,14 @@
|
|||||||
|
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8" />
|
<meta charset="UTF-8" />
|
||||||
|
<link rel="icon" type="image/svg+xml" href="/vite.svg" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
<title>한맥가족 자산관리시스템</title>
|
<title>한맥가족 자산관리시스템</title>
|
||||||
<link rel="stylesheet"
|
<link rel="stylesheet"
|
||||||
href="https://cdn.jsdelivr.net/gh/orioncactus/pretendard@v1.3.9/dist/web/variable/pretendardvariable.min.css" />
|
href="https://cdn.jsdelivr.net/gh/orioncactus/pretendard@v1.3.9/dist/web/variable/pretendardvariable.min.css" />
|
||||||
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
|
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
|
||||||
<script src="https://cdn.jsdelivr.net/npm/chartjs-plugin-datalabels@2.0.0"></script>
|
<script src="https://cdn.jsdelivr.net/npm/chartjs-plugin-datalabels@2.0.0"></script>
|
||||||
|
<script src="/qrcode.min.js"></script>
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
@@ -17,7 +19,7 @@
|
|||||||
<header class="main-header">
|
<header class="main-header">
|
||||||
<div class="header-container" id="nav-container">
|
<div class="header-container" id="nav-container">
|
||||||
<div class="brand">
|
<div class="brand">
|
||||||
<!-- <img src="/image 92.png" alt="Logo" class="main-logo" /> -->
|
<img src="/image 92.png" alt="Logo" class="main-logo" />
|
||||||
<h1>한맥자산관리시스템</h1>
|
<h1>한맥자산관리시스템</h1>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
BIN
label/DevExpress.Data.v14.1.dll
Normal file
BIN
label/DevExpress.Data.v14.1.dll
Normal file
Binary file not shown.
BIN
label/DevExpress.Printing.v14.1.Core.dll
Normal file
BIN
label/DevExpress.Printing.v14.1.Core.dll
Normal file
Binary file not shown.
BIN
label/DevExpress.Utils.v14.1.dll
Normal file
BIN
label/DevExpress.Utils.v14.1.dll
Normal file
Binary file not shown.
BIN
label/DevExpress.XtraEditors.v14.1.dll
Normal file
BIN
label/DevExpress.XtraEditors.v14.1.dll
Normal file
Binary file not shown.
BIN
label/DevExpress.XtraGrid.v14.1.dll
Normal file
BIN
label/DevExpress.XtraGrid.v14.1.dll
Normal file
Binary file not shown.
BIN
label/DevExpress.XtraLayout.v14.1.dll
Normal file
BIN
label/DevExpress.XtraLayout.v14.1.dll
Normal file
Binary file not shown.
BIN
label/DevExpress.XtraPrinting.v14.1.dll
Normal file
BIN
label/DevExpress.XtraPrinting.v14.1.dll
Normal file
Binary file not shown.
BIN
label/LabelPrinter.exe
Normal file
BIN
label/LabelPrinter.exe
Normal file
Binary file not shown.
BIN
label/Newtonsoft.Json.dll
Normal file
BIN
label/Newtonsoft.Json.dll
Normal file
Binary file not shown.
BIN
label/WebQuery.dll
Normal file
BIN
label/WebQuery.dll
Normal file
Binary file not shown.
4
label/config.ini
Normal file
4
label/config.ini
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
[PRINT]
|
||||||
|
FONT=8
|
||||||
|
LEFT=143
|
||||||
|
TOP=40
|
||||||
BIN
label/de/DevExpress.Data.v14.1.resources.dll
Normal file
BIN
label/de/DevExpress.Data.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/de/DevExpress.Printing.v14.1.Core.resources.dll
Normal file
BIN
label/de/DevExpress.Printing.v14.1.Core.resources.dll
Normal file
Binary file not shown.
BIN
label/de/DevExpress.Utils.v14.1.resources.dll
Normal file
BIN
label/de/DevExpress.Utils.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/de/DevExpress.XtraEditors.v14.1.resources.dll
Normal file
BIN
label/de/DevExpress.XtraEditors.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/de/DevExpress.XtraGrid.v14.1.resources.dll
Normal file
BIN
label/de/DevExpress.XtraGrid.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/de/DevExpress.XtraLayout.v14.1.resources.dll
Normal file
BIN
label/de/DevExpress.XtraLayout.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/de/DevExpress.XtraPrinting.v14.1.resources.dll
Normal file
BIN
label/de/DevExpress.XtraPrinting.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/es/DevExpress.Data.v14.1.resources.dll
Normal file
BIN
label/es/DevExpress.Data.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/es/DevExpress.Printing.v14.1.Core.resources.dll
Normal file
BIN
label/es/DevExpress.Printing.v14.1.Core.resources.dll
Normal file
Binary file not shown.
BIN
label/es/DevExpress.Utils.v14.1.resources.dll
Normal file
BIN
label/es/DevExpress.Utils.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/es/DevExpress.XtraEditors.v14.1.resources.dll
Normal file
BIN
label/es/DevExpress.XtraEditors.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/es/DevExpress.XtraGrid.v14.1.resources.dll
Normal file
BIN
label/es/DevExpress.XtraGrid.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/es/DevExpress.XtraLayout.v14.1.resources.dll
Normal file
BIN
label/es/DevExpress.XtraLayout.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/es/DevExpress.XtraPrinting.v14.1.resources.dll
Normal file
BIN
label/es/DevExpress.XtraPrinting.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/ja/DevExpress.Data.v14.1.resources.dll
Normal file
BIN
label/ja/DevExpress.Data.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/ja/DevExpress.Printing.v14.1.Core.resources.dll
Normal file
BIN
label/ja/DevExpress.Printing.v14.1.Core.resources.dll
Normal file
Binary file not shown.
BIN
label/ja/DevExpress.Utils.v14.1.resources.dll
Normal file
BIN
label/ja/DevExpress.Utils.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/ja/DevExpress.XtraEditors.v14.1.resources.dll
Normal file
BIN
label/ja/DevExpress.XtraEditors.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/ja/DevExpress.XtraGrid.v14.1.resources.dll
Normal file
BIN
label/ja/DevExpress.XtraGrid.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/ja/DevExpress.XtraLayout.v14.1.resources.dll
Normal file
BIN
label/ja/DevExpress.XtraLayout.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/ja/DevExpress.XtraPrinting.v14.1.resources.dll
Normal file
BIN
label/ja/DevExpress.XtraPrinting.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/ru/DevExpress.Data.v14.1.resources.dll
Normal file
BIN
label/ru/DevExpress.Data.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/ru/DevExpress.Printing.v14.1.Core.resources.dll
Normal file
BIN
label/ru/DevExpress.Printing.v14.1.Core.resources.dll
Normal file
Binary file not shown.
BIN
label/ru/DevExpress.Utils.v14.1.resources.dll
Normal file
BIN
label/ru/DevExpress.Utils.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/ru/DevExpress.XtraEditors.v14.1.resources.dll
Normal file
BIN
label/ru/DevExpress.XtraEditors.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/ru/DevExpress.XtraGrid.v14.1.resources.dll
Normal file
BIN
label/ru/DevExpress.XtraGrid.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/ru/DevExpress.XtraLayout.v14.1.resources.dll
Normal file
BIN
label/ru/DevExpress.XtraLayout.v14.1.resources.dll
Normal file
Binary file not shown.
BIN
label/ru/DevExpress.XtraPrinting.v14.1.resources.dll
Normal file
BIN
label/ru/DevExpress.XtraPrinting.v14.1.resources.dll
Normal file
Binary file not shown.
7
label/tmp/file_1.txt
Normal file
7
label/tmp/file_1.txt
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
자산번호 : 210312
|
||||||
|
자산명 : 가을-PC(i5-12400F)
|
||||||
|
공급사 : (주)가을디에스
|
||||||
|
자산위치 : 지반부
|
||||||
|
관리부서 : 전산
|
||||||
|
사용자 : 박노석
|
||||||
|
취득일자 : 2024-08-05
|
||||||
BIN
label/tmp/file_1.txt - 바로 가기.lnk
Normal file
BIN
label/tmp/file_1.txt - 바로 가기.lnk
Normal file
Binary file not shown.
@@ -112,7 +112,7 @@
|
|||||||
"y": "32.01",
|
"y": "32.01",
|
||||||
"w": "40.87",
|
"w": "40.87",
|
||||||
"h": "6.24",
|
"h": "6.24",
|
||||||
"asset_id": null
|
"asset_id": "9pvkqyi"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"img/location_photo/IDC/서관203.png": [
|
"img/location_photo/IDC/서관203.png": [
|
||||||
@@ -722,5 +722,21 @@
|
|||||||
"h": "6.75",
|
"h": "6.75",
|
||||||
"asset_id": "server_1779761946023_64"
|
"asset_id": "server_1779761946023_64"
|
||||||
}
|
}
|
||||||
|
],
|
||||||
|
"img/location_photo/TDD_TEST_MAP.png": [
|
||||||
|
{
|
||||||
|
"x": "30.50",
|
||||||
|
"y": "40.25",
|
||||||
|
"w": "10.00",
|
||||||
|
"h": "12.00",
|
||||||
|
"asset_id": null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"x": "50.00",
|
||||||
|
"y": "60.00",
|
||||||
|
"w": "5.00",
|
||||||
|
"h": "5.00",
|
||||||
|
"asset_id": null
|
||||||
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
@@ -5,6 +5,7 @@
|
|||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
<title>ITAM Map Coordinate Editor v3.0</title>
|
<title>ITAM Map Coordinate Editor v3.0</title>
|
||||||
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/orioncactus/pretendard@v1.3.9/dist/web/variable/pretendardvariable.min.css" />
|
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/orioncactus/pretendard@v1.3.9/dist/web/variable/pretendardvariable.min.css" />
|
||||||
|
<script src="/qrcode.min.js"></script>
|
||||||
</head>
|
</head>
|
||||||
<body class="editor-body">
|
<body class="editor-body">
|
||||||
|
|
||||||
@@ -30,8 +31,9 @@
|
|||||||
|
|
||||||
<div class="box-list" id="box-list"></div>
|
<div class="box-list" id="box-list"></div>
|
||||||
|
|
||||||
<div class="actions">
|
<div class="actions" style="display: flex; flex-direction: column; gap: 0.5rem;">
|
||||||
<button id="btn-clear-all" class="btn btn-outline">전체 삭제</button>
|
<button id="btn-clear-all" class="btn btn-outline">전체 삭제</button>
|
||||||
|
<button id="btn-print-map-qrs" class="btn btn-outline btn-primary">이 도면 QR 일괄인쇄</button>
|
||||||
<button id="btn-save-server" class="btn btn-primary">서버에 즉시 저장</button>
|
<button id="btn-save-server" class="btn btn-primary">서버에 즉시 저장</button>
|
||||||
<div id="save-status"></div>
|
<div id="save-status"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
299
mobile.html
Normal file
299
mobile.html
Normal file
@@ -0,0 +1,299 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="ko">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no">
|
||||||
|
<title>ITAM 모바일 실사 점검</title>
|
||||||
|
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/orioncactus/pretendard@v1.3.9/dist/web/variable/pretendardvariable.min.css" />
|
||||||
|
<script src="https://unpkg.com/html5-qrcode@2.3.8/html5-qrcode.min.js"></script>
|
||||||
|
<style>
|
||||||
|
:root {
|
||||||
|
--bg: #09090b;
|
||||||
|
--card: #18181b;
|
||||||
|
--card-border: #27272a;
|
||||||
|
--primary: #3b82f6;
|
||||||
|
--primary-hover: #2563eb;
|
||||||
|
--success: #10b981;
|
||||||
|
--danger: #ef4444;
|
||||||
|
--text: #f4f4f5;
|
||||||
|
--text-muted: #a1a1aa;
|
||||||
|
--font-family: 'Pretendard Variable', -apple-system, BlinkMacSystemFont, system-ui, sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
* {
|
||||||
|
box-sizing: border-box;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
font-family: var(--font-family);
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
background-color: var(--bg);
|
||||||
|
color: var(--text);
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
height: 100vh;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
header {
|
||||||
|
background-color: var(--card);
|
||||||
|
border-bottom: 1px solid var(--card-border);
|
||||||
|
padding: 1rem;
|
||||||
|
text-align: center;
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
header h1 {
|
||||||
|
font-size: 1.1rem;
|
||||||
|
font-weight: 700;
|
||||||
|
background: linear-gradient(135deg, #60a5fa, #3b82f6);
|
||||||
|
-webkit-background-clip: text;
|
||||||
|
-webkit-text-fill-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.status-dot {
|
||||||
|
width: 8px;
|
||||||
|
height: 8px;
|
||||||
|
border-radius: 50%;
|
||||||
|
background-color: var(--success);
|
||||||
|
box-shadow: 0 0 8px var(--success);
|
||||||
|
}
|
||||||
|
|
||||||
|
main {
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
padding: 1rem;
|
||||||
|
gap: 1rem;
|
||||||
|
overflow-y: auto;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Scanner Viewport */
|
||||||
|
.scanner-container {
|
||||||
|
width: 100%;
|
||||||
|
max-width: 400px;
|
||||||
|
aspect-ratio: 1;
|
||||||
|
border-radius: 16px;
|
||||||
|
overflow: hidden;
|
||||||
|
border: 2px dashed var(--card-border);
|
||||||
|
position: relative;
|
||||||
|
background-color: #000;
|
||||||
|
}
|
||||||
|
|
||||||
|
#reader {
|
||||||
|
width: 100% !important;
|
||||||
|
height: 100% !important;
|
||||||
|
border: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
#reader video {
|
||||||
|
object-fit: cover !important;
|
||||||
|
width: 100% !important;
|
||||||
|
height: 100% !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Scan Laser Line Animation */
|
||||||
|
.scan-laser {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 2px;
|
||||||
|
background: linear-gradient(to right, transparent, var(--primary), transparent);
|
||||||
|
animation: scan 2s linear infinite;
|
||||||
|
z-index: 10;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes scan {
|
||||||
|
0% { top: 0%; }
|
||||||
|
50% { top: 100%; }
|
||||||
|
100% { top: 0%; }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Bottom Info Card */
|
||||||
|
.info-panel {
|
||||||
|
width: 100%;
|
||||||
|
max-width: 400px;
|
||||||
|
background-color: var(--card);
|
||||||
|
border: 1px solid var(--card-border);
|
||||||
|
border-radius: 16px;
|
||||||
|
padding: 1rem;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.75rem;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-section {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-label {
|
||||||
|
font-size: 0.75rem;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--text-muted);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.05em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-value {
|
||||||
|
font-size: 0.95rem;
|
||||||
|
font-weight: 700;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: 0.5rem;
|
||||||
|
min-height: 24px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-lock {
|
||||||
|
background-color: rgba(59, 130, 246, 0.15);
|
||||||
|
color: var(--primary);
|
||||||
|
padding: 0.25rem 0.5rem;
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 0.8rem;
|
||||||
|
border: 1px solid rgba(59, 130, 246, 0.3);
|
||||||
|
font-weight: 700;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-empty {
|
||||||
|
color: var(--text-muted);
|
||||||
|
font-weight: 400;
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-action {
|
||||||
|
background-color: var(--primary);
|
||||||
|
color: var(--text);
|
||||||
|
border: none;
|
||||||
|
padding: 0.5rem 1rem;
|
||||||
|
border-radius: 8px;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
font-weight: 600;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background-color 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-action:hover {
|
||||||
|
background-color: var(--primary-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-action.btn-danger {
|
||||||
|
background-color: rgba(239, 68, 68, 0.15);
|
||||||
|
color: var(--danger);
|
||||||
|
border: 1px solid rgba(239, 68, 68, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-action.btn-danger:hover {
|
||||||
|
background-color: rgba(239, 68, 68, 0.25);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Manual Input Section */
|
||||||
|
.manual-toggle {
|
||||||
|
text-align: center;
|
||||||
|
font-size: 0.8rem;
|
||||||
|
color: var(--primary);
|
||||||
|
cursor: pointer;
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
.manual-form {
|
||||||
|
display: none;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.5rem;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.input-field {
|
||||||
|
width: 100%;
|
||||||
|
background-color: var(--bg);
|
||||||
|
border: 1px solid var(--card-border);
|
||||||
|
border-radius: 8px;
|
||||||
|
color: var(--text);
|
||||||
|
padding: 0.5rem;
|
||||||
|
font-size: 0.9rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.input-field:focus {
|
||||||
|
outline: 1px solid var(--primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Feedbacks Overlay */
|
||||||
|
.feedback-message {
|
||||||
|
text-align: center;
|
||||||
|
padding: 0.5rem;
|
||||||
|
border-radius: 8px;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
display: none;
|
||||||
|
animation: fadeIn 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes fadeIn {
|
||||||
|
from { opacity: 0; transform: translateY(5px); }
|
||||||
|
to { opacity: 1; transform: translateY(0); }
|
||||||
|
}
|
||||||
|
|
||||||
|
.feedback-success {
|
||||||
|
background-color: rgba(16, 185, 129, 0.15);
|
||||||
|
color: var(--success);
|
||||||
|
border: 1px solid rgba(16, 185, 129, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.feedback-error {
|
||||||
|
background-color: rgba(239, 68, 68, 0.15);
|
||||||
|
color: var(--danger);
|
||||||
|
border: 1px solid rgba(239, 68, 68, 0.3);
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<header>
|
||||||
|
<h1>ITAM 모바일 실사</h1>
|
||||||
|
<div class="status-dot"></div>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<main>
|
||||||
|
<div class="scanner-container">
|
||||||
|
<div id="reader"></div>
|
||||||
|
<div class="scan-laser"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="info-panel">
|
||||||
|
<!-- 1. 위치 락 정보 -->
|
||||||
|
<div class="info-section">
|
||||||
|
<span class="info-label">현재 점검 위치 (Location)</span>
|
||||||
|
<div class="info-value">
|
||||||
|
<span id="loc-display" class="badge-empty">위치 QR 코드를 먼저 스캔하세요.</span>
|
||||||
|
<button id="btn-unlock-loc" class="btn-action btn-danger" style="display: none;">해제</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr style="border: 0; border-top: 1px solid var(--card-border); margin: 0.25rem 0;" />
|
||||||
|
|
||||||
|
<!-- 2. 자산 스캔 결과 및 피드백 -->
|
||||||
|
<div id="scan-feedback" class="feedback-message"></div>
|
||||||
|
|
||||||
|
<!-- 3. 수동 입력 토글 및 양식 -->
|
||||||
|
<div class="info-section">
|
||||||
|
<span id="btn-toggle-manual" class="manual-toggle">카메라가 안 되나요? 수동 코드로 입력</span>
|
||||||
|
<div id="manual-form" class="manual-form">
|
||||||
|
<input type="text" id="manual-code-input" class="input-field" placeholder="위치 또는 자산 코드 입력" />
|
||||||
|
<button id="btn-submit-manual" class="btn-action w-full">입력 확인</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<script type="module" src="/src/mobile-main.ts"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
291
package-lock.json
generated
291
package-lock.json
generated
@@ -14,9 +14,11 @@
|
|||||||
"iconv-lite": "^0.7.2",
|
"iconv-lite": "^0.7.2",
|
||||||
"lucide": "^0.364.0",
|
"lucide": "^0.364.0",
|
||||||
"mysql2": "^3.22.1",
|
"mysql2": "^3.22.1",
|
||||||
|
"qrcode": "^1.5.4",
|
||||||
"xlsx": "^0.18.5"
|
"xlsx": "^0.18.5"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@types/qrcode": "^1.5.6",
|
||||||
"typescript": "^5.2.2",
|
"typescript": "^5.2.2",
|
||||||
"vite": "^5.2.0"
|
"vite": "^5.2.0"
|
||||||
}
|
}
|
||||||
@@ -774,11 +776,19 @@
|
|||||||
"resolved": "https://registry.npmjs.org/@types/node/-/node-25.6.0.tgz",
|
"resolved": "https://registry.npmjs.org/@types/node/-/node-25.6.0.tgz",
|
||||||
"integrity": "sha512-+qIYRKdNYJwY3vRCZMdJbPLJAtGjQBudzZzdzwQYkEPQd+PJGixUL5QfvCLDaULoLv+RhT3LDkwEfKaAkgSmNQ==",
|
"integrity": "sha512-+qIYRKdNYJwY3vRCZMdJbPLJAtGjQBudzZzdzwQYkEPQd+PJGixUL5QfvCLDaULoLv+RhT3LDkwEfKaAkgSmNQ==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"peer": true,
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"undici-types": "~7.19.0"
|
"undici-types": "~7.19.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@types/qrcode": {
|
||||||
|
"version": "1.5.6",
|
||||||
|
"resolved": "https://registry.npmjs.org/@types/qrcode/-/qrcode-1.5.6.tgz",
|
||||||
|
"integrity": "sha512-te7NQcV2BOvdj2b1hCAHzAoMNuj65kNBMz0KBaxM6c3VGBOhU0dURQKOtH8CFNI/dsKkwlv32p26qYQTWoB5bw==",
|
||||||
|
"dev": true,
|
||||||
|
"dependencies": {
|
||||||
|
"@types/node": "*"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/accepts": {
|
"node_modules/accepts": {
|
||||||
"version": "2.0.0",
|
"version": "2.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/accepts/-/accepts-2.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/accepts/-/accepts-2.0.0.tgz",
|
||||||
@@ -801,6 +811,28 @@
|
|||||||
"node": ">=0.8"
|
"node": ">=0.8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/ansi-regex": {
|
||||||
|
"version": "5.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
|
||||||
|
"integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/ansi-styles": {
|
||||||
|
"version": "4.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
|
||||||
|
"integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
|
||||||
|
"dependencies": {
|
||||||
|
"color-convert": "^2.0.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/chalk/ansi-styles?sponsor=1"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/aws-ssl-profiles": {
|
"node_modules/aws-ssl-profiles": {
|
||||||
"version": "1.1.2",
|
"version": "1.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/aws-ssl-profiles/-/aws-ssl-profiles-1.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/aws-ssl-profiles/-/aws-ssl-profiles-1.1.2.tgz",
|
||||||
@@ -872,6 +904,14 @@
|
|||||||
"url": "https://github.com/sponsors/ljharb"
|
"url": "https://github.com/sponsors/ljharb"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/camelcase": {
|
||||||
|
"version": "5.3.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz",
|
||||||
|
"integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=6"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/cfb": {
|
"node_modules/cfb": {
|
||||||
"version": "1.2.2",
|
"version": "1.2.2",
|
||||||
"resolved": "https://registry.npmjs.org/cfb/-/cfb-1.2.2.tgz",
|
"resolved": "https://registry.npmjs.org/cfb/-/cfb-1.2.2.tgz",
|
||||||
@@ -885,6 +925,16 @@
|
|||||||
"node": ">=0.8"
|
"node": ">=0.8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/cliui": {
|
||||||
|
"version": "6.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/cliui/-/cliui-6.0.0.tgz",
|
||||||
|
"integrity": "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"string-width": "^4.2.0",
|
||||||
|
"strip-ansi": "^6.0.0",
|
||||||
|
"wrap-ansi": "^6.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/codepage": {
|
"node_modules/codepage": {
|
||||||
"version": "1.15.0",
|
"version": "1.15.0",
|
||||||
"resolved": "https://registry.npmjs.org/codepage/-/codepage-1.15.0.tgz",
|
"resolved": "https://registry.npmjs.org/codepage/-/codepage-1.15.0.tgz",
|
||||||
@@ -894,6 +944,22 @@
|
|||||||
"node": ">=0.8"
|
"node": ">=0.8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/color-convert": {
|
||||||
|
"version": "2.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
|
||||||
|
"integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"color-name": "~1.1.4"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=7.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/color-name": {
|
||||||
|
"version": "1.1.4",
|
||||||
|
"resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
|
||||||
|
"integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA=="
|
||||||
|
},
|
||||||
"node_modules/content-disposition": {
|
"node_modules/content-disposition": {
|
||||||
"version": "1.1.0",
|
"version": "1.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-1.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-1.1.0.tgz",
|
||||||
@@ -980,6 +1046,14 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/decamelize": {
|
||||||
|
"version": "1.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz",
|
||||||
|
"integrity": "sha512-z2S+W9X73hAUUki+N+9Za2lBlun89zigOyGrsax+KUQ6wKW4ZoWpEYBkGhQjwAjjDCkWxhY0VKEhk8wzY7F5cA==",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.10.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/denque": {
|
"node_modules/denque": {
|
||||||
"version": "2.1.0",
|
"version": "2.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/denque/-/denque-2.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/denque/-/denque-2.1.0.tgz",
|
||||||
@@ -998,6 +1072,11 @@
|
|||||||
"node": ">= 0.8"
|
"node": ">= 0.8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/dijkstrajs": {
|
||||||
|
"version": "1.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/dijkstrajs/-/dijkstrajs-1.0.3.tgz",
|
||||||
|
"integrity": "sha512-qiSlmBq9+BCdCA/L46dw8Uy93mloxsPSbwnm5yrKn2vMPiy8KyAskTF6zuV/j5BMsmOGZDPs7KjU+mjb670kfA=="
|
||||||
|
},
|
||||||
"node_modules/dotenv": {
|
"node_modules/dotenv": {
|
||||||
"version": "17.4.2",
|
"version": "17.4.2",
|
||||||
"resolved": "https://registry.npmjs.org/dotenv/-/dotenv-17.4.2.tgz",
|
"resolved": "https://registry.npmjs.org/dotenv/-/dotenv-17.4.2.tgz",
|
||||||
@@ -1030,6 +1109,11 @@
|
|||||||
"integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==",
|
"integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/emoji-regex": {
|
||||||
|
"version": "8.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
|
||||||
|
"integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="
|
||||||
|
},
|
||||||
"node_modules/encodeurl": {
|
"node_modules/encodeurl": {
|
||||||
"version": "2.0.0",
|
"version": "2.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz",
|
||||||
@@ -1187,6 +1271,18 @@
|
|||||||
"url": "https://opencollective.com/express"
|
"url": "https://opencollective.com/express"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/find-up": {
|
||||||
|
"version": "4.1.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz",
|
||||||
|
"integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==",
|
||||||
|
"dependencies": {
|
||||||
|
"locate-path": "^5.0.0",
|
||||||
|
"path-exists": "^4.0.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/forwarded": {
|
"node_modules/forwarded": {
|
||||||
"version": "0.2.0",
|
"version": "0.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
|
||||||
@@ -1247,6 +1343,14 @@
|
|||||||
"is-property": "^1.0.2"
|
"is-property": "^1.0.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/get-caller-file": {
|
||||||
|
"version": "2.0.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz",
|
||||||
|
"integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==",
|
||||||
|
"engines": {
|
||||||
|
"node": "6.* || 8.* || >= 10.*"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/get-intrinsic": {
|
"node_modules/get-intrinsic": {
|
||||||
"version": "1.3.0",
|
"version": "1.3.0",
|
||||||
"resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz",
|
"resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz",
|
||||||
@@ -1371,6 +1475,14 @@
|
|||||||
"node": ">= 0.10"
|
"node": ">= 0.10"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/is-fullwidth-code-point": {
|
||||||
|
"version": "3.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
|
||||||
|
"integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/is-promise": {
|
"node_modules/is-promise": {
|
||||||
"version": "4.0.0",
|
"version": "4.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/is-promise/-/is-promise-4.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/is-promise/-/is-promise-4.0.0.tgz",
|
||||||
@@ -1383,6 +1495,17 @@
|
|||||||
"integrity": "sha512-Ks/IoX00TtClbGQr4TWXemAnktAQvYB7HzcCxDGqEZU6oCmb2INHuOoKxbtR+HFkmYWBKv/dOZtGRiAjDhj92g==",
|
"integrity": "sha512-Ks/IoX00TtClbGQr4TWXemAnktAQvYB7HzcCxDGqEZU6oCmb2INHuOoKxbtR+HFkmYWBKv/dOZtGRiAjDhj92g==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/locate-path": {
|
||||||
|
"version": "5.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz",
|
||||||
|
"integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==",
|
||||||
|
"dependencies": {
|
||||||
|
"p-locate": "^4.1.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/long": {
|
"node_modules/long": {
|
||||||
"version": "5.3.2",
|
"version": "5.3.2",
|
||||||
"resolved": "https://registry.npmjs.org/long/-/long-5.3.2.tgz",
|
"resolved": "https://registry.npmjs.org/long/-/long-5.3.2.tgz",
|
||||||
@@ -1575,6 +1698,39 @@
|
|||||||
"wrappy": "1"
|
"wrappy": "1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/p-limit": {
|
||||||
|
"version": "2.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz",
|
||||||
|
"integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==",
|
||||||
|
"dependencies": {
|
||||||
|
"p-try": "^2.0.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=6"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/sindresorhus"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/p-locate": {
|
||||||
|
"version": "4.1.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz",
|
||||||
|
"integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==",
|
||||||
|
"dependencies": {
|
||||||
|
"p-limit": "^2.2.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/p-try": {
|
||||||
|
"version": "2.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz",
|
||||||
|
"integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=6"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/parseurl": {
|
"node_modules/parseurl": {
|
||||||
"version": "1.3.3",
|
"version": "1.3.3",
|
||||||
"resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
|
"resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
|
||||||
@@ -1584,6 +1740,14 @@
|
|||||||
"node": ">= 0.8"
|
"node": ">= 0.8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/path-exists": {
|
||||||
|
"version": "4.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
|
||||||
|
"integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/path-to-regexp": {
|
"node_modules/path-to-regexp": {
|
||||||
"version": "8.4.2",
|
"version": "8.4.2",
|
||||||
"resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-8.4.2.tgz",
|
"resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-8.4.2.tgz",
|
||||||
@@ -1601,6 +1765,14 @@
|
|||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "ISC"
|
"license": "ISC"
|
||||||
},
|
},
|
||||||
|
"node_modules/pngjs": {
|
||||||
|
"version": "5.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/pngjs/-/pngjs-5.0.0.tgz",
|
||||||
|
"integrity": "sha512-40QW5YalBNfQo5yRYmiw7Yz6TKKVr3h6970B2YE+3fQpsWcrbj1PzJgxeJ19DRQjhMbKPIuMY8rFaXc8moolVw==",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10.13.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/postcss": {
|
"node_modules/postcss": {
|
||||||
"version": "8.5.9",
|
"version": "8.5.9",
|
||||||
"resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.9.tgz",
|
"resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.9.tgz",
|
||||||
@@ -1643,6 +1815,22 @@
|
|||||||
"node": ">= 0.10"
|
"node": ">= 0.10"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/qrcode": {
|
||||||
|
"version": "1.5.4",
|
||||||
|
"resolved": "https://registry.npmjs.org/qrcode/-/qrcode-1.5.4.tgz",
|
||||||
|
"integrity": "sha512-1ca71Zgiu6ORjHqFBDpnSMTR2ReToX4l1Au1VFLyVeBTFavzQnv5JxMFr3ukHVKpSrSA2MCk0lNJSykjUfz7Zg==",
|
||||||
|
"dependencies": {
|
||||||
|
"dijkstrajs": "^1.0.1",
|
||||||
|
"pngjs": "^5.0.0",
|
||||||
|
"yargs": "^15.3.1"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"qrcode": "bin/qrcode"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10.13.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/qs": {
|
"node_modules/qs": {
|
||||||
"version": "6.15.1",
|
"version": "6.15.1",
|
||||||
"resolved": "https://registry.npmjs.org/qs/-/qs-6.15.1.tgz",
|
"resolved": "https://registry.npmjs.org/qs/-/qs-6.15.1.tgz",
|
||||||
@@ -1682,6 +1870,19 @@
|
|||||||
"node": ">= 0.10"
|
"node": ">= 0.10"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/require-directory": {
|
||||||
|
"version": "2.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz",
|
||||||
|
"integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.10.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/require-main-filename": {
|
||||||
|
"version": "2.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz",
|
||||||
|
"integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg=="
|
||||||
|
},
|
||||||
"node_modules/rollup": {
|
"node_modules/rollup": {
|
||||||
"version": "4.60.1",
|
"version": "4.60.1",
|
||||||
"resolved": "https://registry.npmjs.org/rollup/-/rollup-4.60.1.tgz",
|
"resolved": "https://registry.npmjs.org/rollup/-/rollup-4.60.1.tgz",
|
||||||
@@ -1794,6 +1995,11 @@
|
|||||||
"url": "https://opencollective.com/express"
|
"url": "https://opencollective.com/express"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/set-blocking": {
|
||||||
|
"version": "2.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz",
|
||||||
|
"integrity": "sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw=="
|
||||||
|
},
|
||||||
"node_modules/setprototypeof": {
|
"node_modules/setprototypeof": {
|
||||||
"version": "1.2.0",
|
"version": "1.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
|
||||||
@@ -1918,6 +2124,30 @@
|
|||||||
"node": ">= 0.8"
|
"node": ">= 0.8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/string-width": {
|
||||||
|
"version": "4.2.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
|
||||||
|
"integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
|
||||||
|
"dependencies": {
|
||||||
|
"emoji-regex": "^8.0.0",
|
||||||
|
"is-fullwidth-code-point": "^3.0.0",
|
||||||
|
"strip-ansi": "^6.0.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/strip-ansi": {
|
||||||
|
"version": "6.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
|
||||||
|
"integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
|
||||||
|
"dependencies": {
|
||||||
|
"ansi-regex": "^5.0.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/toidentifier": {
|
"node_modules/toidentifier": {
|
||||||
"version": "1.0.1",
|
"version": "1.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz",
|
||||||
@@ -1959,8 +2189,7 @@
|
|||||||
"version": "7.19.2",
|
"version": "7.19.2",
|
||||||
"resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.19.2.tgz",
|
"resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.19.2.tgz",
|
||||||
"integrity": "sha512-qYVnV5OEm2AW8cJMCpdV20CDyaN3g0AjDlOGf1OW4iaDEx8MwdtChUp4zu4H0VP3nDRF/8RKWH+IPp9uW0YGZg==",
|
"integrity": "sha512-qYVnV5OEm2AW8cJMCpdV20CDyaN3g0AjDlOGf1OW4iaDEx8MwdtChUp4zu4H0VP3nDRF/8RKWH+IPp9uW0YGZg==",
|
||||||
"license": "MIT",
|
"license": "MIT"
|
||||||
"peer": true
|
|
||||||
},
|
},
|
||||||
"node_modules/unpipe": {
|
"node_modules/unpipe": {
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
@@ -2040,6 +2269,11 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/which-module": {
|
||||||
|
"version": "2.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.1.tgz",
|
||||||
|
"integrity": "sha512-iBdZ57RDvnOR9AGBhML2vFZf7h8vmBjhoaZqODJBFWHVtKkDmKuHai3cx5PgVMrX5YDNp27AofYbAwctSS+vhQ=="
|
||||||
|
},
|
||||||
"node_modules/wmf": {
|
"node_modules/wmf": {
|
||||||
"version": "1.0.2",
|
"version": "1.0.2",
|
||||||
"resolved": "https://registry.npmjs.org/wmf/-/wmf-1.0.2.tgz",
|
"resolved": "https://registry.npmjs.org/wmf/-/wmf-1.0.2.tgz",
|
||||||
@@ -2058,6 +2292,19 @@
|
|||||||
"node": ">=0.8"
|
"node": ">=0.8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/wrap-ansi": {
|
||||||
|
"version": "6.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz",
|
||||||
|
"integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==",
|
||||||
|
"dependencies": {
|
||||||
|
"ansi-styles": "^4.0.0",
|
||||||
|
"string-width": "^4.1.0",
|
||||||
|
"strip-ansi": "^6.0.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/wrappy": {
|
"node_modules/wrappy": {
|
||||||
"version": "1.0.2",
|
"version": "1.0.2",
|
||||||
"resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
|
"resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
|
||||||
@@ -2084,6 +2331,44 @@
|
|||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=0.8"
|
"node": ">=0.8"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"node_modules/y18n": {
|
||||||
|
"version": "4.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.3.tgz",
|
||||||
|
"integrity": "sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ=="
|
||||||
|
},
|
||||||
|
"node_modules/yargs": {
|
||||||
|
"version": "15.4.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/yargs/-/yargs-15.4.1.tgz",
|
||||||
|
"integrity": "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A==",
|
||||||
|
"dependencies": {
|
||||||
|
"cliui": "^6.0.0",
|
||||||
|
"decamelize": "^1.2.0",
|
||||||
|
"find-up": "^4.1.0",
|
||||||
|
"get-caller-file": "^2.0.1",
|
||||||
|
"require-directory": "^2.1.1",
|
||||||
|
"require-main-filename": "^2.0.0",
|
||||||
|
"set-blocking": "^2.0.0",
|
||||||
|
"string-width": "^4.2.0",
|
||||||
|
"which-module": "^2.0.0",
|
||||||
|
"y18n": "^4.0.0",
|
||||||
|
"yargs-parser": "^18.1.2"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/yargs-parser": {
|
||||||
|
"version": "18.1.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-18.1.3.tgz",
|
||||||
|
"integrity": "sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"camelcase": "^5.0.0",
|
||||||
|
"decamelize": "^1.2.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=6"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,6 +11,7 @@
|
|||||||
"db-init": "node db_init.js"
|
"db-init": "node db_init.js"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@types/qrcode": "^1.5.6",
|
||||||
"typescript": "^5.2.2",
|
"typescript": "^5.2.2",
|
||||||
"vite": "^5.2.0"
|
"vite": "^5.2.0"
|
||||||
},
|
},
|
||||||
@@ -21,6 +22,7 @@
|
|||||||
"iconv-lite": "^0.7.2",
|
"iconv-lite": "^0.7.2",
|
||||||
"lucide": "^0.364.0",
|
"lucide": "^0.364.0",
|
||||||
"mysql2": "^3.22.1",
|
"mysql2": "^3.22.1",
|
||||||
|
"qrcode": "^1.5.4",
|
||||||
"xlsx": "^0.18.5"
|
"xlsx": "^0.18.5"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,247 +0,0 @@
|
|||||||
# ITAM 운영 배포 작업 로드맵
|
|
||||||
|
|
||||||
## 1. 문서 목적
|
|
||||||
|
|
||||||
이 문서는 ITAM 저장소를 Windows/WSL 시범 구동 상태에서 Linux 운영 서버 배포 상태로 전환하기 위한 구체적인 작업 목록과 우선순위를 정리한 로드맵이다.
|
|
||||||
|
|
||||||
현재 상태: 개발/시범용 Docker 구조 (Vite dev server + bind mount + external DB)
|
|
||||||
목표 상태: 운영 배포 구조 (정적 빌드 + 영속 스토리지 분리 + reverse proxy + external DB)
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 2. 작업 페이즈 분류
|
|
||||||
|
|
||||||
### 2.1 Phase 1: 핵심 배포 파일 (우선순위: 높음)
|
|
||||||
|
|
||||||
운영 환경에서 실제 배포를 가능하게 하는 기초 작업이다.
|
|
||||||
|
|
||||||
1. ✅ **Add production compose file** (`docker-compose.prod.yaml`)
|
|
||||||
- 목표: 운영 서버 기준 최종 compose 파일 작성
|
|
||||||
- 범위: backend, frontend, nginx 서비스 정의
|
|
||||||
- 입력: 외부 `.env`, 호스트 경로 마운트 정의
|
|
||||||
- 출력: `/deploy/docker-compose.prod.yaml`
|
|
||||||
- 완료 기준: `docker compose -f docker-compose.prod.yaml config` 성공
|
|
||||||
|
|
||||||
2. ✅ **Create production frontend Dockerfile (multi-stage build)**
|
|
||||||
- 목표: 정적 자산 기반 프런트엔드 이미지 생성
|
|
||||||
- 범위: Stage 1 = 빌드 (Node.js + npm build), Stage 2 = 정적 서빙 (Nginx)
|
|
||||||
- 입력: 현재 `Dockerfile.frontend` 참고, `npm run build` 검증
|
|
||||||
- 출력: `Dockerfile.frontend.prod` 또는 분기 처리
|
|
||||||
- 완료 기준: 로컬에서 `npm run build` 성공, 이미지 빌드 성공, 정적 파일 8080 서빙 확인
|
|
||||||
|
|
||||||
3. ✅ **Harden backend Dockerfile for production**
|
|
||||||
- 목표: production 환경 최적화된 백엔드 이미지
|
|
||||||
- 범위: NODE_ENV=production, production 의존성만, 비루트 사용자, health check 추가
|
|
||||||
- 입력: 현재 `Dockerfile.backend`, `server.js` 검토
|
|
||||||
- 출력: 수정된 `Dockerfile.backend` 또는 `.prod` 버전
|
|
||||||
- 완료 기준: 이미지 빌드 성공, 시작 후 health endpoint 응답 200
|
|
||||||
|
|
||||||
4. ✅ **Define host paths and named volumes for persistence**
|
|
||||||
- 목표: Linux 서버의 운영 디렉터리 구조 정의
|
|
||||||
- 범위: `/srv/itam/{app,env,config,uploads,logs,deploy}` 마운트 정책 확정
|
|
||||||
- 입력: doc_readme3.md 섹션 7 참고
|
|
||||||
- 출력: docker-compose.prod.yaml에 적용, 볼륨 마운트 확인
|
|
||||||
- 완료 기준: compose 파일에 경로 마운트 명시, 영속성 테스트 (컨테이너 재시작 후 데이터 유지)
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
### 2.2 Phase 2: 네트워킹 및 보안 (우선순위: 높음)
|
|
||||||
|
|
||||||
외부 접근 경로와 보안을 담당하는 작업이다.
|
|
||||||
|
|
||||||
5. ✅ **Provide Nginx reverse-proxy and frontend static config**
|
|
||||||
- 목표: Nginx 설정파일 작성 (frontend 정적 서빙 + API 프록시)
|
|
||||||
- 범위: `/` → frontend, `/api/` → backend:3000, 기본 보안 헤더, gzip
|
|
||||||
- 입력: doc_readme3.md 섹션 12 참고 (예시 개념)
|
|
||||||
- 출력: `/deploy/nginx/default.conf`
|
|
||||||
- 완료 기준: `docker compose -f docker-compose.prod.yaml up -d` 후 `http://localhost/api/assets/master` 응답 200
|
|
||||||
|
|
||||||
6. ✅ **Externalize and secure environment variables (.env.example + secrets guidance)**
|
|
||||||
- 목표: 민감정보 보호 기준 문서화
|
|
||||||
- 범위: `.env.example` 생성, Git 제외 확인, 운영 환경 분리 지침
|
|
||||||
- 입력: 현재 `.env` 파일, `.gitignore` 점검
|
|
||||||
- 출력: `.env.example`, env 관리 가이드 추가 (doc_readme3.md 또는 SECURITY.md)
|
|
||||||
- 완료 기준: `.env`가 `.gitignore`에 등록, `.env.example` 배포 파일 작성됨
|
|
||||||
|
|
||||||
7. ✅ **Define TLS certificate handling strategy (Let's Encrypt / mount certs)**
|
|
||||||
- 목표: HTTPS 인증서 관리 정책 확정
|
|
||||||
- 범위: 자동 갱신 (certbot + Let's Encrypt) 또는 마운트 기반 수동 관리 선택
|
|
||||||
- 입력: 사내 정책 확인, 운영 도메인 확인
|
|
||||||
- 출력: `deploy/nginx/tls-strategy.md` 또는 compose 파일 주석으로 정리
|
|
||||||
- 완료 기준: 선택 방식 문서화, nginx 설정 적용 준비
|
|
||||||
|
|
||||||
8. ✅ **Security review: non-root users, image scan, secret rotation**
|
|
||||||
- 목표: 보안 체크리스트 작성 및 초기 적용
|
|
||||||
- 범위: Dockerfile non-root 사용자 추가, 이미지 취약점 스캔 지침, 비밀 로테이션 정책
|
|
||||||
- 입력: doc_readme3.md 섹션 13.3 (보안) 참고
|
|
||||||
- 출력: 수정된 Dockerfile, `SECURITY.md` 또는 운영 가이드 추가
|
|
||||||
- 완료 기준: 백엔드/프런트엔드 모두 비루트 사용자로 실행 확인
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
### 2.3 Phase 3: 모니터링 및 운영 준비 (우선순위: 중간)
|
|
||||||
|
|
||||||
배포 후 운영을 원활하게 하기 위한 작업이다.
|
|
||||||
|
|
||||||
9. ✅ **Add healthcheck and readiness endpoint in backend**
|
|
||||||
- 목표: backend 헬스 체크 엔드포인트 추가
|
|
||||||
- 범위: `GET /health` 또는 `/ready` 엔드포인트 추가 (DB 연결 확인)
|
|
||||||
- 입력: `server.js` 현재 코드 검토
|
|
||||||
- 출력: backend 에서 health 응답, docker-compose.prod.yaml에 healthcheck 설정
|
|
||||||
- 완료 기준: `curl http://localhost:3000/health` 응답 200, 컨테이너 헬스 상태 healthy 표시
|
|
||||||
|
|
||||||
10. ✅ **Add logging and log rotation guidance**
|
|
||||||
- 목표: 컨테이너 로그 관리 정책 문서화
|
|
||||||
- 범위: Docker logging driver 설정, log rotation 정책, 저장소 경로 정의
|
|
||||||
- 입력: `/srv/itam/logs` 마운트 계획
|
|
||||||
- 출력: docker-compose.prod.yaml에 로깅 설정, docs에 로그 확인 가이드
|
|
||||||
- 완료 기준: 로그 파일이 `/srv/itam/logs`에 저장됨, rotation 정책 명시
|
|
||||||
|
|
||||||
11. ✅ **Document backup and restore procedures for DB and uploads**
|
|
||||||
- 목표: 운영 데이터 백업/복구 절차 문서화
|
|
||||||
- 범위: 외부 MySQL 백업 정책, `/srv/itam/uploads` 백업, 복구 절차 스크립트 예시
|
|
||||||
- 입력: doc_readme3.md 섹션 14 참고
|
|
||||||
- 출력: `BACKUP_RESTORE.md` 또는 운영 가이드 추가 섹션
|
|
||||||
- 완료 기준: 백업 스크립트 예시 작성, 복구 절차 테스트 완료
|
|
||||||
|
|
||||||
12. ✅ **Add smoke tests and post-deploy checks**
|
|
||||||
- 목표: 배포 후 빠른 검증 스크립트 작성
|
|
||||||
- 범위: 컨테이너 상태 확인, API 응답 테스트, 파일 업로드 테스트, DB 연결 확인
|
|
||||||
- 입력: doc_readme3.md 섹션 13 (점검 체크리스트) 참고
|
|
||||||
- 출력: `scripts/smoke-test.sh` 또는 배포 후 확인 스크립트
|
|
||||||
- 완료 기준: 스크립트 실행 후 모든 검사 통과
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
### 2.4 Phase 4: 자동화 및 CI/CD (우선순위: 중간)
|
|
||||||
|
|
||||||
장기 운영을 위한 자동화 작업이다.
|
|
||||||
|
|
||||||
13. ✅ **Prepare CI/CD build & push scripts (image registry)**
|
|
||||||
- 목표: 이미지 빌드 및 레지스트리 푸시 자동화
|
|
||||||
- 범위: `docker build`, `docker tag`, `docker push` 스크립트 또는 GitHub Actions/GitLab CI 예시
|
|
||||||
- 입력: 이미지 레지스트리 주소 확인 (e.g., registry.example.com, Docker Hub, etc.)
|
|
||||||
- 출력: `.github/workflows/build.yml` 또는 `scripts/build-push.sh`
|
|
||||||
- 완료 기준: 수동 빌드/푸시 스크립트 작동 확인, 이미지 태그 정책 확정
|
|
||||||
|
|
||||||
14. ✅ **Create deploy directory with compose.prod and nginx configs**
|
|
||||||
- 목표: 운영 배포 디렉터리 정리
|
|
||||||
- 범위: `/deploy/docker-compose.prod.yaml`, `/deploy/nginx/default.conf`, 기타 설정 파일 조직화
|
|
||||||
- 입력: Phase 1-3 결과물
|
|
||||||
- 출력: 다음 구조로 정리됨:
|
|
||||||
```
|
|
||||||
deploy/
|
|
||||||
docker-compose.prod.yaml
|
|
||||||
nginx/
|
|
||||||
default.conf
|
|
||||||
tls-strategy.md
|
|
||||||
scripts/
|
|
||||||
smoke-test.sh
|
|
||||||
backup.sh
|
|
||||||
```
|
|
||||||
- 완료 기준: 디렉터리 구조 확정, 모든 파일 위치 일관성 있음
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
### 2.5 Phase 5: 절차 및 문서화 (우선순위: 중간)
|
|
||||||
|
|
||||||
운영 절차와 문서를 정리하는 작업이다.
|
|
||||||
|
|
||||||
15. ✅ **Write rollout and rollback procedures (steps, checklist)**
|
|
||||||
- 목표: 배포 및 복구 절차 문서화
|
|
||||||
- 범위: 배포 전 체크리스트, 배포 단계별 절차, 장애 시 롤백 절차
|
|
||||||
- 입력: doc_readme3.md 섹션 16 참고
|
|
||||||
- 출력: `DEPLOYMENT_PROCEDURES.md` 또는 운영 가이드 통합
|
|
||||||
- 완료 기준: 절차 문서 완성, 체크리스트 확인 가능
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 3. 작업 우선순위 및 권장 순서
|
|
||||||
|
|
||||||
### 3.1 필수 우선 작업 (Phase 1 완료 필수)
|
|
||||||
|
|
||||||
1. Add production compose file
|
|
||||||
2. Create production frontend Dockerfile (multi-stage build)
|
|
||||||
3. Harden backend Dockerfile for production
|
|
||||||
4. Define host paths and named volumes for persistence
|
|
||||||
|
|
||||||
**목표**: 기본 배포 구조 완성, Linux 서버에서 최소 기동 가능 상태
|
|
||||||
|
|
||||||
### 3.2 보안 필수 작업 (Phase 2 완료 권장)
|
|
||||||
|
|
||||||
5. Provide Nginx reverse-proxy and frontend static config
|
|
||||||
6. Externalize and secure environment variables
|
|
||||||
7. Define TLS certificate handling strategy
|
|
||||||
8. Security review: non-root users, image scan, secret rotation
|
|
||||||
|
|
||||||
**목표**: 운영 환경 최소 보안 기준 충족
|
|
||||||
|
|
||||||
### 3.3 운영 안정화 작업 (Phase 3 진행)
|
|
||||||
|
|
||||||
9. Add healthcheck and readiness endpoint in backend
|
|
||||||
10. Add logging and log rotation guidance
|
|
||||||
11. Document backup and restore procedures
|
|
||||||
12. Add smoke tests and post-deploy checks
|
|
||||||
|
|
||||||
**목표**: 배포 후 문제 식별 및 백업/복구 가능 상태
|
|
||||||
|
|
||||||
### 3.4 장기 운영 자동화 (Phase 4-5는 선택)
|
|
||||||
|
|
||||||
13. Prepare CI/CD build & push scripts
|
|
||||||
14. Create deploy directory with compose.prod and nginx configs
|
|
||||||
15. Write rollout and rollback procedures
|
|
||||||
|
|
||||||
**목표**: 반복 배포 자동화, 절차 표준화
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 4. 작업 환경 및 검증 기준
|
|
||||||
|
|
||||||
### 4.1 개발/테스트 환경
|
|
||||||
|
|
||||||
- 로컬 Linux VM 또는 WSL에서 Phase 1-2 테스트
|
|
||||||
- Docker Desktop or Docker Engine 필수
|
|
||||||
- 외부 테스트 MySQL 또는 mock DB
|
|
||||||
|
|
||||||
### 4.2 스테이징 환경
|
|
||||||
|
|
||||||
- 실제 Linux 서버에서 전체 배포 절차 테스트
|
|
||||||
- 운영 환경과 동일 아키텍처
|
|
||||||
|
|
||||||
### 4.3 운영 배포
|
|
||||||
|
|
||||||
- 위 모든 Phase 완료 후 진행
|
|
||||||
- 백업 확인 후 배포
|
|
||||||
- 배포 후 smoke test 자동 실행
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 5. 진행 추적
|
|
||||||
|
|
||||||
진행 상황은 아래 상태로 추적한다.
|
|
||||||
|
|
||||||
- `not-started`: 아직 시작 안 함
|
|
||||||
- `in-progress`: 현재 진행 중
|
|
||||||
- `completed`: 완료됨
|
|
||||||
|
|
||||||
현재 상태는 모두 `not-started`이며, Phase 1 우선 순위 항목부터 순차적으로 진행한다.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 6. 예상 일정
|
|
||||||
|
|
||||||
- Phase 1 (핵심 배포 파일): 1-2일
|
|
||||||
- Phase 2 (네트워킹 및 보안): 1-2일
|
|
||||||
- Phase 3 (모니터링 및 운영 준비): 1일
|
|
||||||
- Phase 4-5 (자동화 및 절차): 1-2일
|
|
||||||
|
|
||||||
**전체 예상 소요 시간**: 4-7일
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 7. 추가 고려사항
|
|
||||||
|
|
||||||
1. 현재 `doc_readme3.md`는 가이드 문서이고, 이 로드맵은 구현 작업 목록이다.
|
|
||||||
2. Phase 1-2 완료 후 실제 코드 커밋은 `Dockerizing` 브랜치에만 한다.
|
|
||||||
3. 각 Phase 완료 후 관련 문서도 함께 업데이트한다.
|
|
||||||
4. 운영 전환 전에 최소 1회 스테이징 환경에서 전체 배포 절차 테스트를 수행한다.
|
|
||||||
1
public/qrcode.min.js
vendored
Normal file
1
public/qrcode.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
@@ -1,24 +0,0 @@
|
|||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config();
|
|
||||||
|
|
||||||
async function analyzeCodes() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
// 새 자산들의 연도 분포 확인
|
|
||||||
const [years] = await connection.query('SELECT DISTINCT purchase_date FROM asset_core WHERE id LIKE "PC_20260615_%"');
|
|
||||||
console.log('New assets years:', years.map(y => y.purchase_date));
|
|
||||||
|
|
||||||
// 기존 자산 코드 패턴 확인
|
|
||||||
const [existing] = await connection.query('SELECT asset_code FROM asset_core WHERE asset_code LIKE "PC-%" LIMIT 5');
|
|
||||||
console.log('Existing code sample:', existing);
|
|
||||||
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
analyzeCodes().catch(console.error);
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
const XLSX = require('xlsx');
|
|
||||||
const workbook = XLSX.readFile('backupDB_20260602.xlsx');
|
|
||||||
console.log('Sheet Names:', workbook.SheetNames);
|
|
||||||
if (workbook.SheetNames.includes('system_users')) {
|
|
||||||
const sheet = workbook.Sheets['system_users'];
|
|
||||||
const data = XLSX.utils.sheet_to_json(sheet);
|
|
||||||
console.log('system_users found! Count:', data.length);
|
|
||||||
console.log('Sample:', data.slice(0, 2));
|
|
||||||
} else {
|
|
||||||
console.log('system_users sheet not found in backupDB_20260602.xlsx');
|
|
||||||
}
|
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config();
|
|
||||||
|
|
||||||
async function checkCodes() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('--- Asset Codes Sample ---');
|
|
||||||
const [rows] = await connection.query('SELECT id, asset_code, purchase_date FROM asset_core WHERE id LIKE "PC_20260615_%" LIMIT 10');
|
|
||||||
console.log(rows);
|
|
||||||
|
|
||||||
console.log('\n--- Other Asset Codes Sample ---');
|
|
||||||
const [rows2] = await connection.query('SELECT id, asset_code, purchase_date FROM asset_core WHERE id NOT LIKE "PC_20260615_%" AND asset_code IS NOT NULL LIMIT 5');
|
|
||||||
console.log(rows2);
|
|
||||||
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
checkCodes().catch(console.error);
|
|
||||||
@@ -1,40 +0,0 @@
|
|||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config();
|
|
||||||
|
|
||||||
async function checkPublicPCs() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('🔍 공용 PC(Public PC)로 추정되는 자산 조회 중...');
|
|
||||||
|
|
||||||
// 사번이 없거나, 사용자명에 '공용'이 포함된 데이터 조회
|
|
||||||
const [rows] = await connection.query(`
|
|
||||||
SELECT id, asset_code, user_current, emp_no, current_dept, asset_type
|
|
||||||
FROM asset_core
|
|
||||||
WHERE (emp_no IS NULL OR emp_no = '' OR user_current LIKE '%공용%')
|
|
||||||
AND id LIKE 'PC_20260615_%'
|
|
||||||
`);
|
|
||||||
|
|
||||||
console.log(`📊 발견된 공용 PC 후보: ${rows.length}건`);
|
|
||||||
|
|
||||||
if (rows.length > 0) {
|
|
||||||
console.table(rows.slice(0, 20)); // 상위 20개 샘플 출력
|
|
||||||
|
|
||||||
// 요약 통계
|
|
||||||
const summary = {
|
|
||||||
only_no_emp: rows.filter(r => (!r.emp_no) && !r.user_current.includes('공용')).length,
|
|
||||||
only_public_name: rows.filter(r => r.emp_no && r.user_current.includes('공용')).length,
|
|
||||||
both: rows.filter(r => (!r.emp_no) && r.user_current.includes('공용')).length
|
|
||||||
};
|
|
||||||
console.log('\n📈 요약 통계:', summary);
|
|
||||||
}
|
|
||||||
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
checkPublicPCs().catch(console.error);
|
|
||||||
@@ -1,77 +0,0 @@
|
|||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config();
|
|
||||||
|
|
||||||
async function updateAndCompare() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('🚀 [Step 1 & 2] "undefined" 사번 및 빈 사용자명 정리 중...');
|
|
||||||
const [updateResult] = await connection.query(`
|
|
||||||
UPDATE asset_core
|
|
||||||
SET user_current = '공용', emp_no = NULL
|
|
||||||
WHERE id LIKE "PC_20260615_%" AND (emp_no = 'undefined' OR emp_no IS NULL OR emp_no = '')
|
|
||||||
`);
|
|
||||||
console.log(`✅ 업데이트 완료: ${updateResult.affectedRows}건`);
|
|
||||||
|
|
||||||
console.log('\n🔍 [Step 3] 엑셀 데이터와 DB asset_type 비교 분석 중...');
|
|
||||||
const XLSX = require('xlsx');
|
|
||||||
const workbook = XLSX.readFile('asset_pc (2026.06.15).xlsx');
|
|
||||||
const sheet = workbook.Sheets[workbook.SheetNames[0]];
|
|
||||||
const excelData = XLSX.utils.sheet_to_json(sheet);
|
|
||||||
|
|
||||||
// DB 데이터 로드
|
|
||||||
const [dbRows] = await connection.query('SELECT id, asset_type, user_current, emp_no FROM asset_core WHERE id LIKE "PC_20260615_%"');
|
|
||||||
const dbMap = new Map();
|
|
||||||
dbRows.forEach(r => dbMap.set(r.id, r));
|
|
||||||
|
|
||||||
const mismatches = [];
|
|
||||||
const publicButExcelPersonal = [];
|
|
||||||
|
|
||||||
for (let i = 0; i < excelData.length; i++) {
|
|
||||||
const excelRow = excelData[i];
|
|
||||||
const assetId = `PC_20260615_${String(i + 1).padStart(4, '0')}`;
|
|
||||||
const dbRow = dbMap.get(assetId);
|
|
||||||
|
|
||||||
if (!dbRow) continue;
|
|
||||||
|
|
||||||
const excelType = excelRow.asset_type || '개인PC';
|
|
||||||
|
|
||||||
// 1. 단순 타입 불일치 체크
|
|
||||||
if (dbRow.asset_type !== excelType) {
|
|
||||||
mismatches.push({
|
|
||||||
id: assetId,
|
|
||||||
excel_type: excelType,
|
|
||||||
db_type: dbRow.asset_type,
|
|
||||||
user: dbRow.user_current
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// 2. 엑셀은 '개인PC'인데 데이터는 공용(사번없음)인 경우 탐색
|
|
||||||
if (excelType === '개인PC' && (!dbRow.emp_no || dbRow.user_current === '공용')) {
|
|
||||||
publicButExcelPersonal.push({
|
|
||||||
id: assetId,
|
|
||||||
excel_user: excelRow.user_current,
|
|
||||||
excel_dept: excelRow.current_dept,
|
|
||||||
db_user: dbRow.user_current
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`\n📊 분석 결과:`);
|
|
||||||
console.log(`- 엑셀과 DB의 asset_type 불일치: ${mismatches.length}건`);
|
|
||||||
console.log(`- 엑셀은 '개인PC'이나 사번이 없어 '공용'으로 잡힌 항목: ${publicButExcelPersonal.length}건`);
|
|
||||||
|
|
||||||
if (publicButExcelPersonal.length > 0) {
|
|
||||||
console.log('\n⚠️ 엑셀은 개인PC이나 데이터가 미비한 항목 (상위 10개):');
|
|
||||||
console.table(publicButExcelPersonal.slice(0, 10));
|
|
||||||
}
|
|
||||||
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
updateAndCompare().catch(console.error);
|
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config();
|
|
||||||
|
|
||||||
async function debugPublic() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
const [rows] = await connection.query(`
|
|
||||||
SELECT user_current, emp_no, COUNT(*) as count
|
|
||||||
FROM asset_core
|
|
||||||
WHERE id LIKE "PC_20260615_%"
|
|
||||||
GROUP BY user_current, emp_no
|
|
||||||
HAVING emp_no IS NULL OR emp_no = '' OR user_current LIKE '%공용%' OR user_current = ''
|
|
||||||
`);
|
|
||||||
|
|
||||||
console.table(rows);
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
debugPublic().catch(console.error);
|
|
||||||
@@ -1,69 +0,0 @@
|
|||||||
const XLSX = require('xlsx');
|
|
||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config();
|
|
||||||
|
|
||||||
async function deepAudit() {
|
|
||||||
const workbook = XLSX.readFile('asset_pc (2026.06.15).xlsx');
|
|
||||||
const sheet = workbook.Sheets[workbook.SheetNames[0]];
|
|
||||||
const excelData = XLSX.utils.sheet_to_json(sheet);
|
|
||||||
|
|
||||||
console.log('📊 [Excel Audit] Total Rows:', excelData.length);
|
|
||||||
|
|
||||||
// 1. 엑셀 내 asset_type 종류 확인
|
|
||||||
const excelTypes = new Set();
|
|
||||||
excelData.forEach(r => excelTypes.add(r.asset_type));
|
|
||||||
console.log('Excel Asset Types:', Array.from(excelTypes));
|
|
||||||
|
|
||||||
// 2. '공용' 키워드가 들어간 모든 행 추출
|
|
||||||
const publicKeywords = ['공용', '공통', '테스트', 'TEST'];
|
|
||||||
const potentialPublicInExcel = excelData.filter(r => {
|
|
||||||
const name = String(r.user_current || '');
|
|
||||||
const type = String(r.asset_type || '');
|
|
||||||
const memo = String(r.memo || '');
|
|
||||||
return publicKeywords.some(k => name.includes(k) || type.includes(k) || memo.includes(k)) || !r.emp_no;
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log(`\n🔍 [Potential Public/Issue Rows in Excel]: ${potentialPublicInExcel.length}건`);
|
|
||||||
console.table(potentialPublicInExcel.slice(0, 30).map(r => ({
|
|
||||||
emp_no: r.emp_no,
|
|
||||||
user: r.user_current,
|
|
||||||
dept: r.current_dept,
|
|
||||||
type: r.asset_type,
|
|
||||||
memo: r.memo
|
|
||||||
})));
|
|
||||||
|
|
||||||
// 3. DB와 대조 (특히 엑셀엔 사번이 있는데 DB엔 공용으로 된 게 있는지)
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
const [dbRows] = await connection.query('SELECT id, user_current, emp_no, asset_type FROM asset_core WHERE id LIKE "PC_20260615_%"');
|
|
||||||
|
|
||||||
// 엑셀은 개인PC인데 DB는 공용인 경우 (또는 그 반대)
|
|
||||||
const issues = [];
|
|
||||||
for (let i = 0; i < excelData.length; i++) {
|
|
||||||
const ex = excelData[i];
|
|
||||||
const id = `PC_20260615_${String(i + 1).padStart(4, '0')}`;
|
|
||||||
const db = dbRows.find(r => r.id === id);
|
|
||||||
|
|
||||||
if (!db) continue;
|
|
||||||
|
|
||||||
const isExcelPublic = !ex.emp_no || String(ex.user_current).includes('공용');
|
|
||||||
const isDbPublic = !db.emp_no || String(db.user_current).includes('공용');
|
|
||||||
|
|
||||||
if (isExcelPublic !== isDbPublic) {
|
|
||||||
issues.push({ id, excel_user: ex.user_current, db_user: db.user_current, excel_emp: ex.emp_no, db_emp: db.emp_no });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`\n⚠️ [Consistency Issues]: ${issues.length}건`);
|
|
||||||
if (issues.length > 0) console.table(issues);
|
|
||||||
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
deepAudit().catch(console.error);
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
const XLSX = require('xlsx');
|
|
||||||
const mysql = require('mysql2/promise');
|
|
||||||
const dotenv = require('dotenv');
|
|
||||||
const path = require('path');
|
|
||||||
|
|
||||||
dotenv.config({ path: path.join(__dirname, '../.env') });
|
|
||||||
|
|
||||||
const { DB_HOST, DB_USER, DB_PASS, DB_NAME, DB_PORT } = process.env;
|
|
||||||
|
|
||||||
async function extractFailures() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: DB_HOST,
|
|
||||||
user: DB_USER,
|
|
||||||
password: DB_PASS,
|
|
||||||
database: DB_NAME,
|
|
||||||
port: parseInt(DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('🔍 실패 데이터 추출 중...');
|
|
||||||
|
|
||||||
const workbook = XLSX.readFile('asset_pc (2026.06.15).xlsx');
|
|
||||||
const sheet = workbook.Sheets[workbook.SheetNames[0]];
|
|
||||||
const rawData = XLSX.utils.sheet_to_json(sheet);
|
|
||||||
|
|
||||||
// 현재 DB에 존재하는 모든 asset_core ID 조회
|
|
||||||
const [existingRows] = await connection.query('SELECT id FROM asset_core');
|
|
||||||
const existingIds = new Set(existingRows.map(r => r.id));
|
|
||||||
|
|
||||||
const failures = [];
|
|
||||||
|
|
||||||
for (let i = 0; i < rawData.length; i++) {
|
|
||||||
const row = rawData[i];
|
|
||||||
const assetId = `PC_20260615_${String(i + 1).padStart(4, '0')}`;
|
|
||||||
|
|
||||||
// DB에 해당 ID가 없는 경우 = 실패(충돌 등의 이유로 입력되지 않음) 또는 스킵된 데이터
|
|
||||||
// 하지만 이전 로그에서 'Duplicate entry'로 에러가 났던 항목들을 찾는 것이 목적
|
|
||||||
// 로직상 ID 생성 규칙에 따라 해당 ID가 DB에 없으면 입력에 실패한 행임
|
|
||||||
if (!existingIds.has(assetId)) {
|
|
||||||
failures.push({
|
|
||||||
excel_row: i + 2,
|
|
||||||
generated_id: assetId,
|
|
||||||
...row
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (failures.length > 0) {
|
|
||||||
const newWb = XLSX.utils.book_new();
|
|
||||||
const newWs = XLSX.utils.json_to_sheet(failures);
|
|
||||||
XLSX.utils.book_append_sheet(newWb, newWs, 'Failures');
|
|
||||||
const fileName = 'asset_pc_failures_20260615.xlsx';
|
|
||||||
XLSX.writeFile(newWb, fileName);
|
|
||||||
console.log(`✅ 추출 완료: ${failures.length}건의 실패 데이터를 ${fileName}에 저장했습니다.`);
|
|
||||||
} else {
|
|
||||||
console.log('입력되지 않은 데이터가 없습니다.');
|
|
||||||
}
|
|
||||||
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
extractFailures().catch(console.error);
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config();
|
|
||||||
|
|
||||||
async function findPotentialPublic() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('--- Searching for rows with no emp_no or "공용" in user_current ---');
|
|
||||||
|
|
||||||
// 사번이 'undefined', 'null', 빈값, 또는 사용자명에 '공용'이 들어간 데이터
|
|
||||||
const [rows] = await connection.query(`
|
|
||||||
SELECT id, user_current, emp_no
|
|
||||||
FROM asset_core
|
|
||||||
WHERE id LIKE "PC_20260615_%"
|
|
||||||
AND (emp_no IS NULL OR emp_no = '' OR emp_no = 'undefined' OR user_current LIKE '%공용%')
|
|
||||||
`);
|
|
||||||
|
|
||||||
console.log('Count:', rows.length);
|
|
||||||
if (rows.length > 0) console.table(rows);
|
|
||||||
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
findPotentialPublic().catch(console.error);
|
|
||||||
@@ -1,47 +0,0 @@
|
|||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config();
|
|
||||||
|
|
||||||
async function fixAssetTypes() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('🚀 [데이터 정상화] 사번 기준 자산 유형 재설정 시작...');
|
|
||||||
|
|
||||||
// 1. 사번이 있는 모든 신규 자산을 '개인PC'로 강제 전환
|
|
||||||
const [personalResult] = await connection.query(`
|
|
||||||
UPDATE asset_core
|
|
||||||
SET asset_type = '개인PC'
|
|
||||||
WHERE id LIKE "PC_20260615_%"
|
|
||||||
AND emp_no IS NOT NULL
|
|
||||||
AND emp_no != ''
|
|
||||||
`);
|
|
||||||
console.log(`✅ 개인PC 정상화 완료: ${personalResult.affectedRows}건 (사번 존재 항목)`);
|
|
||||||
|
|
||||||
// 2. 사번이 없는 모든 신규 자산을 '공용PC'로 강제 전환
|
|
||||||
const [publicResult] = await connection.query(`
|
|
||||||
UPDATE asset_core
|
|
||||||
SET asset_type = '공용PC', user_current = '공용'
|
|
||||||
WHERE id LIKE "PC_20260615_%"
|
|
||||||
AND (emp_no IS NULL OR emp_no = '')
|
|
||||||
`);
|
|
||||||
console.log(`✅ 공용PC 정상화 완료: ${publicResult.affectedRows}건 (사번 부재 항목)`);
|
|
||||||
|
|
||||||
// 3. 최종 결과 확인
|
|
||||||
const [rows] = await connection.query(`
|
|
||||||
SELECT asset_type, COUNT(*) as count
|
|
||||||
FROM asset_core
|
|
||||||
WHERE id LIKE "PC_20260615_%"
|
|
||||||
GROUP BY asset_type
|
|
||||||
`);
|
|
||||||
console.log('\n📊 최종 자산 유형 분포:');
|
|
||||||
console.table(rows);
|
|
||||||
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
fixAssetTypes().catch(console.error);
|
|
||||||
@@ -1,118 +0,0 @@
|
|||||||
import mysql from 'mysql2/promise';
|
|
||||||
import dotenv from 'dotenv';
|
|
||||||
|
|
||||||
dotenv.config();
|
|
||||||
|
|
||||||
const pool = mysql.createPool({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306'),
|
|
||||||
});
|
|
||||||
|
|
||||||
// 하드웨어 출시 연도 데이터베이스 (CPU/GPU)
|
|
||||||
const RELEASE_DATES = {
|
|
||||||
// Intel CPU Generations (Mainstream desktop release month/year)
|
|
||||||
'i9-14': '2023-10', 'i7-14': '2023-10', 'i5-14': '2023-10',
|
|
||||||
'i9-13': '2022-10', 'i7-13': '2022-10', 'i5-13': '2022-10',
|
|
||||||
'i9-12': '2021-11', 'i7-12': '2021-11', 'i5-12': '2021-11',
|
|
||||||
'i9-11': '2021-03', 'i7-11': '2021-03', 'i5-11': '2021-03',
|
|
||||||
'i9-10': '2020-05', 'i7-10': '2020-05', 'i5-10': '2020-05',
|
|
||||||
'i9-9': '2018-10', 'i7-9': '2018-10', 'i5-9': '2018-10',
|
|
||||||
'i7-8': '2017-10', 'i5-8': '2017-10',
|
|
||||||
'i7-7': '2017-01', 'i5-7': '2017-01',
|
|
||||||
'i7-6': '2015-08', 'i5-6': '2015-08',
|
|
||||||
'i7-4': '2013-06', 'i5-4': '2013-06',
|
|
||||||
'i7-3': '2012-04', 'i5-3': '2012-04',
|
|
||||||
'i7-2': '2011-01', 'i5-2': '2011-01',
|
|
||||||
|
|
||||||
// NVIDIA GPU Series
|
|
||||||
'RTX 4090': '2022-10', 'RTX 4080': '2022-11', 'RTX 4070': '2023-04', 'RTX 4060': '2023-06',
|
|
||||||
'RTX 3090': '2020-09', 'RTX 3080': '2020-09', 'RTX 3070': '2020-10', 'RTX 3060': '2021-02',
|
|
||||||
'RTX 2080': '2018-09', 'RTX 2070': '2018-10', 'RTX 2060': '2019-01',
|
|
||||||
'GTX 1660': '2019-03', 'GTX 1650': '2019-04',
|
|
||||||
'GTX 1080': '2016-05', 'GTX 1070': '2016-06', 'GTX 1060': '2016-07', 'GTX 1050': '2016-10',
|
|
||||||
'GTX 980': '2014-09', 'GTX 970': '2014-09', 'GTX 960': '2015-01'
|
|
||||||
};
|
|
||||||
|
|
||||||
function inferDateFromSpecs(cpu, gpu) {
|
|
||||||
const cpuStr = (cpu || '').toUpperCase();
|
|
||||||
const gpuStr = (gpu || '').toUpperCase();
|
|
||||||
|
|
||||||
let inferred = null;
|
|
||||||
|
|
||||||
// 1. GPU 기준 (최신 그래픽카드가 꽂혀있으면 그 시기 이후 구매일 확률이 높음)
|
|
||||||
for (const [key, date] of Object.entries(RELEASE_DATES)) {
|
|
||||||
if (gpuStr.includes(key)) {
|
|
||||||
inferred = date;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 2. CPU 기준 (GPU에서 못 찾았거나, CPU가 더 최신일 경우)
|
|
||||||
if (!inferred) {
|
|
||||||
for (const [key, date] of Object.entries(RELEASE_DATES)) {
|
|
||||||
// i7-13700 등을 찾기 위해 정규식 또는 포함 여부 확인
|
|
||||||
if (cpuStr.includes(key)) {
|
|
||||||
inferred = date;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return inferred ? `${inferred}-01` : null;
|
|
||||||
}
|
|
||||||
|
|
||||||
async function run() {
|
|
||||||
const connection = await pool.getConnection();
|
|
||||||
try {
|
|
||||||
const [rows] = await connection.query(`
|
|
||||||
SELECT c.id, c.asset_code, c.purchase_date, s.cpu, s.gpu
|
|
||||||
FROM asset_core c
|
|
||||||
LEFT JOIN asset_spec s ON c.id = s.asset_id
|
|
||||||
`);
|
|
||||||
|
|
||||||
const updates = [];
|
|
||||||
const unchanged = [];
|
|
||||||
|
|
||||||
for (const row of rows) {
|
|
||||||
const currentVal = (row.purchase_date || '').trim();
|
|
||||||
|
|
||||||
// 구매일자가 없거나 부정확한 경우만 처리
|
|
||||||
if (!currentVal || currentVal === '-' || currentVal === 'undefined' || currentVal.startsWith('2024-01-01')) {
|
|
||||||
const specDate = inferDateFromSpecs(row.cpu, row.gpu);
|
|
||||||
|
|
||||||
if (specDate) {
|
|
||||||
updates.push({ id: row.id, date: specDate, code: row.asset_code, cpu: row.cpu, gpu: row.gpu });
|
|
||||||
} else {
|
|
||||||
unchanged.push({ code: row.asset_code, cpu: row.cpu, gpu: row.gpu });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`🚀 스펙 분석 결과: ${updates.length}건의 자산 구매일자를 보정합니다.`);
|
|
||||||
|
|
||||||
for (const item of updates) {
|
|
||||||
await connection.query('UPDATE asset_core SET purchase_date = ? WHERE id = ?', [item.date, item.id]);
|
|
||||||
console.log(`[Update] ${item.code.padEnd(15)} | CPU: ${String(item.cpu).padEnd(20)} | GPU: ${String(item.gpu).padEnd(15)} -> ${item.date}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (unchanged.length > 0) {
|
|
||||||
console.log('\n⚠️ 스펙 정보를 찾을 수 없어 보정하지 못한 자산:');
|
|
||||||
unchanged.forEach(u => {
|
|
||||||
if (u.code) console.log(`[Skip] ${u.code.padEnd(15)} | CPU: ${u.cpu || '-'} | GPU: ${u.gpu || '-'}`);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`\n✅ 완료: ${updates.length}건 보정됨.`);
|
|
||||||
|
|
||||||
} catch (err) {
|
|
||||||
console.error('Error:', err);
|
|
||||||
} finally {
|
|
||||||
connection.release();
|
|
||||||
pool.end();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
run();
|
|
||||||
@@ -1,128 +0,0 @@
|
|||||||
import mysql from 'mysql2/promise';
|
|
||||||
import dotenv from 'dotenv';
|
|
||||||
|
|
||||||
dotenv.config();
|
|
||||||
|
|
||||||
const pool = mysql.createPool({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306'),
|
|
||||||
});
|
|
||||||
|
|
||||||
// 하드웨어 출시 연도/월 데이터베이스
|
|
||||||
const RELEASE_DATES = {
|
|
||||||
// Intel CPU
|
|
||||||
'i9-14': '2023-10', 'i7-14': '2023-10', 'i5-14': '2023-10',
|
|
||||||
'i9-13': '2022-10', 'i7-13': '2022-10', 'i5-13': '2022-10',
|
|
||||||
'i9-12': '2021-11', 'i7-12': '2021-11', 'i5-12': '2021-11',
|
|
||||||
'i9-11': '2021-03', 'i7-11': '2021-03', 'i5-11': '2021-03',
|
|
||||||
'i9-10': '2020-05', 'i7-10': '2020-05', 'i5-10': '2020-05',
|
|
||||||
'i9-9': '2018-10', 'i7-9': '2018-10', 'i5-9': '2018-10',
|
|
||||||
'i7-8': '2017-10', 'i5-8': '2017-10',
|
|
||||||
'i7-7': '2017-01', 'i5-7': '2017-01',
|
|
||||||
'i7-6': '2015-08', 'i5-6': '2015-08',
|
|
||||||
'i7-5': '2014-06', 'i5-5': '2015-06', // Broadwell
|
|
||||||
'i7-4': '2013-06', 'i5-4': '2013-06',
|
|
||||||
'i7-3': '2012-04', 'i5-3': '2012-04',
|
|
||||||
'i7-2': '2011-01', 'i5-2': '2011-01',
|
|
||||||
|
|
||||||
// NVIDIA GPU
|
|
||||||
'RTX 40': '2022-10',
|
|
||||||
'RTX 30': '2020-09',
|
|
||||||
'RTX 20': '2018-09',
|
|
||||||
'GTX 16': '2019-02',
|
|
||||||
'GTX 10': '2016-05',
|
|
||||||
'GTX 9': '2014-09',
|
|
||||||
'GTX 750': '2014-02',
|
|
||||||
'GTX 7': '2013-05',
|
|
||||||
'GTX 6': '2012-03'
|
|
||||||
};
|
|
||||||
|
|
||||||
// 출시 연도만 있는 경우 (지시에 따라 후속년도 12월 적용을 위함)
|
|
||||||
const YEAR_ONLY = {
|
|
||||||
'I5-4': 2013,
|
|
||||||
'I5-6': 2015,
|
|
||||||
'I7-7': 2017,
|
|
||||||
'GTX 750': 2014
|
|
||||||
};
|
|
||||||
|
|
||||||
function inferDateFromSpecs(cpu, gpu) {
|
|
||||||
const cpuStr = (cpu || '').toUpperCase();
|
|
||||||
const gpuStr = (gpu || '').toUpperCase();
|
|
||||||
|
|
||||||
let latestYear = 0;
|
|
||||||
let latestMonth = 0;
|
|
||||||
|
|
||||||
// 모든 매핑 데이터를 순회하며 가장 최신 날짜를 찾음
|
|
||||||
for (const [key, dateStr] of Object.entries(RELEASE_DATES)) {
|
|
||||||
if (cpuStr.includes(key) || gpuStr.includes(key)) {
|
|
||||||
const [y, m] = dateStr.split('-').map(Number);
|
|
||||||
if (y > latestYear || (y === latestYear && m > latestMonth)) {
|
|
||||||
latestYear = y;
|
|
||||||
latestMonth = m;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 매칭된 정보가 있는 경우
|
|
||||||
if (latestYear > 0) {
|
|
||||||
// 월 정보가 명확히 매핑된 경우 (RELEASE_DATES 사용)
|
|
||||||
// 하지만 지시사항에 따라 "월을 못찾으면 12월" & "후속년도" 규칙 적용 여부 판단
|
|
||||||
// RELEASE_DATES는 월이 이미 있으므로 그대로 사용하되,
|
|
||||||
// 만약 YEAR_ONLY에만 걸리는 경우를 위해 로직 보강
|
|
||||||
return `${latestYear}-${String(latestMonth).padStart(2, '0')}-01`;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 연도만 매칭되는 경우 (지시사항: 후속년도 12월)
|
|
||||||
for (const [key, year] of Object.entries(YEAR_ONLY)) {
|
|
||||||
if (cpuStr.includes(key) || gpuStr.includes(key)) {
|
|
||||||
return `${year + 1}-12-01`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
async function run() {
|
|
||||||
const connection = await pool.getConnection();
|
|
||||||
try {
|
|
||||||
const [rows] = await connection.query(`
|
|
||||||
SELECT c.id, c.asset_code, c.purchase_date, s.cpu, s.gpu
|
|
||||||
FROM asset_core c
|
|
||||||
LEFT JOIN asset_spec s ON c.id = s.asset_id
|
|
||||||
`);
|
|
||||||
|
|
||||||
const updates = [];
|
|
||||||
|
|
||||||
for (const row of rows) {
|
|
||||||
const currentVal = (row.purchase_date || '').trim();
|
|
||||||
|
|
||||||
// 구매일자가 없거나 '-', 'undefined'인 경우 + 혹은 아직 보정이 필요한 자산
|
|
||||||
if (!currentVal || currentVal === '-' || currentVal === 'undefined' || currentVal.startsWith('0000') || currentVal === '2024-01-01') {
|
|
||||||
const specDate = inferDateFromSpecs(row.cpu, row.gpu);
|
|
||||||
if (specDate) {
|
|
||||||
updates.push({ id: row.id, date: specDate, code: row.asset_code, cpu: row.cpu, gpu: row.gpu });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`🚀 지시사항 반영: ${updates.length}건의 자산을 보정합니다. (후속년도/12월 규칙 적용)`);
|
|
||||||
|
|
||||||
for (const item of updates) {
|
|
||||||
await connection.query('UPDATE asset_core SET purchase_date = ? WHERE id = ?', [item.date, item.id]);
|
|
||||||
console.log(`[Update] ${item.code.padEnd(15)} | CPU: ${String(item.cpu).padEnd(20)} | GPU: ${String(item.gpu).padEnd(15)} -> ${item.date}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`\n✅ 완료: ${updates.length}건 보정됨.`);
|
|
||||||
|
|
||||||
} catch (err) {
|
|
||||||
console.error('Error:', err);
|
|
||||||
} finally {
|
|
||||||
connection.release();
|
|
||||||
pool.end();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
run();
|
|
||||||
@@ -1,88 +0,0 @@
|
|||||||
import mysql from 'mysql2/promise';
|
|
||||||
import dotenv from 'dotenv';
|
|
||||||
|
|
||||||
dotenv.config();
|
|
||||||
|
|
||||||
const pool = mysql.createPool({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306'),
|
|
||||||
});
|
|
||||||
|
|
||||||
async function run() {
|
|
||||||
const connection = await pool.getConnection();
|
|
||||||
try {
|
|
||||||
// 먼저 잘못 들어간 0000-00-01 등 복구
|
|
||||||
console.log('잘못된 형식(0000-00-01 등)을 초기화합니다...');
|
|
||||||
await connection.query("UPDATE asset_core SET purchase_date = '-' WHERE purchase_date LIKE '0000%' OR purchase_date = '2020-01-01'");
|
|
||||||
|
|
||||||
const [rows] = await connection.query('SELECT id, asset_code, purchase_date, category FROM asset_core');
|
|
||||||
|
|
||||||
const updates = [];
|
|
||||||
const missing = [];
|
|
||||||
|
|
||||||
for (const row of rows) {
|
|
||||||
const code = (row.asset_code || '').trim();
|
|
||||||
const currentVal = (row.purchase_date || '').trim();
|
|
||||||
|
|
||||||
// 구매일자가 없거나 '-', 'undefined' 인 경우 대상
|
|
||||||
if (!currentVal || currentVal === '-' || currentVal === 'undefined') {
|
|
||||||
let inferredDate = null;
|
|
||||||
|
|
||||||
// 1. PREFIX-YYYYMM-NNNN 형식 (예: PC-202406-0001)
|
|
||||||
const match6 = code.match(/[A-Z]+-(\d{4})(0[1-9]|1[0-2])-\d+/);
|
|
||||||
if (match6) {
|
|
||||||
inferredDate = `${match6[1]}-${match6[2]}-01`;
|
|
||||||
} else {
|
|
||||||
// 2. PREFIX-YYYYNN 형식 (예: PC-202423) -> 연도만 있고 뒤에 순번 2자리
|
|
||||||
const matchYearSeq = code.match(/[A-Z]+-(20\d{2})(\d{2})$/);
|
|
||||||
if (matchYearSeq) {
|
|
||||||
inferredDate = `${matchYearSeq[1]}-01-01`; // 월을 모르므로 1월로 통일
|
|
||||||
} else {
|
|
||||||
// 3. PREFIX-YYNNN 형식 (예: PC-24001)
|
|
||||||
const matchShort = code.match(/[A-Z]+-(1\d|2\d)(\d{3})/);
|
|
||||||
if (matchShort) {
|
|
||||||
inferredDate = `20${matchShort[1]}-01-01`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 0000 등의 잘못된 매칭 방지
|
|
||||||
if (inferredDate && !inferredDate.startsWith('0000')) {
|
|
||||||
updates.push({ id: row.id, date: inferredDate, code: code });
|
|
||||||
} else {
|
|
||||||
missing.push({ id: row.id, code: code, category: row.category });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`총 ${updates.length}건의 자산을 업데이트합니다.`);
|
|
||||||
for (const item of updates) {
|
|
||||||
await connection.query('UPDATE asset_core SET purchase_date = ? WHERE id = ?', [item.date, item.id]);
|
|
||||||
console.log(`[Update] ${item.code} -> ${item.date}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\n--- 구매일자를 추정할 수 없는 자산 목록 ---');
|
|
||||||
if (missing.length === 0) {
|
|
||||||
console.log('없음');
|
|
||||||
} else {
|
|
||||||
// 중복 제거 및 정렬하여 보고
|
|
||||||
const uniqueMissing = missing.filter(m => m.code !== '');
|
|
||||||
uniqueMissing.forEach(m => {
|
|
||||||
console.log(`[Missing] 코드: ${m.code.padEnd(20)} | 카테고리: ${m.category}`);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`\n완료: ${updates.length}건 업데이트됨, ${missing.length}건 미결정.`);
|
|
||||||
|
|
||||||
} catch (err) {
|
|
||||||
console.error('Error:', err);
|
|
||||||
} finally {
|
|
||||||
connection.release();
|
|
||||||
pool.end();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
run();
|
|
||||||
@@ -1,122 +0,0 @@
|
|||||||
const XLSX = require('xlsx');
|
|
||||||
const mysql = require('mysql2/promise');
|
|
||||||
const dotenv = require('dotenv');
|
|
||||||
const path = require('path');
|
|
||||||
|
|
||||||
dotenv.config({ path: path.join(__dirname, '../.env') });
|
|
||||||
|
|
||||||
const { DB_HOST, DB_USER, DB_PASS, DB_NAME, DB_PORT } = process.env;
|
|
||||||
|
|
||||||
async function importAssets() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: DB_HOST,
|
|
||||||
user: DB_USER,
|
|
||||||
password: DB_PASS,
|
|
||||||
database: DB_NAME,
|
|
||||||
port: parseInt(DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('🚀 [Step 1] 데이터 로드 및 사전 준비...');
|
|
||||||
|
|
||||||
// 1. 엑셀 파일 로드
|
|
||||||
const workbook = XLSX.readFile('asset_pc (2026.06.15).xlsx');
|
|
||||||
const sheet = workbook.Sheets[workbook.SheetNames[0]];
|
|
||||||
const rawData = XLSX.utils.sheet_to_json(sheet);
|
|
||||||
|
|
||||||
// 2. system_users 데이터 맵 생성 (사번 기준 빠른 조회를 위함)
|
|
||||||
const [userRows] = await connection.query('SELECT emp_no, user_name, dept_name, position, status FROM system_users');
|
|
||||||
const userMap = new Map();
|
|
||||||
userRows.forEach(u => userMap.set(String(u.emp_no), u));
|
|
||||||
|
|
||||||
// 3. 기존 자산 중복 체크용 맵 생성 (emp_no + asset_type + category)
|
|
||||||
const [existingAssets] = await connection.query('SELECT emp_no, asset_type, category FROM asset_core');
|
|
||||||
const existingSet = new Set();
|
|
||||||
existingAssets.forEach(a => {
|
|
||||||
existingSet.add(`${a.emp_no}|${a.asset_type}|${a.category}`);
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log(`📊 처리 대상 데이터: ${rawData.length}건`);
|
|
||||||
|
|
||||||
let skipCount = 0;
|
|
||||||
let insertCount = 0;
|
|
||||||
|
|
||||||
for (let i = 0; i < rawData.length; i++) {
|
|
||||||
const row = rawData[i];
|
|
||||||
const empNo = String(row.emp_no);
|
|
||||||
const assetType = row.asset_type || '개인PC';
|
|
||||||
const category = row.category || 'PC';
|
|
||||||
|
|
||||||
// 중복 체크
|
|
||||||
if (existingSet.has(`${empNo}|${assetType}|${category}`)) {
|
|
||||||
skipCount++;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// [Step 2] 데이터 정제
|
|
||||||
// 1. 사용자 정보 매칭
|
|
||||||
const matchedUser = userMap.get(empNo);
|
|
||||||
const userName = matchedUser ? matchedUser.user_name : row.user_current;
|
|
||||||
const deptName = matchedUser ? matchedUser.dept_name : row.current_dept;
|
|
||||||
const position = matchedUser ? matchedUser.position : '';
|
|
||||||
|
|
||||||
// 2. 날짜 최적화 (purchase_date_1, purchase_date_2 중 최신값)
|
|
||||||
const d1 = parseInt(row.purchase_date_1) || 0;
|
|
||||||
const d2 = parseInt(row.purchase_date_2) || 0;
|
|
||||||
const latestDate = Math.max(d1, d2);
|
|
||||||
const purchaseDate = latestDate > 0 ? String(latestDate) : '';
|
|
||||||
|
|
||||||
// 3. 고유 ID 생성
|
|
||||||
const assetId = `PC_20260615_${String(i + 1).padStart(4, '0')}`;
|
|
||||||
const now = new Date().toISOString().replace('T', ' ').substring(0, 19);
|
|
||||||
|
|
||||||
try {
|
|
||||||
// [Step 3] DB 입력
|
|
||||||
// A. asset_core 입력
|
|
||||||
await connection.query(
|
|
||||||
`INSERT INTO asset_core (id, asset_code, category, asset_type, current_role, asset_purpose, service_type,
|
|
||||||
purchase_corp, purchase_date, memo, manager_primary, current_dept, user_current, emp_no, user_position, created_at, updated_at)
|
|
||||||
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
|
|
||||||
[assetId, assetId, category, assetType, row.current_role, row.asset_purpose, row.service_type,
|
|
||||||
'', purchaseDate, row.memo || '', '', deptName, userName, empNo, position, now, now]
|
|
||||||
);
|
|
||||||
|
|
||||||
// B. asset_spec 입력
|
|
||||||
await connection.query(
|
|
||||||
`INSERT INTO asset_spec (asset_id, model_name, mainboard, cpu, ram, gpu) VALUES (?, ?, ?, ?, ?, ?)`,
|
|
||||||
[assetId, '', row.mainboard || '', row.cpu || '', row.ram || '', row.gpu || '']
|
|
||||||
);
|
|
||||||
|
|
||||||
// C. asset_volume 입력 (SSD1, SSD2, HDD1~4)
|
|
||||||
const volumes = [
|
|
||||||
{ type: 'SSD', cap: row.SDD1, slot: 1 },
|
|
||||||
{ type: 'SSD', cap: row.SDD2, slot: 2 },
|
|
||||||
{ type: 'HDD', cap: row.HDD1, slot: 3 },
|
|
||||||
{ type: 'HDD', cap: row.HDD2, slot: 4 },
|
|
||||||
{ type: 'HDD', cap: row.HDD3, slot: 5 },
|
|
||||||
{ type: 'HDD', cap: row.HDD4, slot: 6 }
|
|
||||||
];
|
|
||||||
|
|
||||||
for (const vol of volumes) {
|
|
||||||
if (vol.cap && vol.cap !== '0' && vol.cap !== 0) {
|
|
||||||
await connection.query(
|
|
||||||
`INSERT INTO asset_volume (asset_id, disk_type, capacity, slot_no) VALUES (?, ?, ?, ?)`,
|
|
||||||
[assetId, vol.type, String(vol.cap), vol.slot]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
insertCount++;
|
|
||||||
existingSet.add(`${empNo}|${assetType}|${category}`); // 실시간 중복 방지 추가
|
|
||||||
} catch (err) {
|
|
||||||
console.error(`❌ [${empNo}] 처리 중 오류:`, err.message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`\n✨ 작업 완료!`);
|
|
||||||
console.log(`- 신규 입력: ${insertCount}건`);
|
|
||||||
console.log(`- 중복 스킵: ${skipCount}건`);
|
|
||||||
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
importAssets().catch(console.error);
|
|
||||||
@@ -1,164 +0,0 @@
|
|||||||
const XLSX = require('xlsx');
|
|
||||||
const mysql = require('mysql2/promise');
|
|
||||||
const dotenv = require('dotenv');
|
|
||||||
const path = require('path');
|
|
||||||
|
|
||||||
dotenv.config({ path: path.join(__dirname, '../.env') });
|
|
||||||
|
|
||||||
const { DB_HOST, DB_USER, DB_PASS, DB_NAME, DB_PORT } = process.env;
|
|
||||||
|
|
||||||
// 용량 정제 함수
|
|
||||||
function parseCapacity(val) {
|
|
||||||
if (!val || val === '0' || val === 0) return null;
|
|
||||||
|
|
||||||
let str = String(val).toUpperCase();
|
|
||||||
|
|
||||||
// 1. 괄호와 그 안의 내용 제거
|
|
||||||
str = str.replace(/\(.*\)/g, '').trim();
|
|
||||||
|
|
||||||
// 2. 숫자와 단위 분리
|
|
||||||
const numMatch = str.match(/[\d.]+/);
|
|
||||||
if (!numMatch) return null;
|
|
||||||
|
|
||||||
let num = parseFloat(numMatch[0]);
|
|
||||||
let unit = 'GB'; // 기본 단위
|
|
||||||
|
|
||||||
if (str.includes('TB')) {
|
|
||||||
unit = 'TB';
|
|
||||||
} else if (str.includes('GB')) {
|
|
||||||
// 4자리수 GB인 경우 TB로 전환 (지시사항 1번)
|
|
||||||
if (num >= 1000) {
|
|
||||||
num = num / 1000;
|
|
||||||
unit = 'TB';
|
|
||||||
} else {
|
|
||||||
unit = 'GB';
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// 단위가 명시되지 않은 경우 숫자의 크기로 판단
|
|
||||||
if (num >= 1000) {
|
|
||||||
num = num / 1000;
|
|
||||||
unit = 'TB';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
capacity: parseFloat(num.toFixed(2)),
|
|
||||||
unit: unit
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
async function importAssets() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: DB_HOST,
|
|
||||||
user: DB_USER,
|
|
||||||
password: DB_PASS,
|
|
||||||
database: DB_NAME,
|
|
||||||
port: parseInt(DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('🚀 [Step 1] 데이터 로드 및 사전 준비 (정제 로직 강화)...');
|
|
||||||
|
|
||||||
const workbook = XLSX.readFile('asset_pc (2026.06.15).xlsx');
|
|
||||||
const sheet = workbook.Sheets[workbook.SheetNames[0]];
|
|
||||||
const rawData = XLSX.utils.sheet_to_json(sheet);
|
|
||||||
|
|
||||||
// system_users 데이터 맵
|
|
||||||
const [userRows] = await connection.query('SELECT emp_no, user_name, dept_name, position, status FROM system_users');
|
|
||||||
const userMap = new Map();
|
|
||||||
userRows.forEach(u => userMap.set(String(u.emp_no), u));
|
|
||||||
|
|
||||||
// 기존 자산 중복 체크용 (emp_no + asset_type + category + user_current)
|
|
||||||
const [existingAssets] = await connection.query('SELECT emp_no, asset_type, category, user_current FROM asset_core');
|
|
||||||
const existingSet = new Set();
|
|
||||||
existingAssets.forEach(a => {
|
|
||||||
existingSet.add(`${a.emp_no || ''}|${a.asset_type}|${a.category}|${a.user_current}`);
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log(`📊 처리 대상 데이터: ${rawData.length}건`);
|
|
||||||
|
|
||||||
let skipCount = 0;
|
|
||||||
let insertCount = 0;
|
|
||||||
let errorCount = 0;
|
|
||||||
|
|
||||||
for (let i = 0; i < rawData.length; i++) {
|
|
||||||
const row = rawData[i];
|
|
||||||
const empNo = row.emp_no ? String(row.emp_no) : ''; // 사번 없는 행 처리 (지시사항 3번)
|
|
||||||
const assetType = row.asset_type || '개인PC';
|
|
||||||
const category = row.category || 'PC';
|
|
||||||
const userCurrent = row.user_current || '';
|
|
||||||
|
|
||||||
// 중복 체크
|
|
||||||
const dupKey = `${empNo}|${assetType}|${category}|${userCurrent}`;
|
|
||||||
if (existingSet.has(dupKey)) {
|
|
||||||
skipCount++;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// [Step 2] 데이터 정제
|
|
||||||
const matchedUser = empNo ? userMap.get(empNo) : null;
|
|
||||||
const userName = matchedUser ? matchedUser.user_name : userCurrent;
|
|
||||||
const deptName = matchedUser ? matchedUser.dept_name : (row.current_dept || '');
|
|
||||||
const position = matchedUser ? matchedUser.position : '';
|
|
||||||
|
|
||||||
const d1 = parseInt(row.purchase_date_1) || 0;
|
|
||||||
const d2 = parseInt(row.purchase_date_2) || 0;
|
|
||||||
const purchaseDate = Math.max(d1, d2) > 0 ? String(Math.max(d1, d2)) : '';
|
|
||||||
|
|
||||||
const assetId = `PC_20260615_${String(i + 1).padStart(4, '0')}`;
|
|
||||||
const now = new Date().toISOString().replace('T', ' ').substring(0, 19);
|
|
||||||
|
|
||||||
try {
|
|
||||||
// [Step 3] DB 입력
|
|
||||||
// A. asset_core
|
|
||||||
await connection.query(
|
|
||||||
`INSERT INTO asset_core (id, asset_code, category, asset_type, current_role, asset_purpose, service_type,
|
|
||||||
purchase_date, memo, current_dept, user_current, emp_no, user_position, created_at, updated_at)
|
|
||||||
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
|
|
||||||
[assetId, assetId, category, assetType, row.current_role || '', row.asset_purpose || '', row.service_type || '',
|
|
||||||
purchaseDate, row.memo || '', deptName, userName, empNo, position, now, now]
|
|
||||||
);
|
|
||||||
|
|
||||||
// B. asset_spec
|
|
||||||
await connection.query(
|
|
||||||
`INSERT INTO asset_spec (asset_id, mainboard, cpu, ram, gpu) VALUES (?, ?, ?, ?, ?)`,
|
|
||||||
[assetId, row.mainboard || '', row.cpu || '', row.ram || '', row.gpu || '']
|
|
||||||
);
|
|
||||||
|
|
||||||
// C. asset_volume
|
|
||||||
const volCols = [
|
|
||||||
{ key: 'SDD1', type: 'SSD', slot: 1 },
|
|
||||||
{ key: 'SDD2', type: 'SSD', slot: 2 },
|
|
||||||
{ key: 'HDD1', type: 'HDD', slot: 3 },
|
|
||||||
{ key: 'HDD2', type: 'HDD', slot: 4 },
|
|
||||||
{ key: 'HDD3', type: 'HDD', slot: 5 },
|
|
||||||
{ key: 'HDD4', type: 'HDD', slot: 6 }
|
|
||||||
];
|
|
||||||
|
|
||||||
for (const col of volCols) {
|
|
||||||
const rawVol = row[col.key];
|
|
||||||
const parsed = parseCapacity(rawVol);
|
|
||||||
if (parsed) {
|
|
||||||
await connection.query(
|
|
||||||
`INSERT INTO asset_volume (asset_id, disk_type, capacity, unit, slot_no) VALUES (?, ?, ?, ?, ?)`,
|
|
||||||
[assetId, col.type, parsed.capacity, parsed.unit, col.slot]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
insertCount++;
|
|
||||||
existingSet.add(dupKey);
|
|
||||||
} catch (err) {
|
|
||||||
errorCount++;
|
|
||||||
console.error(`❌ [Row ${i + 2}] ${empNo || 'Public'}: ${err.message}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`\n✨ 작업 완료!`);
|
|
||||||
console.log(`- 신규 입력: ${insertCount}건`);
|
|
||||||
console.log(`- 중복 스킵: ${skipCount}건`);
|
|
||||||
console.log(`- 오류 실패: ${errorCount}건`);
|
|
||||||
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
importAssets().catch(console.error);
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
const XLSX = require('xlsx');
|
|
||||||
const mysql = require('mysql2/promise');
|
|
||||||
const dotenv = require('dotenv');
|
|
||||||
const path = require('path');
|
|
||||||
|
|
||||||
dotenv.config({ path: path.join(__dirname, '../.env') });
|
|
||||||
|
|
||||||
const { DB_HOST, DB_USER, DB_PASS, DB_NAME, DB_PORT } = process.env;
|
|
||||||
|
|
||||||
async function importUsers() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: DB_HOST,
|
|
||||||
user: DB_USER,
|
|
||||||
password: DB_PASS,
|
|
||||||
database: DB_NAME,
|
|
||||||
port: parseInt(DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('🚀 Excel 데이터 로드 중...');
|
|
||||||
const workbook = XLSX.readFile('system_User (20260615).xlsx');
|
|
||||||
const sheetName = workbook.SheetNames[0];
|
|
||||||
const sheet = workbook.Sheets[sheetName];
|
|
||||||
const data = XLSX.utils.sheet_to_json(sheet);
|
|
||||||
|
|
||||||
console.log(`📊 총 ${data.length}개의 데이터를 찾았습니다.`);
|
|
||||||
|
|
||||||
// 기존 데이터 삭제 여부 (사용자 요구사항에 따라 결정 가능하지만, 보통 초기화 후 재입입)
|
|
||||||
// 여기서는 중복 방지를 위해 기존 데이터를 삭제하고 새로 넣는 방식을 취하겠습니다.
|
|
||||||
console.log('🧹 기존 system_users 데이터 삭제 중...');
|
|
||||||
await connection.query('DELETE FROM system_users');
|
|
||||||
|
|
||||||
console.log('📥 데이터 삽입 중...');
|
|
||||||
let successCount = 0;
|
|
||||||
|
|
||||||
for (let i = 0; i < data.length; i++) {
|
|
||||||
const row = data[i];
|
|
||||||
const { emp_no, user_name, dept_name, position, status } = row;
|
|
||||||
|
|
||||||
// ID 생성 (USR_ + 인덱스 001 형식)
|
|
||||||
const id = `USR_${String(i + 1).padStart(3, '0')}`;
|
|
||||||
const createdAt = new Date().toISOString().replace('T', ' ').substring(0, 19);
|
|
||||||
|
|
||||||
try {
|
|
||||||
await connection.query(
|
|
||||||
'INSERT INTO system_users (id, emp_no, user_name, dept_name, position, status, created_at) VALUES (?, ?, ?, ?, ?, ?, ?)',
|
|
||||||
[id, String(emp_no), user_name, dept_name, position, status, createdAt]
|
|
||||||
);
|
|
||||||
successCount++;
|
|
||||||
} catch (err) {
|
|
||||||
console.error(`❌ 삽입 실패 (Row ${i + 2}):`, err.message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`✅ 완료: ${successCount}개의 사용자가 성공적으로 등록되었습니다.`);
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
importUsers().catch(err => {
|
|
||||||
console.error('❌ 작업 중 오류 발생:', err);
|
|
||||||
process.exit(1);
|
|
||||||
});
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
const XLSX = require('xlsx');
|
|
||||||
const workbook = XLSX.readFile('asset_pc (2026.06.15).xlsx');
|
|
||||||
const sheetName = workbook.SheetNames[0];
|
|
||||||
const sheet = workbook.Sheets[sheetName];
|
|
||||||
const data = XLSX.utils.sheet_to_json(sheet, { header: 1 });
|
|
||||||
console.log('Headers:', JSON.stringify(data[0], null, 2));
|
|
||||||
console.log('Sample Row 1:', JSON.stringify(data[1], null, 2));
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
const XLSX = require('xlsx');
|
|
||||||
const workbook = XLSX.readFile('system_User (20260615).xlsx');
|
|
||||||
const sheetName = workbook.SheetNames[0];
|
|
||||||
const sheet = workbook.Sheets[sheetName];
|
|
||||||
const data = XLSX.utils.sheet_to_json(sheet, { header: 1 });
|
|
||||||
console.log(JSON.stringify(data.slice(0, 5), null, 2));
|
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config({ path: './.env' });
|
|
||||||
|
|
||||||
async function main() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
try {
|
|
||||||
const [schema] = await connection.query(`SHOW CREATE TABLE asset_volume`);
|
|
||||||
console.log('Schema:', schema[0]['Create Table']);
|
|
||||||
const [rows] = await connection.query(`SELECT * FROM asset_volume LIMIT 20`);
|
|
||||||
console.log('Sample Data:', JSON.stringify(rows, null, 2));
|
|
||||||
} catch (err) {
|
|
||||||
console.error(err);
|
|
||||||
} finally {
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
main();
|
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config({ path: './.env' });
|
|
||||||
|
|
||||||
async function main() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
try {
|
|
||||||
const [rows] = await connection.query(
|
|
||||||
`SELECT * FROM asset_core WHERE asset_purpose LIKE '%바론%' OR asset_purpose LIKE '%SSO%'`
|
|
||||||
);
|
|
||||||
console.log('Results:', JSON.stringify(rows, null, 2));
|
|
||||||
} catch (err) {
|
|
||||||
console.error(err);
|
|
||||||
} finally {
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
main();
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
require('dotenv').config({ path: './.env' });
|
|
||||||
|
|
||||||
async function main() {
|
|
||||||
const url = `http://${process.env.DB_HOST || 'localhost'}:3000/api/assets/master`;
|
|
||||||
try {
|
|
||||||
const res = await fetch(url);
|
|
||||||
const data = await res.json();
|
|
||||||
// find asset with id "9pvkqyi"
|
|
||||||
const serverList = data.server || [];
|
|
||||||
const target = serverList.find(s => s.id === '9pvkqyi');
|
|
||||||
console.log('Server list asset:', JSON.stringify(target, null, 2));
|
|
||||||
} catch (err) {
|
|
||||||
console.error(err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
main();
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config();
|
|
||||||
|
|
||||||
async function rawCheck() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
const [rows] = await connection.query('SELECT user_current, emp_no FROM asset_core WHERE id LIKE "PC_20260615_%" LIMIT 10');
|
|
||||||
console.log(rows);
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
rawCheck().catch(console.error);
|
|
||||||
@@ -1,85 +0,0 @@
|
|||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config();
|
|
||||||
|
|
||||||
async function rebuildAssetCodes() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('🚀 [Step 1] 신규 자산 구매일 업데이트 (YYYY-12-01)...');
|
|
||||||
|
|
||||||
// 1. 오늘 입력한 자산들 조회
|
|
||||||
const [rows] = await connection.query(
|
|
||||||
'SELECT id, purchase_date FROM asset_core WHERE id LIKE "PC_20260615_%"'
|
|
||||||
);
|
|
||||||
console.log(`대상 자산: ${rows.length}건`);
|
|
||||||
|
|
||||||
// 2. 구매일자 업데이트 (연도만 있는 경우 -12-01 추가)
|
|
||||||
for (const row of rows) {
|
|
||||||
if (row.purchase_date && row.purchase_date.length === 4) {
|
|
||||||
const newDate = `${row.purchase_date}-12-01`;
|
|
||||||
await connection.query(
|
|
||||||
'UPDATE asset_core SET purchase_date = ? WHERE id = ?',
|
|
||||||
[newDate, row.id]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
console.log('✅ 구매일 업데이트 완료.');
|
|
||||||
|
|
||||||
console.log('\n🚀 [Step 2] 자산번호(asset_code) 재매핑 시작...');
|
|
||||||
|
|
||||||
// 3. 연도별로 그룹화하여 자산번호 부여
|
|
||||||
// 연도 목록 추출
|
|
||||||
const [yearRows] = await connection.query(
|
|
||||||
'SELECT DISTINCT LEFT(purchase_date, 4) as year FROM asset_core WHERE id LIKE "PC_20260615_%" ORDER BY year'
|
|
||||||
);
|
|
||||||
|
|
||||||
for (const yRow of yearRows) {
|
|
||||||
const year = yRow.year;
|
|
||||||
const yearMonth = `${year}12`;
|
|
||||||
const pattern = `PC-${yearMonth}-%`;
|
|
||||||
|
|
||||||
console.log(`--- [${year}년] 처리 중 ---`);
|
|
||||||
|
|
||||||
// 해당 연도/월의 기존 최대 순번 조회
|
|
||||||
const [maxRows] = await connection.query(
|
|
||||||
'SELECT asset_code FROM asset_core WHERE asset_code LIKE ? AND id NOT LIKE "PC_20260615_%"',
|
|
||||||
[pattern]
|
|
||||||
);
|
|
||||||
|
|
||||||
let maxSeq = 0;
|
|
||||||
maxRows.forEach(r => {
|
|
||||||
const parts = r.asset_code.split('-');
|
|
||||||
const seq = parseInt(parts[2]);
|
|
||||||
if (seq > maxSeq) maxSeq = seq;
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log(`기존 최대 순번: ${maxSeq}`);
|
|
||||||
|
|
||||||
// 해당 연도 자산들 순차적으로 번호 부여
|
|
||||||
const [assetsOfYear] = await connection.query(
|
|
||||||
'SELECT id FROM asset_core WHERE id LIKE "PC_20260615_%" AND purchase_date LIKE ? ORDER BY id',
|
|
||||||
[`${year}-12%`]
|
|
||||||
);
|
|
||||||
|
|
||||||
let currentSeq = maxSeq + 1;
|
|
||||||
for (const asset of assetsOfYear) {
|
|
||||||
const newCode = `PC-${yearMonth}-${String(currentSeq).padStart(4, '0')}`;
|
|
||||||
await connection.query(
|
|
||||||
'UPDATE asset_core SET asset_code = ? WHERE id = ?',
|
|
||||||
[newCode, asset.id]
|
|
||||||
);
|
|
||||||
currentSeq++;
|
|
||||||
}
|
|
||||||
console.log(`신규 부여 완료: ${assetsOfYear.length}건 (순번 ${maxSeq + 1} ~ ${currentSeq - 1})`);
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\n✨ 모든 작업이 완료되었습니다.');
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
rebuildAssetCodes().catch(console.error);
|
|
||||||
@@ -1,85 +0,0 @@
|
|||||||
const XLSX = require('xlsx');
|
|
||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config();
|
|
||||||
|
|
||||||
async function reexamineData() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('🧐 [전수 조사] 엑셀 vs DB 데이터 비교 분석...');
|
|
||||||
|
|
||||||
// 1. 엑셀 데이터 로드
|
|
||||||
const workbook = XLSX.readFile('asset_pc (2026.06.15).xlsx');
|
|
||||||
const sheet = workbook.Sheets[workbook.SheetNames[0]];
|
|
||||||
const excelRows = XLSX.utils.sheet_to_json(sheet);
|
|
||||||
|
|
||||||
// 2. DB 데이터 로드
|
|
||||||
const [dbRows] = await connection.query(`
|
|
||||||
SELECT id, asset_code, asset_type, user_current, emp_no, current_dept
|
|
||||||
FROM asset_core
|
|
||||||
WHERE id LIKE "PC_20260615_%"
|
|
||||||
`);
|
|
||||||
const dbMap = new Map();
|
|
||||||
dbRows.forEach(r => dbMap.set(r.id, r));
|
|
||||||
|
|
||||||
const report = {
|
|
||||||
total: excelRows.length,
|
|
||||||
publicInExcelWithEmpNo: [], // 엑셀은 공용PC인데 사번이 있는 경우
|
|
||||||
personalInExcelNoEmpNo: [], // 엑셀은 개인PC인데 사번이 없는 경우
|
|
||||||
typeMismatch: [], // 엑셀과 DB의 asset_type이 다른 경우
|
|
||||||
userMismatch: [] // 사용자명이 크게 다른 경우
|
|
||||||
};
|
|
||||||
|
|
||||||
for (let i = 0; i < excelRows.length; i++) {
|
|
||||||
const ex = excelRows[i];
|
|
||||||
const id = `PC_20260615_${String(i + 1).padStart(4, '0')}`;
|
|
||||||
const db = dbMap.get(id);
|
|
||||||
|
|
||||||
if (!db) continue;
|
|
||||||
|
|
||||||
const exType = ex.asset_type || '개인PC';
|
|
||||||
const exEmpNo = ex.emp_no ? String(ex.emp_no) : null;
|
|
||||||
const exUser = ex.user_current || '';
|
|
||||||
|
|
||||||
// A. 공용PC인데 사번이 있는 경우 (가장 큰 혼란 포인트)
|
|
||||||
if (exType === '공용PC' && exEmpNo) {
|
|
||||||
report.publicInExcelWithEmpNo.push({ id, exUser, exEmpNo, exDept: ex.current_dept });
|
|
||||||
}
|
|
||||||
|
|
||||||
// B. 개인PC인데 사번이 없는 경우
|
|
||||||
if (exType === '개인PC' && !exEmpNo) {
|
|
||||||
report.personalInExcelNoEmpNo.push({ id, exUser, exDept: ex.current_dept });
|
|
||||||
}
|
|
||||||
|
|
||||||
// C. DB와의 타입 불일치 (현재 DB 상태 체크)
|
|
||||||
if (db.asset_type !== exType) {
|
|
||||||
report.typeMismatch.push({ id, exType, dbType: db.asset_type, user: db.user_current });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\n================================================');
|
|
||||||
console.log(`📊 전수 조사 요약 (총 ${report.total}건)`);
|
|
||||||
console.log(`1. 엑셀은 '공용PC'이나 '사번'이 있는 항목: ${report.publicInExcelWithEmpNo.length}건`);
|
|
||||||
console.log(`2. 엑셀은 '개인PC'이나 '사번'이 없는 항목: ${report.personalInExcelNoEmpNo.length}건`);
|
|
||||||
console.log(`3. 현재 DB와 엑셀의 '자산유형' 불일치: ${report.typeMismatch.length}건`);
|
|
||||||
console.log('================================================\n');
|
|
||||||
|
|
||||||
if (report.publicInExcelWithEmpNo.length > 0) {
|
|
||||||
console.log('⚠️ [그룹 1] 공용PC인데 실사용자/관리자가 지정된 사례 (샘플 15건):');
|
|
||||||
console.table(report.publicInExcelWithEmpNo.slice(0, 15));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (report.personalInExcelNoEmpNo.length > 0) {
|
|
||||||
console.log('\n⚠️ [그룹 2] 개인PC인데 사번 정보가 누락된 사례 (샘플 15건):');
|
|
||||||
console.table(report.personalInExcelNoEmpNo.slice(0, 15));
|
|
||||||
}
|
|
||||||
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
reexamineData().catch(console.error);
|
|
||||||
@@ -1,92 +0,0 @@
|
|||||||
const XLSX = require('xlsx');
|
|
||||||
const mysql = require('mysql2/promise');
|
|
||||||
const dotenv = require('dotenv');
|
|
||||||
const path = require('path');
|
|
||||||
|
|
||||||
dotenv.config({ path: path.join(__dirname, '../.env') });
|
|
||||||
|
|
||||||
const { DB_HOST, DB_USER, DB_PASS, DB_NAME, DB_PORT } = process.env;
|
|
||||||
|
|
||||||
async function restoreAndMerge() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: DB_HOST,
|
|
||||||
user: DB_USER,
|
|
||||||
password: DB_PASS,
|
|
||||||
database: DB_NAME,
|
|
||||||
port: parseInt(DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('🔄 데이터 복구 및 병합 시작...');
|
|
||||||
|
|
||||||
// 1. 백업 파일에서 기존 데이터(212건) 로드
|
|
||||||
const workbookBackup = XLSX.readFile('backupDB_20260602.xlsx');
|
|
||||||
const oldUsers = XLSX.utils.sheet_to_json(workbookBackup.Sheets['system_users']);
|
|
||||||
|
|
||||||
// 2. 신규 파일에서 데이터(987건) 로드
|
|
||||||
const workbookNew = XLSX.readFile('system_User (20260615).xlsx');
|
|
||||||
const newUsers = XLSX.utils.sheet_to_json(workbookNew.Sheets[workbookNew.SheetNames[0]]);
|
|
||||||
|
|
||||||
console.log(`기본 백업 데이터: ${oldUsers.length}건`);
|
|
||||||
console.log(`신규 추가 데이터: ${newUsers.length}건`);
|
|
||||||
|
|
||||||
// 테이블 비우기 (실수를 바로잡기 위해 다시 시작)
|
|
||||||
await connection.query('DELETE FROM system_users');
|
|
||||||
|
|
||||||
const insertedEmpNos = new Set();
|
|
||||||
let restoreCount = 0;
|
|
||||||
let addCount = 0;
|
|
||||||
|
|
||||||
// 3. 기존 데이터 복구 (ID 보존 시도)
|
|
||||||
for (const user of oldUsers) {
|
|
||||||
const { id, emp_no, user_name, dept_name, position, status, created_at } = user;
|
|
||||||
|
|
||||||
// 엑셀 날짜 처리 (숫자로 되어 있을 경우)
|
|
||||||
let finalCreatedAt = created_at;
|
|
||||||
if (typeof created_at === 'number') {
|
|
||||||
const date = new Date((created_at - 25569) * 86400 * 1000);
|
|
||||||
finalCreatedAt = date.toISOString().replace('T', ' ').substring(0, 19);
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
await connection.query(
|
|
||||||
'INSERT INTO system_users (id, emp_no, user_name, dept_name, position, status, created_at) VALUES (?, ?, ?, ?, ?, ?, ?)',
|
|
||||||
[id, String(emp_no), user_name, dept_name, position, status, finalCreatedAt]
|
|
||||||
);
|
|
||||||
insertedEmpNos.add(String(emp_no));
|
|
||||||
restoreCount++;
|
|
||||||
} catch (err) {
|
|
||||||
console.error(`❌ 복구 실패 (emp_no: ${emp_no}):`, err.message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 4. 신규 데이터 추가 (중복 제외)
|
|
||||||
for (let i = 0; i < newUsers.length; i++) {
|
|
||||||
const user = newUsers[i];
|
|
||||||
const { emp_no, user_name, dept_name, position, status } = user;
|
|
||||||
const strEmpNo = String(emp_no);
|
|
||||||
|
|
||||||
if (insertedEmpNos.has(strEmpNo)) {
|
|
||||||
continue; // 이미 복구된 데이터는 스킵
|
|
||||||
}
|
|
||||||
|
|
||||||
// 신규 데이터용 ID 생성 (기존 ID와 겹치지 않게 'NEW_' 접두어 또는 시퀀스 사용)
|
|
||||||
// 여기서는 단순히 시퀀스로 처리 (최대 ID 확인 후 +1 하는 방식이 좋으나 여기선 간단히)
|
|
||||||
const id = `USR_N_${String(i + 1).padStart(4, '0')}`;
|
|
||||||
const createdAt = new Date().toISOString().replace('T', ' ').substring(0, 19);
|
|
||||||
|
|
||||||
try {
|
|
||||||
await connection.query(
|
|
||||||
'INSERT INTO system_users (id, emp_no, user_name, dept_name, position, status, created_at) VALUES (?, ?, ?, ?, ?, ?, ?)',
|
|
||||||
[id, strEmpNo, user_name, dept_name, position, status, createdAt]
|
|
||||||
);
|
|
||||||
addCount++;
|
|
||||||
} catch (err) {
|
|
||||||
console.error(`❌ 추가 실패 (emp_no: ${emp_no}):`, err.message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`✅ 복구 완료: 기존 ${restoreCount}건 복구, 신규 ${addCount}건 추가 (총 ${restoreCount + addCount}건)`);
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
restoreAndMerge().catch(console.error);
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
const mysql = require('mysql2/promise');
|
|
||||||
require('dotenv').config();
|
|
||||||
|
|
||||||
async function updateDepartments() {
|
|
||||||
const connection = await mysql.createConnection({
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
user: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASS,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
port: parseInt(process.env.DB_PORT || '3306')
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log("🚀 부서명 '삼안' 통합 업데이트 시작...");
|
|
||||||
|
|
||||||
const [result] = await connection.query(`
|
|
||||||
UPDATE asset_core
|
|
||||||
SET current_dept = '삼안'
|
|
||||||
WHERE current_dept NOT IN ('총괄기획실', '기술개발센터', '현타', '장헌', '한맥', 'PTC', '', '삼안')
|
|
||||||
AND current_dept IS NOT NULL
|
|
||||||
`);
|
|
||||||
|
|
||||||
console.log(`✅ 업데이트 완료: ${result.affectedRows}건의 부서명이 '삼안'으로 변경되었습니다.`);
|
|
||||||
|
|
||||||
// 최종 확인용 카운트
|
|
||||||
const [rows] = await connection.query('SELECT current_dept, COUNT(*) as count FROM asset_core GROUP BY current_dept');
|
|
||||||
console.log('\n📊 최종 부서 분포:');
|
|
||||||
console.table(rows);
|
|
||||||
|
|
||||||
await connection.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
updateDepartments().catch(console.error);
|
|
||||||
482
server.js
482
server.js
@@ -113,6 +113,30 @@ const ASSET_TABLES = [
|
|||||||
'asset_core'
|
'asset_core'
|
||||||
];
|
];
|
||||||
|
|
||||||
|
// --- Helper Functions for Maps ---
|
||||||
|
function getCleanMapKey(path) {
|
||||||
|
let clean = path.replace('img/location_photo/', '').replace('.png', '');
|
||||||
|
clean = clean.replace('서관', 'W').replace('동관', 'E');
|
||||||
|
clean = clean.replace('한맥빌딩/MDF실/MDF_', 'HAN-MDF-');
|
||||||
|
clean = clean.replace('기술개발센터/서버실/서버실_', 'DEV-SVR-');
|
||||||
|
clean = clean.replace(/\//g, '-');
|
||||||
|
return clean;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getLocationName(path) {
|
||||||
|
if (path.includes('IDC')) return 'IDC';
|
||||||
|
if (path.includes('한맥빌딩')) return '한맥빌딩';
|
||||||
|
if (path.includes('기술개발센터')) return '기술개발센터';
|
||||||
|
return '기타';
|
||||||
|
}
|
||||||
|
|
||||||
|
function getLocationDetail(path, idx) {
|
||||||
|
let clean = path.replace('img/location_photo/', '').replace('.png', '');
|
||||||
|
let parts = clean.split('/');
|
||||||
|
let lastPart = parts[parts.length - 1];
|
||||||
|
return `${lastPart} 구역 자리 #${idx + 1}`;
|
||||||
|
}
|
||||||
|
|
||||||
// --- API Endpoints ---
|
// --- API Endpoints ---
|
||||||
|
|
||||||
// 1. Generic Batch Save (Dynamic Table Detection)
|
// 1. Generic Batch Save (Dynamic Table Detection)
|
||||||
@@ -174,6 +198,9 @@ app.get('/api/assets/master', async (req, res) => {
|
|||||||
s.hw_status, s.model_name, s.mainboard, s.os, s.cpu, s.ram, s.gpu,
|
s.hw_status, s.model_name, s.mainboard, s.os, s.cpu, s.ram, s.gpu,
|
||||||
s.monitoring, s.price, s.monitor_inch, s.serial_num,
|
s.monitoring, s.price, s.monitor_inch, s.serial_num,
|
||||||
l.location, l.location_detail, l.location_photo, l.loc_x, l.loc_y,
|
l.location, l.location_detail, l.location_photo, l.loc_x, l.loc_y,
|
||||||
|
(
|
||||||
|
SELECT EXISTS(SELECT 1 FROM asset_audit_pending WHERE asset_code = c.asset_code AND status = 'APPROVED')
|
||||||
|
) AS is_audit_approved,
|
||||||
(
|
(
|
||||||
SELECT JSON_ARRAYAGG(JSON_OBJECT('type', net_type, 'name', net_name, 'val1', net_value1, 'val2', net_value2))
|
SELECT JSON_ARRAYAGG(JSON_OBJECT('type', net_type, 'name', net_name, 'val1', net_value1, 'val2', net_value2))
|
||||||
FROM asset_remote WHERE asset_id = c.id AND is_active = 1
|
FROM asset_remote WHERE asset_id = c.id AND is_active = 1
|
||||||
@@ -193,7 +220,7 @@ app.get('/api/assets/master', async (req, res) => {
|
|||||||
|
|
||||||
const catMap = {
|
const catMap = {
|
||||||
'PC': 'pc', '서버': 'server', '저장매체': 'storage', '네트워크': 'network',
|
'PC': 'pc', '서버': 'server', '저장매체': 'storage', '네트워크': 'network',
|
||||||
'업무지원장비': 'equipment', '사무가구': 'officeSupplies', '공간정보장비': 'survey',
|
'업무지원장비': 'equipment', '시설자산': 'officeSupplies', '공간정보장비': 'survey',
|
||||||
'내빈/외빈': 'vip', 'PC부품': 'pcParts'
|
'내빈/외빈': 'vip', 'PC부품': 'pcParts'
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -234,9 +261,34 @@ app.post('/api/asset/:category/save', async (req, res) => {
|
|||||||
connection = await pool.getConnection();
|
connection = await pool.getConnection();
|
||||||
await connection.beginTransaction();
|
await connection.beginTransaction();
|
||||||
|
|
||||||
|
// 3.0.0 CPU, GPU, RAM 부품 마스터 유효성 검사
|
||||||
|
const partsToCheck = [
|
||||||
|
{ value: asset.cpu, category: 'CPU', label: 'CPU' },
|
||||||
|
{ value: asset.gpu, category: 'GPU', label: 'GPU' },
|
||||||
|
{ value: asset.ram, category: 'RAM', label: 'RAM' }
|
||||||
|
];
|
||||||
|
|
||||||
|
for (const part of partsToCheck) {
|
||||||
|
const val = String(part.value || '').trim();
|
||||||
|
if (val) {
|
||||||
|
const [rows] = await connection.query(
|
||||||
|
'SELECT id FROM hardware_components_master WHERE UPPER(category) = ? AND LOWER(TRIM(component_name)) = ?',
|
||||||
|
[part.category, val.toLowerCase()]
|
||||||
|
);
|
||||||
|
if (rows.length === 0) {
|
||||||
|
await connection.rollback();
|
||||||
|
return res.status(400).json({
|
||||||
|
success: false,
|
||||||
|
message: `입력하신 ${part.label} "${val}"은(는) 부품 마스터에 존재하지 않는 규격입니다.`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// 3.0 History Tracking & Auto Field Update
|
// 3.0 History Tracking & Auto Field Update
|
||||||
const [oldCoreRows] = await connection.query('SELECT * FROM asset_core WHERE id = ?', [asset.id]);
|
const [oldCoreRows] = await connection.query('SELECT * FROM asset_core WHERE id = ?', [asset.id]);
|
||||||
const [oldSpecRows] = await connection.query('SELECT * FROM asset_spec WHERE asset_id = ?', [asset.id]);
|
const [oldSpecRows] = await connection.query('SELECT * FROM asset_spec WHERE asset_id = ?', [asset.id]);
|
||||||
|
const [oldRemoteRows] = await connection.query('SELECT * FROM asset_remote WHERE asset_id = ? AND is_active = 1', [asset.id]);
|
||||||
const oldCore = oldCoreRows[0] || {};
|
const oldCore = oldCoreRows[0] || {};
|
||||||
const oldSpec = oldSpecRows[0] || {};
|
const oldSpec = oldSpecRows[0] || {};
|
||||||
|
|
||||||
@@ -299,6 +351,85 @@ app.post('/api/asset/:category/save', async (req, res) => {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 3.0.4 메모, 용도, 유형 변동 감지
|
||||||
|
const oldMemo = String(oldCore.memo || '').trim();
|
||||||
|
const newMemo = String(asset.memo || '').trim();
|
||||||
|
if (newMemo !== '' && oldMemo !== newMemo) {
|
||||||
|
historyLogs.push({
|
||||||
|
event_type: 'MEMO_CHANGE',
|
||||||
|
details: `[메모 변경] ${oldMemo || '(없음)'} -> ${newMemo}`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const oldPurpose = String(oldCore.asset_purpose || '').trim();
|
||||||
|
const newPurpose = String(asset.asset_purpose || '').trim();
|
||||||
|
if (newPurpose !== '' && oldPurpose !== newPurpose) {
|
||||||
|
historyLogs.push({
|
||||||
|
event_type: 'PURPOSE_CHANGE',
|
||||||
|
details: `[용도 변경] ${oldPurpose || '(없음)'} -> ${newPurpose}`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const oldType = String(oldCore.asset_type || '').trim();
|
||||||
|
const newType = String(asset.asset_type || '').trim();
|
||||||
|
if (newType !== '' && oldType !== newType) {
|
||||||
|
historyLogs.push({
|
||||||
|
event_type: 'TYPE_CHANGE',
|
||||||
|
details: `[유형 변경] ${oldType || '(없음)'} -> ${newType}`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3.0.5 접속정보 변동 감지
|
||||||
|
const formatRemote = (r) => {
|
||||||
|
const type = r.net_type || r.type || '';
|
||||||
|
const name = r.net_name || r.name || '';
|
||||||
|
const val1 = r.net_value1 || r.val1 || '';
|
||||||
|
const val2 = r.net_value2 || r.val2 || '';
|
||||||
|
return `${type}:${name}:${val1}:${val2}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const oldRemotesSummary = oldRemoteRows.map(formatRemote).sort().join(' | ');
|
||||||
|
|
||||||
|
let newNets = [];
|
||||||
|
if (asset.remotes) {
|
||||||
|
try {
|
||||||
|
newNets = typeof asset.remotes === 'string' ? JSON.parse(asset.remotes) : asset.remotes;
|
||||||
|
} catch(e) { newNets = []; }
|
||||||
|
} else if (asset.ip_address || asset.mac_address || asset.remote_tool) {
|
||||||
|
if (asset.ip_address || asset.mac_address) {
|
||||||
|
newNets.push({ type: 'IP', name: '기본망', val1: asset.ip_address, val2: asset.mac_address });
|
||||||
|
}
|
||||||
|
if (asset.remote_tool || asset.remote_id || asset.remote_pw) {
|
||||||
|
newNets.push({ type: 'REMOTE', name: asset.remote_tool, val1: asset.remote_id, val2: asset.remote_pw });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const newRemotesSummary = newNets.map(formatRemote).sort().join(' | ');
|
||||||
|
|
||||||
|
if (newRemotesSummary !== '' && oldRemotesSummary !== newRemotesSummary) {
|
||||||
|
const formatDisplay = (summary) => {
|
||||||
|
if (!summary) return '(없음)';
|
||||||
|
return summary.split(' | ').map(item => {
|
||||||
|
const [type, name, val1, val2] = item.split(':');
|
||||||
|
if (type === 'IP') {
|
||||||
|
return `[IP] ${name}: ${val1} (MAC: ${val2 || '없음'})`;
|
||||||
|
} else {
|
||||||
|
let id = '', pw = '';
|
||||||
|
try {
|
||||||
|
const parsed = JSON.parse(val2);
|
||||||
|
id = parsed.id || '';
|
||||||
|
pw = parsed.pw || '';
|
||||||
|
} catch(e) { id = val1; pw = val2; }
|
||||||
|
return `[원격] ${name}: ID=${id || '없음'}, PW=${pw ? '***' : '없음'}`;
|
||||||
|
}
|
||||||
|
}).join(', ');
|
||||||
|
};
|
||||||
|
|
||||||
|
historyLogs.push({
|
||||||
|
event_type: 'REMOTE_CHANGE',
|
||||||
|
details: `[접속정보 변경] ${formatDisplay(oldRemotesSummary)} -> ${formatDisplay(newRemotesSummary)}`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
// 로그 일괄 삽입
|
// 로그 일괄 삽입
|
||||||
for (const log of historyLogs) {
|
for (const log of historyLogs) {
|
||||||
await connection.query(
|
await connection.query(
|
||||||
@@ -546,13 +677,121 @@ app.get('/api/generate-asset-code', async (req, res) => {
|
|||||||
} catch (err) { handleError(res, err, 'GENERATE CODE'); }
|
} catch (err) { handleError(res, err, 'GENERATE CODE'); }
|
||||||
});
|
});
|
||||||
|
|
||||||
// 6. Map Config API
|
// 6. Map Config API (Adopt database-driven locations from origin/QR_setting)
|
||||||
app.get('/api/maps', (req, res) => {
|
app.get('/api/maps', async (req, res) => {
|
||||||
try {
|
try {
|
||||||
if (!fs.existsSync('map_config.json')) return res.json({});
|
const query = `
|
||||||
const data = fs.readFileSync('map_config.json', 'utf8');
|
SELECT
|
||||||
res.json(JSON.parse(data || '{}'));
|
pl.location_code,
|
||||||
} catch (err) { handleError(res, err, 'GET MAPS'); }
|
pl.location_name,
|
||||||
|
pl.location_detail,
|
||||||
|
pl.map_image,
|
||||||
|
pl.map_x,
|
||||||
|
pl.map_y,
|
||||||
|
pl.map_w,
|
||||||
|
pl.map_h,
|
||||||
|
al.asset_id
|
||||||
|
FROM physical_locations pl
|
||||||
|
LEFT JOIN asset_location al ON al.physical_location_code = pl.location_code AND al.is_active = 1
|
||||||
|
`;
|
||||||
|
const [rows] = await pool.query(query);
|
||||||
|
|
||||||
|
const mapConfig = {};
|
||||||
|
rows.forEach(row => {
|
||||||
|
const mapPath = row.map_image;
|
||||||
|
if (!mapConfig[mapPath]) {
|
||||||
|
mapConfig[mapPath] = [];
|
||||||
|
}
|
||||||
|
mapConfig[mapPath].push({
|
||||||
|
x: parseFloat(row.map_x).toFixed(2),
|
||||||
|
y: parseFloat(row.map_y).toFixed(2),
|
||||||
|
w: parseFloat(row.map_w).toFixed(2),
|
||||||
|
h: parseFloat(row.map_h).toFixed(2),
|
||||||
|
asset_id: row.asset_id
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
res.json(mapConfig);
|
||||||
|
} catch (err) {
|
||||||
|
handleError(res, err, 'GET MAPS');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
app.post('/api/maps/save', async (req, res) => {
|
||||||
|
let connection;
|
||||||
|
try {
|
||||||
|
const { path, boxes } = req.body;
|
||||||
|
if (!path) return res.status(400).json({ error: 'Path is required' });
|
||||||
|
if (!Array.isArray(boxes)) return res.status(400).json({ error: 'Boxes must be an array' });
|
||||||
|
|
||||||
|
connection = await pool.getConnection();
|
||||||
|
await connection.beginTransaction();
|
||||||
|
|
||||||
|
const cleanKey = getCleanMapKey(path);
|
||||||
|
const locName = getLocationName(path);
|
||||||
|
|
||||||
|
// 1. Get old location codes for this map
|
||||||
|
const [oldLocs] = await connection.query(
|
||||||
|
'SELECT location_code FROM physical_locations WHERE map_image = ?',
|
||||||
|
[path]
|
||||||
|
);
|
||||||
|
const oldLocCodes = oldLocs.map(r => r.location_code);
|
||||||
|
|
||||||
|
// 2. Deactivate and clear foreign key references in asset_location to these old location codes
|
||||||
|
if (oldLocCodes.length > 0) {
|
||||||
|
await connection.query(
|
||||||
|
'UPDATE asset_location SET is_active = 0, deactivated_at = NOW(), physical_location_code = NULL WHERE physical_location_code IN (?)',
|
||||||
|
[oldLocCodes]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. Delete old physical locations for this map
|
||||||
|
await connection.query(
|
||||||
|
'DELETE FROM physical_locations WHERE map_image = ?',
|
||||||
|
[path]
|
||||||
|
);
|
||||||
|
|
||||||
|
// 4. Insert new physical locations and setup asset_location mappings
|
||||||
|
for (let i = 0; i < boxes.length; i++) {
|
||||||
|
const box = boxes[i];
|
||||||
|
const padIdx = String(i + 1).padStart(3, '0');
|
||||||
|
const locCode = `LOC-${cleanKey}-${padIdx}`;
|
||||||
|
const locDetail = getLocationDetail(path, i);
|
||||||
|
|
||||||
|
// Insert physical location
|
||||||
|
await connection.query(`
|
||||||
|
INSERT INTO physical_locations
|
||||||
|
(location_code, location_name, location_detail, map_image, map_x, map_y, map_w, map_h)
|
||||||
|
VALUES (?, ?, ?, ?, ?, ?, ?, ?)
|
||||||
|
`, [locCode, locName, locDetail, path, box.x, box.y, box.w, box.h]);
|
||||||
|
|
||||||
|
// If asset_id is mapped, update asset_location
|
||||||
|
if (box.asset_id) {
|
||||||
|
// Deactivate old active locations for this asset
|
||||||
|
await connection.query(
|
||||||
|
'UPDATE asset_location SET is_active = 0, deactivated_at = NOW() WHERE asset_id = ? AND is_active = 1',
|
||||||
|
[box.asset_id]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Insert new active location mapping
|
||||||
|
const pathPartsForMap = path.split('/');
|
||||||
|
const stdDetailForMap = pathPartsForMap[pathPartsForMap.length - 2] || locDetail;
|
||||||
|
await connection.query(`
|
||||||
|
INSERT INTO asset_location
|
||||||
|
(asset_id, location, location_detail, location_photo, loc_x, loc_y, physical_location_code, is_active)
|
||||||
|
VALUES (?, ?, ?, ?, ?, ?, ?, 1)
|
||||||
|
`, [box.asset_id, locName, stdDetailForMap, path, box.x, box.y, locCode]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
await connection.commit();
|
||||||
|
res.json({ success: true, message: 'Map and Database synced successfully' });
|
||||||
|
} catch (err) {
|
||||||
|
if (connection) await connection.rollback();
|
||||||
|
handleError(res, err, 'SAVE MAPS SYNC');
|
||||||
|
} finally {
|
||||||
|
if (connection) connection.release();
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// 6.5. Get Hardware Components Master List
|
// 6.5. Get Hardware Components Master List
|
||||||
@@ -706,38 +945,212 @@ app.delete('/api/system-users/:id', async (req, res) => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
app.post('/api/maps/save', async (req, res) => {
|
// ==========================================
|
||||||
|
// 8. QR Asset Audit & Scan APIs (From origin/QR_setting)
|
||||||
|
// ==========================================
|
||||||
|
|
||||||
|
// GET all physical locations
|
||||||
|
app.get('/api/physical-locations', async (req, res) => {
|
||||||
|
try {
|
||||||
|
const [rows] = await pool.query('SELECT * FROM physical_locations ORDER BY location_code');
|
||||||
|
res.json(rows);
|
||||||
|
} catch (err) {
|
||||||
|
handleError(res, err, 'GET PHYSICAL LOCATIONS');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// POST register scan (mobile)
|
||||||
|
app.post('/api/audit/scan', async (req, res) => {
|
||||||
let connection;
|
let connection;
|
||||||
try {
|
try {
|
||||||
const { path, boxes } = req.body;
|
const { asset_code, physical_location_code } = req.body;
|
||||||
if (!path) return res.status(400).json({ error: 'Path is required' });
|
if (!asset_code || !physical_location_code) {
|
||||||
|
return res.status(400).json({ error: 'asset_code and physical_location_code are required' });
|
||||||
// 1. Get old config to track movements
|
|
||||||
let oldConfig = {};
|
|
||||||
if (fs.existsSync('map_config.json')) {
|
|
||||||
oldConfig = JSON.parse(fs.readFileSync('map_config.json', 'utf8') || '{}');
|
|
||||||
}
|
}
|
||||||
const oldBoxes = oldConfig[path] || [];
|
|
||||||
|
|
||||||
// 2. Save new config to file
|
|
||||||
oldConfig[path] = boxes;
|
|
||||||
fs.writeFileSync('map_config.json', JSON.stringify(oldConfig, null, 2));
|
|
||||||
|
|
||||||
// 3. Sync Database Assets (asset_location table)
|
|
||||||
connection = await pool.getConnection();
|
connection = await pool.getConnection();
|
||||||
for (const box of boxes) {
|
|
||||||
if (box.asset_id) {
|
// Verify if asset exists
|
||||||
console.log(`Syncing asset ${box.asset_id} to new position: [${box.x}, ${box.y}]`);
|
const [assets] = await connection.query('SELECT id FROM asset_core WHERE asset_code = ?', [asset_code]);
|
||||||
await connection.query(
|
if (assets.length === 0) {
|
||||||
'UPDATE asset_location SET loc_x = ?, loc_y = ? WHERE asset_id = ? AND is_active = 1',
|
return res.status(404).json({ error: `Asset with code ${asset_code} not found` });
|
||||||
[box.x, box.y, box.asset_id]
|
}
|
||||||
);
|
|
||||||
|
// Insert pending audit record
|
||||||
|
const [result] = await connection.query(
|
||||||
|
'INSERT INTO asset_audit_pending (asset_code, physical_location_code, status) VALUES (?, ?, ?)',
|
||||||
|
[asset_code, physical_location_code, 'PENDING']
|
||||||
|
);
|
||||||
|
|
||||||
|
res.json({ success: true, pending_id: result.insertId });
|
||||||
|
} catch (err) {
|
||||||
|
handleError(res, err, 'REGISTER SCAN');
|
||||||
|
} finally {
|
||||||
|
if (connection) connection.release();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// GET pending audits list (admin)
|
||||||
|
app.get('/api/audit/pending', async (req, res) => {
|
||||||
|
try {
|
||||||
|
const [rows] = await pool.query(`
|
||||||
|
SELECT
|
||||||
|
ap.*,
|
||||||
|
c.id AS asset_id,
|
||||||
|
c.asset_purpose,
|
||||||
|
c.asset_type,
|
||||||
|
pl.location_name,
|
||||||
|
pl.location_detail,
|
||||||
|
pl.map_image,
|
||||||
|
l.location AS old_location,
|
||||||
|
l.location_detail AS old_location_detail
|
||||||
|
FROM asset_audit_pending ap
|
||||||
|
JOIN asset_core c ON c.asset_code = ap.asset_code
|
||||||
|
JOIN physical_locations pl ON pl.location_code = ap.physical_location_code
|
||||||
|
LEFT JOIN asset_location l ON l.asset_id = c.id AND l.is_active = 1
|
||||||
|
WHERE ap.status = 'PENDING'
|
||||||
|
ORDER BY ap.scanned_at DESC
|
||||||
|
`);
|
||||||
|
res.json(rows);
|
||||||
|
} catch (err) {
|
||||||
|
handleError(res, err, 'GET PENDING AUDITS');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// POST approve audits (admin)
|
||||||
|
app.post('/api/audit/approve', async (req, res) => {
|
||||||
|
let connection;
|
||||||
|
try {
|
||||||
|
const { pending_ids, processed_by } = req.body;
|
||||||
|
if (!Array.isArray(pending_ids) || pending_ids.length === 0) {
|
||||||
|
return res.status(400).json({ error: 'pending_ids must be a non-empty array' });
|
||||||
|
}
|
||||||
|
|
||||||
|
connection = await pool.getConnection();
|
||||||
|
await connection.beginTransaction();
|
||||||
|
|
||||||
|
let mapConfigChanged = false;
|
||||||
|
let mapConfig = {};
|
||||||
|
if (fs.existsSync('map_config.json')) {
|
||||||
|
mapConfig = JSON.parse(fs.readFileSync('map_config.json', 'utf8') || '{}');
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const pendingId of pending_ids) {
|
||||||
|
// 1. Get pending scan details
|
||||||
|
const [pendings] = await connection.query(
|
||||||
|
'SELECT asset_code, physical_location_code FROM asset_audit_pending WHERE id = ? AND status = ?',
|
||||||
|
[pendingId, 'PENDING']
|
||||||
|
);
|
||||||
|
if (pendings.length === 0) continue;
|
||||||
|
|
||||||
|
const { asset_code, physical_location_code } = pendings[0];
|
||||||
|
|
||||||
|
// 2. Get asset ID
|
||||||
|
const [assets] = await connection.query('SELECT id FROM asset_core WHERE asset_code = ?', [asset_code]);
|
||||||
|
if (assets.length === 0) continue;
|
||||||
|
const assetId = assets[0].id;
|
||||||
|
|
||||||
|
// 3. Get physical location details
|
||||||
|
const [locations] = await connection.query(
|
||||||
|
'SELECT location_name, location_detail, map_image, map_x, map_y FROM physical_locations WHERE location_code = ?',
|
||||||
|
[physical_location_code]
|
||||||
|
);
|
||||||
|
if (locations.length === 0) continue;
|
||||||
|
const loc = locations[0];
|
||||||
|
|
||||||
|
// 4. Deactivate old active locations for this asset
|
||||||
|
await connection.query(
|
||||||
|
'UPDATE asset_location SET is_active = 0, deactivated_at = NOW() WHERE asset_id = ? AND is_active = 1',
|
||||||
|
[assetId]
|
||||||
|
);
|
||||||
|
|
||||||
|
// 5. Insert new active location
|
||||||
|
const pathPartsForApprove = loc.map_image.split('/');
|
||||||
|
const stdDetailForApprove = pathPartsForApprove[pathPartsForApprove.length - 2] || loc.location_detail;
|
||||||
|
await connection.query(`
|
||||||
|
INSERT INTO asset_location
|
||||||
|
(asset_id, location, location_detail, location_photo, loc_x, loc_y, physical_location_code, is_active)
|
||||||
|
VALUES (?, ?, ?, ?, ?, ?, ?, 1)
|
||||||
|
`, [assetId, loc.location_name, stdDetailForApprove, loc.map_image, loc.map_x, loc.map_y, physical_location_code]);
|
||||||
|
|
||||||
|
// 6. Update pending audit status
|
||||||
|
await connection.query(
|
||||||
|
'UPDATE asset_audit_pending SET status = ?, processed_at = NOW(), processed_by = ? WHERE id = ?',
|
||||||
|
['APPROVED', processed_by || 'ADMIN', pendingId]
|
||||||
|
);
|
||||||
|
|
||||||
|
// 7. Sync map_config.json
|
||||||
|
// Remove asset from any other map coordinates
|
||||||
|
for (const [mapPath, boxes] of Object.entries(mapConfig)) {
|
||||||
|
let changed = false;
|
||||||
|
const newBoxes = boxes.map(b => {
|
||||||
|
if (b.asset_id === assetId) {
|
||||||
|
changed = true;
|
||||||
|
return { ...b, asset_id: null };
|
||||||
|
}
|
||||||
|
return b;
|
||||||
|
});
|
||||||
|
if (changed) {
|
||||||
|
mapConfig[mapPath] = newBoxes;
|
||||||
|
mapConfigChanged = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add asset to the new map coordinate box matching map_image, map_x, map_y
|
||||||
|
if (mapConfig[loc.map_image]) {
|
||||||
|
const ax = parseFloat(loc.map_x);
|
||||||
|
const ay = parseFloat(loc.map_y);
|
||||||
|
const boxes = mapConfig[loc.map_image];
|
||||||
|
const matchedBox = boxes.find(b => {
|
||||||
|
const bx = parseFloat(b.x);
|
||||||
|
const by = parseFloat(b.y);
|
||||||
|
return Math.abs(bx - ax) < 0.1 && Math.abs(by - ay) < 0.1;
|
||||||
|
});
|
||||||
|
if (matchedBox) {
|
||||||
|
matchedBox.asset_id = assetId;
|
||||||
|
mapConfigChanged = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
res.json({ success: true, message: 'Map and Database synced successfully' });
|
if (mapConfigChanged) {
|
||||||
|
fs.writeFileSync('map_config.json', JSON.stringify(mapConfig, null, 2));
|
||||||
|
}
|
||||||
|
|
||||||
|
await connection.commit();
|
||||||
|
res.json({ success: true, message: 'Audits approved successfully' });
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
handleError(res, err, 'SAVE MAPS SYNC');
|
if (connection) await connection.rollback();
|
||||||
|
handleError(res, err, 'APPROVE AUDITS');
|
||||||
|
} finally {
|
||||||
|
if (connection) connection.release();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// POST reject audits (admin)
|
||||||
|
app.post('/api/audit/reject', async (req, res) => {
|
||||||
|
let connection;
|
||||||
|
try {
|
||||||
|
const { pending_ids, processed_by } = req.body;
|
||||||
|
if (!Array.isArray(pending_ids) || pending_ids.length === 0) {
|
||||||
|
return res.status(400).json({ error: 'pending_ids must be a non-empty array' });
|
||||||
|
}
|
||||||
|
|
||||||
|
connection = await pool.getConnection();
|
||||||
|
await connection.beginTransaction();
|
||||||
|
|
||||||
|
for (const pendingId of pending_ids) {
|
||||||
|
await connection.query(
|
||||||
|
'UPDATE asset_audit_pending SET status = ?, processed_at = NOW(), processed_by = ? WHERE id = ? AND status = ?',
|
||||||
|
['REJECTED', processed_by || 'ADMIN', pendingId, 'PENDING']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
await connection.commit();
|
||||||
|
res.json({ success: true, message: 'Audits rejected successfully' });
|
||||||
|
} catch (err) {
|
||||||
|
if (connection) await connection.rollback();
|
||||||
|
handleError(res, err, 'REJECT AUDITS');
|
||||||
} finally {
|
} finally {
|
||||||
if (connection) connection.release();
|
if (connection) connection.release();
|
||||||
}
|
}
|
||||||
@@ -771,11 +1184,10 @@ app.post('/api/upload', (req, res) => {
|
|||||||
app.get('/health', async (req, res) => {
|
app.get('/health', async (req, res) => {
|
||||||
try {
|
try {
|
||||||
const connection = await pool.getConnection();
|
const connection = await pool.getConnection();
|
||||||
const result = await connection.query('SELECT 1');
|
await connection.query('SELECT 1');
|
||||||
connection.release();
|
connection.release();
|
||||||
res.status(200).json({ status: 'ok', db: 'connected' });
|
res.status(200).json({ status: 'ok', db: 'connected' });
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
// Return degraded status if DB unreachable, but still report service as alive
|
|
||||||
res.status(200).json({ status: 'degraded', db: 'unreachable', error: err.message });
|
res.status(200).json({ status: 'degraded', db: 'unreachable', error: err.message });
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -784,7 +1196,7 @@ app.get('/health', async (req, res) => {
|
|||||||
app.get('/ready', async (req, res) => {
|
app.get('/ready', async (req, res) => {
|
||||||
try {
|
try {
|
||||||
const connection = await pool.getConnection();
|
const connection = await pool.getConnection();
|
||||||
const result = await connection.query('SELECT 1');
|
await connection.query('SELECT 1');
|
||||||
connection.release();
|
connection.release();
|
||||||
res.status(200).json({ status: 'ready' });
|
res.status(200).json({ status: 'ready' });
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
@@ -792,6 +1204,6 @@ app.get('/ready', async (req, res) => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
app.listen(3000, '0.0.0.0', () => {
|
app.listen(process.env.PORT || 3000, '0.0.0.0', () => {
|
||||||
console.log('📡 ITAM BACKEND SERVER RUNNING ON PORT 3000 (V3 Normalized)');
|
console.log(`📡 ITAM BACKEND SERVER RUNNING ON PORT ${process.env.PORT || 3000} (V3 Normalized)`);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -202,7 +202,57 @@ class DomainAssetModal extends BaseModal {
|
|||||||
if (logs.length === 0) {
|
if (logs.length === 0) {
|
||||||
container.innerHTML = '<div style="color:var(--mute); padding:1rem; text-align:center;">이력이 없습니다.</div>';
|
container.innerHTML = '<div style="color:var(--mute); padding:1rem; text-align:center;">이력이 없습니다.</div>';
|
||||||
} else {
|
} else {
|
||||||
container.innerHTML = logs.map(l => `<div class="history-item"><div class="history-date">${l.log_date || ''}</div><div class="history-user">${l.log_user || '시스템'}</div><div class="history-details">${l.details}</div></div>`).join('');
|
const createdDate = this.currentAsset?.created_at ? this.currentAsset.created_at.substring(0, 10) : '';
|
||||||
|
|
||||||
|
const grouped: Record<string, typeof logs> = {};
|
||||||
|
logs.forEach(l => {
|
||||||
|
const date = l.log_date || '날짜 미지정';
|
||||||
|
if (!grouped[date]) grouped[date] = [];
|
||||||
|
grouped[date].push(l);
|
||||||
|
});
|
||||||
|
|
||||||
|
container.innerHTML = Object.entries(grouped).map(([date, dateLogs]) => {
|
||||||
|
const entriesHtml = dateLogs.map((l, idx) => {
|
||||||
|
const isLast = idx === dateLogs.length - 1;
|
||||||
|
const borderStyle = isLast ? '' : 'border-bottom: 1px dashed var(--hairline); padding-bottom: 8px; margin-bottom: 8px;';
|
||||||
|
|
||||||
|
let displayDetails = l.details;
|
||||||
|
if (l.details && l.details.trim().startsWith('{')) {
|
||||||
|
try {
|
||||||
|
const data = JSON.parse(l.details);
|
||||||
|
if (data.type === 'checkout') {
|
||||||
|
displayDetails = `[불출] ${data.user || ''} (${data.dept || ''}) ${data.memo ? `| 메모: ${data.memo}` : ''}`;
|
||||||
|
} else if (data.type === 'return') {
|
||||||
|
displayDetails = `[반납] ${data.user || ''} (${data.dept || ''}) ${data.memo ? `| 메모: ${data.memo}` : ''}`;
|
||||||
|
} else if (data.type === 'move') {
|
||||||
|
displayDetails = `[이동] ${data.user || ''} (${data.dept || ''}) ➔ ${data.targetUser || ''} (${data.targetDept || ''}) ${data.memo ? `| 메모: ${data.memo}` : ''}`;
|
||||||
|
}
|
||||||
|
} catch (e) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
return `
|
||||||
|
<div class="history-entry" style="${borderStyle}">
|
||||||
|
<div style="font-weight: 600; color: var(--primary); opacity: 0.8; margin-bottom: 4px; display: flex; align-items: center; gap: 6px;">
|
||||||
|
<span style="display: inline-block; width: 4px; height: 4px; background-color: var(--primary); border-radius: 50%;"></span>
|
||||||
|
${l.log_user || '시스템'}
|
||||||
|
</div>
|
||||||
|
<div style="color: var(--primary); padding-left: 10px; line-height: 1.5;">${displayDetails}</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}).join('');
|
||||||
|
|
||||||
|
const isInitialReg = date === createdDate;
|
||||||
|
const regBadge = isInitialReg ? `<span class="badge-reg" style="font-size: 10px; padding: 1px 5px; margin-left: 6px; background-color: rgba(16, 185, 129, 0.1); color: #10b981; border: 1px solid rgba(16, 185, 129, 0.2); border-radius: 4px; font-weight: 600;">최초등록</span>` : '';
|
||||||
|
|
||||||
|
return `
|
||||||
|
<div class="history-item">
|
||||||
|
<div class="history-date" style="display: flex; align-items: center;">${date} ${regBadge}</div>
|
||||||
|
<div class="history-details" style="display: flex; flex-direction: column; gap: 4px;">
|
||||||
|
${entriesHtml}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}).join('');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { state, saveAsset, deleteAsset } from '../../core/state';
|
import { state, saveAsset, deleteAsset } from '../../core/state';
|
||||||
import { ASSET_SCHEMA, UI_TEXT } from '../../core/schema';
|
import { ASSET_SCHEMA, UI_TEXT } from '../../core/schema';
|
||||||
import { API_BASE_URL, calculatePcScoreDeductive, getPcGrade } from '../../core/utils';
|
import { calculatePcScoreDeductive, getPcGrade, API_BASE_URL } from '../../core/utils';
|
||||||
import {
|
import {
|
||||||
generateOptionsHTML,
|
generateOptionsHTML,
|
||||||
setFieldValue,
|
setFieldValue,
|
||||||
@@ -11,6 +11,7 @@ import {
|
|||||||
} from './ModalUtils';
|
} from './ModalUtils';
|
||||||
import { CORP_LIST, LOCATION_DATA, CATEGORY_TYPE_MAP, HW_STATUS_LIST, ORG_LIST, IMAGE_LOCATIONS, TYPE_PREFIX_MAP } from './SharedData';
|
import { CORP_LIST, LOCATION_DATA, CATEGORY_TYPE_MAP, HW_STATUS_LIST, ORG_LIST, IMAGE_LOCATIONS, TYPE_PREFIX_MAP } from './SharedData';
|
||||||
import { BaseModal } from './BaseModal';
|
import { BaseModal } from './BaseModal';
|
||||||
|
import { QRPrinter } from '../../core/qr_print';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 하드웨어 자산 상세 모달 (Styled Main Edition)
|
* 하드웨어 자산 상세 모달 (Styled Main Edition)
|
||||||
@@ -30,9 +31,11 @@ class HwAssetModal extends BaseModal {
|
|||||||
<div id="hw-asset-modal" class="modal-overlay hidden">
|
<div id="hw-asset-modal" class="modal-overlay hidden">
|
||||||
<div class="modal-content wide">
|
<div class="modal-content wide">
|
||||||
<div class="modal-header">
|
<div class="modal-header">
|
||||||
<div class="header-left">
|
<div class="header-left" style="display: flex; align-items: center; gap: 0.75rem; flex-wrap: wrap;">
|
||||||
<h2 id="hw-modal-title" class="modal-title">${this.title}</h2>
|
<h2 id="hw-modal-title" class="modal-title" style="display: none;">${this.title}</h2>
|
||||||
<div id="hw-header-identity" class="header-identity"></div>
|
<div id="hw-header-identity" class="header-identity" style="display: inline-flex; gap: 0.5rem; align-items: center;"></div>
|
||||||
|
<button id="btn-print-hw-qr" class="btn btn-outline btn-primary hidden" style="padding: 2px 8px; font-size: 11px; height: 22px; margin: 0; line-height: 1; display: inline-flex; align-items: center; justify-content: center; cursor: pointer;">QR 인쇄</button>
|
||||||
|
<span id="hw-modal-audit-approved-badge" style="display: none; align-items: center; background-color: rgba(16, 185, 129, 0.08); color: #059669; border: 1px solid rgba(16, 185, 129, 0.18); padding: 1px 6px; border-radius: 4px; font-size: 10px; font-weight: 600; height: 20px; line-height: 1; vertical-align: middle; white-space: nowrap; margin-left: 4px;">승인완료</span>
|
||||||
</div>
|
</div>
|
||||||
<button id="btn-close-hw-modal" class="btn-icon" aria-label="닫기">×</button>
|
<button id="btn-close-hw-modal" class="btn-icon" aria-label="닫기">×</button>
|
||||||
</div>
|
</div>
|
||||||
@@ -74,7 +77,7 @@ class HwAssetModal extends BaseModal {
|
|||||||
<label>${ASSET_SCHEMA.HW_STATUS.ui}</label>
|
<label>${ASSET_SCHEMA.HW_STATUS.ui}</label>
|
||||||
<select id="hw-hw_status" name="hw_status">${generateOptionsHTML(HW_STATUS_LIST)}</select>
|
<select id="hw-hw_status" name="hw_status">${generateOptionsHTML(HW_STATUS_LIST)}</select>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group">
|
<div class="form-group service-type-field">
|
||||||
<label>${ASSET_SCHEMA.SERVICE_TYPE.ui}</label>
|
<label>${ASSET_SCHEMA.SERVICE_TYPE.ui}</label>
|
||||||
<select id="hw-service_type" name="service_type">
|
<select id="hw-service_type" name="service_type">
|
||||||
<option value="외부">외부</option>
|
<option value="외부">외부</option>
|
||||||
@@ -95,6 +98,9 @@ class HwAssetModal extends BaseModal {
|
|||||||
|
|
||||||
<!-- [SECTION 2] 조직 및 사용자 정보 -->
|
<!-- [SECTION 2] 조직 및 사용자 정보 -->
|
||||||
<div class="form-section-title">사용자 및 조직 정보</div>
|
<div class="form-section-title">사용자 및 조직 정보</div>
|
||||||
|
<div id="hw-pc-workflow-notice" class="form-group full-width hidden" style="background-color: rgba(59, 130, 246, 0.05); border: 1px solid rgba(59, 130, 246, 0.15); padding: 8px 12px; border-radius: 6px; font-size: 11px; color: var(--primary); line-height: 1.5; margin-bottom: 12px;">
|
||||||
|
💡 PC 자산은 데이터 정합성을 위해 '사용자 및 조직 정보'만 수정이 제한되며, 사양 및 기타 정보는 수정창에서 수정할 수 있습니다.
|
||||||
|
</div>
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label>${ASSET_SCHEMA.CURRENT_DEPT.ui}</label>
|
<label>${ASSET_SCHEMA.CURRENT_DEPT.ui}</label>
|
||||||
<select id="hw-current_dept" name="current_dept">${generateOptionsHTML(ORG_LIST)}</select>
|
<select id="hw-current_dept" name="current_dept">${generateOptionsHTML(ORG_LIST)}</select>
|
||||||
@@ -107,9 +113,14 @@ class HwAssetModal extends BaseModal {
|
|||||||
<label>${ASSET_SCHEMA.MANAGER_SUB.ui}</label>
|
<label>${ASSET_SCHEMA.MANAGER_SUB.ui}</label>
|
||||||
<input type="text" id="hw-manager_secondary" name="manager_secondary" />
|
<input type="text" id="hw-manager_secondary" name="manager_secondary" />
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group personal-only">
|
<div class="form-group personal-only relative">
|
||||||
<label>${ASSET_SCHEMA.CURRENT_USER.ui}</label>
|
<label>${ASSET_SCHEMA.CURRENT_USER.ui}</label>
|
||||||
<input type="text" id="hw-user_current" name="user_current" />
|
<input type="text" id="hw-user_current" name="user_current" autocomplete="off" />
|
||||||
|
<div id="hw-user-current-list" class="autocomplete-list hidden"></div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group personal-only">
|
||||||
|
<label>${ASSET_SCHEMA.EMP_NO.ui}</label>
|
||||||
|
<input type="text" id="hw-emp_no" name="emp_no" readonly style="background-color: #f1f5f9; cursor: not-allowed;" />
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group personal-only">
|
<div class="form-group personal-only">
|
||||||
<label>${ASSET_SCHEMA.USER_POSITION.ui}</label>
|
<label>${ASSET_SCHEMA.USER_POSITION.ui}</label>
|
||||||
@@ -130,6 +141,10 @@ class HwAssetModal extends BaseModal {
|
|||||||
<label>${ASSET_SCHEMA.SERIAL_NUM.ui}</label>
|
<label>${ASSET_SCHEMA.SERIAL_NUM.ui}</label>
|
||||||
<input type="text" id="hw-serial_num" name="serial_num" />
|
<input type="text" id="hw-serial_num" name="serial_num" />
|
||||||
</div>
|
</div>
|
||||||
|
<div class="form-group mainboard-only">
|
||||||
|
<label>${ASSET_SCHEMA.MAINBOARD.ui}</label>
|
||||||
|
<input type="text" id="hw-mainboard" name="mainboard" />
|
||||||
|
</div>
|
||||||
<div class="form-group spec-only">
|
<div class="form-group spec-only">
|
||||||
<label>${ASSET_SCHEMA.OS.ui}</label>
|
<label>${ASSET_SCHEMA.OS.ui}</label>
|
||||||
<input type="text" id="hw-os" name="os" />
|
<input type="text" id="hw-os" name="os" />
|
||||||
@@ -264,10 +279,26 @@ class HwAssetModal extends BaseModal {
|
|||||||
const detailSelect = document.getElementById('hw-location_detail') as HTMLSelectElement;
|
const detailSelect = document.getElementById('hw-location_detail') as HTMLSelectElement;
|
||||||
|
|
||||||
this.fetchMapConfig();
|
this.fetchMapConfig();
|
||||||
|
|
||||||
|
const qrPrintBtn = document.getElementById('btn-print-hw-qr');
|
||||||
|
qrPrintBtn?.addEventListener('click', () => {
|
||||||
|
if (this.currentAsset && this.currentAsset.asset_code) {
|
||||||
|
QRPrinter.print([{
|
||||||
|
type: 'asset',
|
||||||
|
code: this.currentAsset.asset_code,
|
||||||
|
title: '[ HM IT ASSET ]',
|
||||||
|
subtitle: this.currentAsset.model_name || this.currentAsset.asset_purpose || this.currentAsset.category || 'IT 자산',
|
||||||
|
dept: this.currentAsset.current_dept || '-',
|
||||||
|
user: this.currentAsset.user_current || '-'
|
||||||
|
}]);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
this.fetchMasterComponents().then(() => {
|
this.fetchMasterComponents().then(() => {
|
||||||
this.bindAutocomplete('hw-cpu', 'hw-cpu-list', 'CPU');
|
this.bindAutocomplete('hw-cpu', 'hw-cpu-list', 'CPU');
|
||||||
this.bindAutocomplete('hw-ram', 'hw-ram-list', 'RAM');
|
this.bindAutocomplete('hw-ram', 'hw-ram-list', 'RAM');
|
||||||
this.bindAutocomplete('hw-gpu', 'hw-gpu-list', 'GPU');
|
this.bindAutocomplete('hw-gpu', 'hw-gpu-list', 'GPU');
|
||||||
|
this.bindUserAutocomplete();
|
||||||
});
|
});
|
||||||
|
|
||||||
categorySelect.addEventListener('change', () => {
|
categorySelect.addEventListener('change', () => {
|
||||||
@@ -285,6 +316,12 @@ class HwAssetModal extends BaseModal {
|
|||||||
typeSelect.addEventListener('change', () => {
|
typeSelect.addEventListener('change', () => {
|
||||||
this.applyRoleVisibility();
|
this.applyRoleVisibility();
|
||||||
this.updateHeaderIdentity(this.currentAsset);
|
this.updateHeaderIdentity(this.currentAsset);
|
||||||
|
|
||||||
|
if (typeSelect.value === '공용PC') {
|
||||||
|
setFieldValue('hw-user_current', '');
|
||||||
|
setFieldValue('hw-emp_no', '');
|
||||||
|
setFieldValue('hw-user_position', '공용PC');
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
bindLocationEvents('hw-bldg-select', 'hw-location_detail', '', '');
|
bindLocationEvents('hw-bldg-select', 'hw-location_detail', '', '');
|
||||||
@@ -296,10 +333,17 @@ class HwAssetModal extends BaseModal {
|
|||||||
document.getElementById('btn-gen-hw-code')?.addEventListener('click', async () => {
|
document.getElementById('btn-gen-hw-code')?.addEventListener('click', async () => {
|
||||||
const cat = categorySelect.value;
|
const cat = categorySelect.value;
|
||||||
if (!cat) { alert('구분을 먼저 선택해주세요.'); return; }
|
if (!cat) { alert('구분을 먼저 선택해주세요.'); return; }
|
||||||
const prefix = TYPE_PREFIX_MAP[cat] || 'ETC';
|
|
||||||
const purchaseDate = (document.getElementById('hw-purchase_date') as HTMLInputElement)?.value || '';
|
const purchaseDate = (document.getElementById('hw-purchase_date') as HTMLInputElement)?.value || '';
|
||||||
|
if (!purchaseDate.trim()) {
|
||||||
|
alert('구매일자를 먼저 입력해 주세요. 구매일자가 없으면 자산번호를 생성할 수 없습니다.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const type = (document.getElementById('hw-asset_type') as HTMLSelectElement)?.value || '';
|
||||||
|
const prefix = TYPE_PREFIX_MAP[type] || TYPE_PREFIX_MAP[cat] || 'ETC';
|
||||||
try {
|
try {
|
||||||
const res = await fetch(`${API_BASE_URL}/api/generate-asset-code?prefix=${prefix}&purchaseDate=${purchaseDate}`);
|
const res = await fetch(`/api/generate-asset-code?prefix=${prefix}&purchaseDate=${purchaseDate}`);
|
||||||
const data = await res.json();
|
const data = await res.json();
|
||||||
if (data.nextCode) setFieldValue('hw-asset_code', data.nextCode);
|
if (data.nextCode) setFieldValue('hw-asset_code', data.nextCode);
|
||||||
} catch (err) { console.error('코드 생성 실패:', err); }
|
} catch (err) { console.error('코드 생성 실패:', err); }
|
||||||
@@ -317,7 +361,7 @@ class HwAssetModal extends BaseModal {
|
|||||||
const reader = new FileReader();
|
const reader = new FileReader();
|
||||||
reader.onload = async () => {
|
reader.onload = async () => {
|
||||||
try {
|
try {
|
||||||
const res = await fetch(`${API_BASE_URL}/api/upload`, {
|
const res = await fetch('/api/upload', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({ fileName: file.name, fileData: reader.result })
|
body: JSON.stringify({ fileName: file.name, fileData: reader.result })
|
||||||
@@ -326,7 +370,7 @@ class HwAssetModal extends BaseModal {
|
|||||||
if (data.success) {
|
if (data.success) {
|
||||||
setFieldValue('hw-approval_document', data.filePath);
|
setFieldValue('hw-approval_document', data.filePath);
|
||||||
if (fileLinkContainer) {
|
if (fileLinkContainer) {
|
||||||
fileLinkContainer.innerHTML = `<a href="http://${location.hostname}:3000${data.filePath}" target="_blank" class="btn btn-outline btn-sm">[파일 보기]</a>`;
|
fileLinkContainer.innerHTML = `<a href="${data.filePath}" target="_blank" class="btn btn-outline btn-sm">[파일 보기]</a>`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (err) { console.error('파일 업로드 실패:', err); alert('파일 업로드 중 오류가 발생했습니다.'); }
|
} catch (err) { console.error('파일 업로드 실패:', err); alert('파일 업로드 중 오류가 발생했습니다.'); }
|
||||||
@@ -382,10 +426,11 @@ class HwAssetModal extends BaseModal {
|
|||||||
if (!assetCode) {
|
if (!assetCode) {
|
||||||
const cat = categorySelect.value;
|
const cat = categorySelect.value;
|
||||||
if (!cat) { alert('구분을 먼저 선택해주세요.'); return; }
|
if (!cat) { alert('구분을 먼저 선택해주세요.'); return; }
|
||||||
const prefix = TYPE_PREFIX_MAP[cat] || 'ETC';
|
const type = (document.getElementById('hw-asset_type') as HTMLSelectElement)?.value || '';
|
||||||
|
const prefix = TYPE_PREFIX_MAP[type] || TYPE_PREFIX_MAP[cat] || 'ETC';
|
||||||
const purchaseDate = (document.getElementById('hw-purchase_date') as HTMLInputElement)?.value || '';
|
const purchaseDate = (document.getElementById('hw-purchase_date') as HTMLInputElement)?.value || '';
|
||||||
try {
|
try {
|
||||||
const res = await fetch(`http://${location.hostname}:3000/api/generate-asset-code?prefix=${prefix}&purchaseDate=${purchaseDate}`);
|
const res = await fetch(`/api/generate-asset-code?prefix=${prefix}&purchaseDate=${purchaseDate}`);
|
||||||
const data = await res.json();
|
const data = await res.json();
|
||||||
if (data.nextCode) {
|
if (data.nextCode) {
|
||||||
setFieldValue('hw-asset_code', data.nextCode);
|
setFieldValue('hw-asset_code', data.nextCode);
|
||||||
@@ -434,6 +479,27 @@ class HwAssetModal extends BaseModal {
|
|||||||
formData.forEach((value, key) => { if (key !== 'id') updated[key] = value; });
|
formData.forEach((value, key) => { if (key !== 'id') updated[key] = value; });
|
||||||
updated.location = bldgSelect.value;
|
updated.location = bldgSelect.value;
|
||||||
|
|
||||||
|
// 부품 마스터 기준 정합성 검증 (CPU, GPU, RAM)
|
||||||
|
const checkFields = [
|
||||||
|
{ name: 'cpu', label: 'CPU', category: 'CPU' },
|
||||||
|
{ name: 'gpu', label: 'GPU', category: 'GPU' },
|
||||||
|
{ name: 'ram', label: 'RAM', category: 'RAM' }
|
||||||
|
];
|
||||||
|
|
||||||
|
for (const field of checkFields) {
|
||||||
|
const value = String(updated[field.name] || '').trim();
|
||||||
|
if (value) {
|
||||||
|
const isExists = this.masterComponents.some(c =>
|
||||||
|
c.category.toUpperCase() === field.category &&
|
||||||
|
c.component_name.trim().toLowerCase() === value.toLowerCase()
|
||||||
|
);
|
||||||
|
if (!isExists) {
|
||||||
|
alert(`입력하신 ${field.label} "${value}"은(는) 부품 마스터에 등록되지 않은 규격입니다. 자동완성 목록에서 선택하거나 부품마스터에 먼저 등록해 주세요.`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (await saveAsset(this.getCategoryKey(updated), updated)) {
|
if (await saveAsset(this.getCategoryKey(updated), updated)) {
|
||||||
alert(UI_TEXT.MESSAGES.SAVE_SUCCESS);
|
alert(UI_TEXT.MESSAGES.SAVE_SUCCESS);
|
||||||
onSave(); this.close(); closeModals();
|
onSave(); this.close(); closeModals();
|
||||||
@@ -563,6 +629,7 @@ class HwAssetModal extends BaseModal {
|
|||||||
setFieldValue('hw-manager_primary', asset.manager_primary || '');
|
setFieldValue('hw-manager_primary', asset.manager_primary || '');
|
||||||
setFieldValue('hw-manager_secondary', asset.manager_secondary || '');
|
setFieldValue('hw-manager_secondary', asset.manager_secondary || '');
|
||||||
setFieldValue('hw-user_current', asset.user_current || '');
|
setFieldValue('hw-user_current', asset.user_current || '');
|
||||||
|
setFieldValue('hw-emp_no', asset.emp_no || '');
|
||||||
setFieldValue('hw-user_position', asset.user_position || '');
|
setFieldValue('hw-user_position', asset.user_position || '');
|
||||||
setFieldValue('hw-previous_user', asset.previous_user || '');
|
setFieldValue('hw-previous_user', asset.previous_user || '');
|
||||||
setFieldValue('hw-model_name', asset.model_name || '');
|
setFieldValue('hw-model_name', asset.model_name || '');
|
||||||
@@ -621,7 +688,7 @@ class HwAssetModal extends BaseModal {
|
|||||||
if (docName) docName.textContent = asset.approval_document ? asset.approval_document.split('/').pop() : '파일 선택...';
|
if (docName) docName.textContent = asset.approval_document ? asset.approval_document.split('/').pop() : '파일 선택...';
|
||||||
const fileLinkContainer = document.getElementById('hw-file-link-container');
|
const fileLinkContainer = document.getElementById('hw-file-link-container');
|
||||||
if (fileLinkContainer && asset.approval_document) {
|
if (fileLinkContainer && asset.approval_document) {
|
||||||
fileLinkContainer.innerHTML = `<a href="http://${location.hostname}:3000${asset.approval_document}" target="_blank" class="btn btn-outline btn-sm">[파일 보기]</a>`;
|
fileLinkContainer.innerHTML = `<a href="${asset.approval_document}" target="_blank" class="btn btn-outline btn-sm">[파일 보기]</a>`;
|
||||||
} else if (fileLinkContainer) {
|
} else if (fileLinkContainer) {
|
||||||
fileLinkContainer.innerHTML = '';
|
fileLinkContainer.innerHTML = '';
|
||||||
}
|
}
|
||||||
@@ -642,6 +709,19 @@ class HwAssetModal extends BaseModal {
|
|||||||
protected onAfterOpen(asset: any, mode: string): void {
|
protected onAfterOpen(asset: any, mode: string): void {
|
||||||
const genBtn = document.getElementById('btn-gen-hw-code');
|
const genBtn = document.getElementById('btn-gen-hw-code');
|
||||||
if (genBtn) genBtn.style.display = (mode === 'add') ? 'inline-flex' : 'none';
|
if (genBtn) genBtn.style.display = (mode === 'add') ? 'inline-flex' : 'none';
|
||||||
|
|
||||||
|
const qrBtn = document.getElementById('btn-print-hw-qr');
|
||||||
|
if (qrBtn) {
|
||||||
|
const hasCode = asset && asset.asset_code && asset.asset_code.trim() !== '';
|
||||||
|
qrBtn.classList.toggle('hidden', mode !== 'view' || !hasCode);
|
||||||
|
}
|
||||||
|
|
||||||
|
const approvedBadge = document.getElementById('hw-modal-audit-approved-badge');
|
||||||
|
if (approvedBadge) {
|
||||||
|
const isApproved = asset && asset.is_audit_approved;
|
||||||
|
approvedBadge.style.display = (mode === 'view' && isApproved) ? 'inline-flex' : 'none';
|
||||||
|
}
|
||||||
|
|
||||||
this.toggleFileUploadUI(mode !== 'view');
|
this.toggleFileUploadUI(mode !== 'view');
|
||||||
this.toggleEditOnlyBtns(mode !== 'view');
|
this.toggleEditOnlyBtns(mode !== 'view');
|
||||||
this.updateMapButtonVisibility();
|
this.updateMapButtonVisibility();
|
||||||
@@ -688,20 +768,97 @@ class HwAssetModal extends BaseModal {
|
|||||||
const hasSpec = specCategories.includes(category) || type.includes('서버PC');
|
const hasSpec = specCategories.includes(category) || type.includes('서버PC');
|
||||||
const noNetCategories = ['저장매체', '네트워크', '공간정보장비', 'PC부품', '사무가구'];
|
const noNetCategories = ['저장매체', '네트워크', '공간정보장비', 'PC부품', '사무가구'];
|
||||||
const showNet = (isInfra || isPersonal) && !noNetCategories.includes(category);
|
const showNet = (isInfra || isPersonal) && !noNetCategories.includes(category);
|
||||||
const hasSN = !['사무가구', 'PC부품'].includes(category);
|
const hasSN = ['외부SW', '내부SW'].includes(category);
|
||||||
|
const showMainboard = category === 'PC';
|
||||||
const isParts = ['PC부품', '사무가구'].includes(category);
|
const isParts = ['PC부품', '사무가구'].includes(category);
|
||||||
const showRemote = category === '서버' || type.includes('서버');
|
const showRemote = category === '서버' || type.includes('서버');
|
||||||
|
const showServiceType = category === '서버' || type === '서버PC';
|
||||||
|
|
||||||
document.querySelectorAll('.remote-section, .remote-field, .monitoring-field').forEach(el => (el as HTMLElement).style.display = showRemote ? '' : 'none');
|
document.querySelectorAll('.remote-section, .remote-field, .monitoring-field').forEach(el => (el as HTMLElement).style.display = showRemote ? '' : 'none');
|
||||||
|
document.querySelectorAll('.service-type-field').forEach(el => (el as HTMLElement).style.display = showServiceType ? '' : 'none');
|
||||||
document.querySelectorAll('.net-only').forEach(el => (el as HTMLElement).style.display = showNet ? '' : 'none');
|
document.querySelectorAll('.net-only').forEach(el => (el as HTMLElement).style.display = showNet ? '' : 'none');
|
||||||
document.querySelectorAll('.spec-only').forEach(el => (el as HTMLElement).style.display = hasSpec ? '' : 'none');
|
document.querySelectorAll('.spec-only').forEach(el => (el as HTMLElement).style.display = hasSpec ? '' : 'none');
|
||||||
document.querySelectorAll('.location-section, .location-field').forEach(el => (el as HTMLElement).style.display = (isInfra || category === '공간정보장비') ? '' : 'none');
|
document.querySelectorAll('.location-section, .location-field').forEach(el => (el as HTMLElement).style.display = (isInfra || category === '공간정보장비') ? '' : 'none');
|
||||||
document.querySelectorAll('.org-user-section, .org-user-field').forEach(el => (el as HTMLElement).style.display = (isPersonal || isParts || category === '업무지원장비') ? '' : 'none');
|
document.querySelectorAll('.org-user-section, .org-user-field').forEach(el => (el as HTMLElement).style.display = (isPersonal || isParts || category === '업무지원장비') ? '' : 'none');
|
||||||
document.querySelectorAll('.personal-only').forEach(el => (el as HTMLElement).style.display = isPersonal ? '' : 'none');
|
document.querySelectorAll('.personal-only').forEach(el => (el as HTMLElement).style.display = isPersonal ? '' : 'none');
|
||||||
document.querySelectorAll('.sn-only').forEach(el => (el as HTMLElement).style.display = hasSN ? '' : 'none');
|
document.querySelectorAll('.sn-only').forEach(el => (el as HTMLElement).style.display = hasSN ? '' : 'none');
|
||||||
|
document.querySelectorAll('.mainboard-only').forEach(el => (el as HTMLElement).style.display = showMainboard ? '' : 'none');
|
||||||
document.querySelectorAll('.monitor-only').forEach(el => (el as HTMLElement).style.display = type.includes('모니터') ? '' : 'none');
|
document.querySelectorAll('.monitor-only').forEach(el => (el as HTMLElement).style.display = type.includes('모니터') ? '' : 'none');
|
||||||
document.querySelectorAll('.parts-only').forEach(el => (el as HTMLElement).style.display = isParts ? '' : 'none');
|
document.querySelectorAll('.parts-only').forEach(el => (el as HTMLElement).style.display = isParts ? '' : 'none');
|
||||||
document.querySelectorAll('.hardware-section').forEach(el => (el as HTMLElement).style.display = (hasSpec || isParts) ? '' : 'none');
|
document.querySelectorAll('.hardware-section').forEach(el => (el as HTMLElement).style.display = (hasSpec || isParts) ? '' : 'none');
|
||||||
|
|
||||||
|
// Lock only User and Organization Information for PC category during edit mode
|
||||||
|
const isEditMode = this.currentMode === 'edit';
|
||||||
|
const isPC = category === 'PC';
|
||||||
|
|
||||||
|
const noticeEl = document.getElementById('hw-pc-workflow-notice');
|
||||||
|
if (noticeEl) {
|
||||||
|
if (isPC && isEditMode) {
|
||||||
|
noticeEl.classList.remove('hidden');
|
||||||
|
} else {
|
||||||
|
noticeEl.classList.add('hidden');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const lockedUserFields = [
|
||||||
|
'hw-current_dept',
|
||||||
|
'hw-manager_primary',
|
||||||
|
'hw-manager_secondary',
|
||||||
|
'hw-user_current',
|
||||||
|
'hw-emp_no',
|
||||||
|
'hw-user_position',
|
||||||
|
'hw-previous_user'
|
||||||
|
];
|
||||||
|
|
||||||
|
const allFormControls = this.formEl ? this.formEl.querySelectorAll('input, select, textarea, button') : [];
|
||||||
|
|
||||||
|
allFormControls.forEach(control => {
|
||||||
|
const el = control as HTMLElement;
|
||||||
|
const id = el.id;
|
||||||
|
|
||||||
|
if (el.tagName === 'INPUT' && (el as HTMLInputElement).type === 'hidden') return;
|
||||||
|
if (id === 'hw-asset_code' || id === 'btn-gen-hw-code') return;
|
||||||
|
|
||||||
|
if (isPC && isEditMode && lockedUserFields.includes(id)) {
|
||||||
|
// Lock user information fields for PC in edit mode
|
||||||
|
if (el.tagName === 'SELECT') {
|
||||||
|
el.setAttribute('disabled', 'true');
|
||||||
|
} else if (el.tagName === 'INPUT' || el.tagName === 'TEXTAREA') {
|
||||||
|
el.setAttribute('readonly', 'true');
|
||||||
|
(el as HTMLInputElement).style.backgroundColor = '#f1f5f9';
|
||||||
|
(el as HTMLInputElement).style.cursor = 'not-allowed';
|
||||||
|
} else if (el.tagName === 'BUTTON') {
|
||||||
|
el.setAttribute('disabled', 'true');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Normal behavior based on modal edit/view mode (includes add mode which has this.isEditMode = true)
|
||||||
|
if (!this.isEditMode) {
|
||||||
|
if (el.tagName === 'SELECT') {
|
||||||
|
el.setAttribute('disabled', 'true');
|
||||||
|
} else if (el.tagName === 'INPUT' || el.tagName === 'TEXTAREA') {
|
||||||
|
el.setAttribute('readonly', 'true');
|
||||||
|
(el as HTMLInputElement).style.backgroundColor = '';
|
||||||
|
(el as HTMLInputElement).style.cursor = '';
|
||||||
|
} else if (el.tagName === 'BUTTON') {
|
||||||
|
if (id !== 'btn-print-hw-qr' && id !== 'btn-close-hw-modal') {
|
||||||
|
el.setAttribute('disabled', 'true');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (el.tagName === 'SELECT') {
|
||||||
|
el.removeAttribute('disabled');
|
||||||
|
} else if (el.tagName === 'INPUT' || el.tagName === 'TEXTAREA') {
|
||||||
|
if (id !== 'hw-emp_no') {
|
||||||
|
el.removeAttribute('readonly');
|
||||||
|
(el as HTMLInputElement).style.backgroundColor = '';
|
||||||
|
(el as HTMLInputElement).style.cursor = '';
|
||||||
|
}
|
||||||
|
} else if (el.tagName === 'BUTTON') {
|
||||||
|
el.removeAttribute('disabled');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private updateMapButtonVisibility() {
|
private updateMapButtonVisibility() {
|
||||||
@@ -902,12 +1059,148 @@ class HwAssetModal extends BaseModal {
|
|||||||
overlay.querySelector('#btn-preview-close')?.addEventListener('click', () => overlay.remove());
|
overlay.querySelector('#btn-preview-close')?.addEventListener('click', () => overlay.remove());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private bindUserAutocomplete() {
|
||||||
|
const input = document.getElementById('hw-user_current') as HTMLInputElement;
|
||||||
|
const list = document.getElementById('hw-user-current-list') as HTMLDivElement;
|
||||||
|
const deptSelect = document.getElementById('hw-current_dept') as HTMLSelectElement;
|
||||||
|
const positionInput = document.getElementById('hw-user_position') as HTMLInputElement;
|
||||||
|
const empNoInput = document.getElementById('hw-emp_no') as HTMLInputElement;
|
||||||
|
|
||||||
|
if (!input || !list) return;
|
||||||
|
|
||||||
|
const showList = (filterText: string = '') => {
|
||||||
|
if (!this.isEditMode) return;
|
||||||
|
const category = (document.getElementById('hw-category') as HTMLSelectElement)?.value || '';
|
||||||
|
if (category === 'PC') return;
|
||||||
|
const users = state.masterData.users || [];
|
||||||
|
const query = filterText.trim().toLowerCase();
|
||||||
|
|
||||||
|
const filtered = query
|
||||||
|
? users.filter((u: any) =>
|
||||||
|
u.user_name.toLowerCase().includes(query) ||
|
||||||
|
(u.dept_name && u.dept_name.toLowerCase().includes(query)) ||
|
||||||
|
(u.emp_no && u.emp_no.toLowerCase().includes(query))
|
||||||
|
)
|
||||||
|
: users;
|
||||||
|
|
||||||
|
if (filtered.length === 0) {
|
||||||
|
list.innerHTML = '<div class="autocomplete-item" style="color: #94a3b8; cursor: default;">일치하는 사원 없음</div>';
|
||||||
|
} else {
|
||||||
|
const seen = new Set();
|
||||||
|
const uniqueFiltered = filtered.filter((u: any) => {
|
||||||
|
const key = `${u.user_name}-${u.dept_name}-${u.emp_no}`;
|
||||||
|
if (seen.has(key)) return false;
|
||||||
|
seen.add(key);
|
||||||
|
return true;
|
||||||
|
}).slice(0, 15);
|
||||||
|
|
||||||
|
list.innerHTML = uniqueFiltered.map((u: any) => `
|
||||||
|
<div class="autocomplete-item user-suggestion-item"
|
||||||
|
data-name="${u.user_name}"
|
||||||
|
data-dept="${u.dept_name || ''}"
|
||||||
|
data-pos="${u.position || ''}"
|
||||||
|
data-emp="${u.emp_no || ''}">
|
||||||
|
<div style="font-weight: 600; color: #1e293b;">${u.user_name}</div>
|
||||||
|
<div style="font-size: 0.75rem; color: #64748b; margin-top: 2px;">
|
||||||
|
${u.dept_name || '부서 없음'} / 사번: ${u.emp_no || '-'} / ${u.position || '직급 없음'}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`).join('');
|
||||||
|
}
|
||||||
|
list.classList.remove('hidden');
|
||||||
|
};
|
||||||
|
|
||||||
|
input.addEventListener('focus', () => showList(input.value));
|
||||||
|
input.addEventListener('input', () => showList(input.value));
|
||||||
|
|
||||||
|
list.addEventListener('mousedown', (e) => {
|
||||||
|
const item = (e.target as HTMLElement).closest('.user-suggestion-item');
|
||||||
|
if (item) {
|
||||||
|
const name = item.getAttribute('data-name') || '';
|
||||||
|
const dept = item.getAttribute('data-dept') || '';
|
||||||
|
const pos = item.getAttribute('data-pos') || '';
|
||||||
|
const emp = item.getAttribute('data-emp') || '';
|
||||||
|
|
||||||
|
input.value = name;
|
||||||
|
if (positionInput) positionInput.value = pos;
|
||||||
|
if (empNoInput) empNoInput.value = emp;
|
||||||
|
|
||||||
|
if (deptSelect && dept) {
|
||||||
|
for (let i = 0; i < deptSelect.options.length; i++) {
|
||||||
|
if (deptSelect.options[i].value === dept) {
|
||||||
|
deptSelect.selectedIndex = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
list.classList.add('hidden');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
document.addEventListener('mousedown', (e) => {
|
||||||
|
if (e.target !== input && !list.contains(e.target as Node)) {
|
||||||
|
list.classList.add('hidden');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
private renderHistory(assetId: string) {
|
private renderHistory(assetId: string) {
|
||||||
const container = document.getElementById('hw-history-list');
|
const container = document.getElementById('hw-history-list');
|
||||||
if (!container) return;
|
if (!container) return;
|
||||||
const logs = (state.masterData.logs || []).filter(l => l.asset_id === assetId);
|
const logs = (state.masterData.logs || []).filter(l => l.asset_id === assetId);
|
||||||
if (logs.length === 0) { container.innerHTML = '<div class="empty-history">기록된 변동 이력이 없습니다.</div>'; return; }
|
if (logs.length === 0) { container.innerHTML = '<div class="empty-history">기록된 변동 이력이 없습니다.</div>'; return; }
|
||||||
container.innerHTML = logs.map(l => `<div class="history-item"><div class="history-date">${l.log_date || ''}</div><div class="history-user">${l.log_user || '시스템'}</div><div class="history-details">${l.details}</div></div>`).join('');
|
|
||||||
|
const createdDate = this.currentAsset?.created_at ? this.currentAsset.created_at.substring(0, 10) : '';
|
||||||
|
|
||||||
|
const grouped: Record<string, typeof logs> = {};
|
||||||
|
logs.forEach(l => {
|
||||||
|
const date = l.log_date || '날짜 미지정';
|
||||||
|
if (!grouped[date]) grouped[date] = [];
|
||||||
|
grouped[date].push(l);
|
||||||
|
});
|
||||||
|
|
||||||
|
container.innerHTML = Object.entries(grouped).map(([date, dateLogs]) => {
|
||||||
|
const entriesHtml = dateLogs.map((l, idx) => {
|
||||||
|
const isLast = idx === dateLogs.length - 1;
|
||||||
|
const borderStyle = isLast ? '' : 'border-bottom: 1px dashed var(--hairline); padding-bottom: 8px; margin-bottom: 8px;';
|
||||||
|
|
||||||
|
let displayDetails = l.details;
|
||||||
|
if (l.details && l.details.trim().startsWith('{')) {
|
||||||
|
try {
|
||||||
|
const data = JSON.parse(l.details);
|
||||||
|
if (data.type === 'checkout') {
|
||||||
|
displayDetails = `[불출] ${data.user || ''} (${data.dept || ''}) ${data.memo ? `| 메모: ${data.memo}` : ''}`;
|
||||||
|
} else if (data.type === 'return') {
|
||||||
|
displayDetails = `[반납] ${data.user || ''} (${data.dept || ''}) ${data.memo ? `| 메모: ${data.memo}` : ''}`;
|
||||||
|
} else if (data.type === 'move') {
|
||||||
|
displayDetails = `[이동] ${data.user || ''} (${data.dept || ''}) ➔ ${data.targetUser || ''} (${data.targetDept || ''}) ${data.memo ? `| 메모: ${data.memo}` : ''}`;
|
||||||
|
}
|
||||||
|
} catch (e) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
return `
|
||||||
|
<div class="history-entry" style="${borderStyle}">
|
||||||
|
<div style="font-weight: 600; color: var(--primary); opacity: 0.8; margin-bottom: 4px; display: flex; align-items: center; gap: 6px;">
|
||||||
|
<span style="display: inline-block; width: 4px; height: 4px; background-color: var(--primary); border-radius: 50%;"></span>
|
||||||
|
${l.log_user || '시스템'}
|
||||||
|
</div>
|
||||||
|
<div style="color: var(--primary); padding-left: 10px; line-height: 1.5;">${displayDetails}</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}).join('');
|
||||||
|
|
||||||
|
const isInitialReg = date === createdDate;
|
||||||
|
const regBadge = isInitialReg ? `<span class="badge-reg" style="font-size: 10px; padding: 1px 5px; margin-left: 6px; background-color: rgba(16, 185, 129, 0.1); color: #10b981; border: 1px solid rgba(16, 185, 129, 0.2); border-radius: 4px; font-weight: 600;">최초등록</span>` : '';
|
||||||
|
|
||||||
|
return `
|
||||||
|
<div class="history-item">
|
||||||
|
<div class="history-date" style="display: flex; align-items: center;">${date} ${regBadge}</div>
|
||||||
|
<div class="history-details" style="display: flex; flex-direction: column; gap: 4px;">
|
||||||
|
${entriesHtml}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}).join('');
|
||||||
}
|
}
|
||||||
|
|
||||||
private getCategoryKey(asset: any): string {
|
private getCategoryKey(asset: any): string {
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user