Compare commits
7 Commits
637b690eda
...
master
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
eebeaf1008 | ||
| ef5c5e63cb | |||
| c4af58d849 | |||
| 61682efb33 | |||
|
|
a820a164cb | ||
|
|
86a6d21a08 | ||
|
|
1299ac261c |
@@ -60,9 +60,15 @@ async def health_check():
|
|||||||
# 전역 예외 처리
|
# 전역 예외 처리
|
||||||
@app.exception_handler(Exception)
|
@app.exception_handler(Exception)
|
||||||
async def global_exception_handler(request: Request, exc: Exception):
|
async def global_exception_handler(request: Request, exc: Exception):
|
||||||
|
# CORS 헤더 추가 (500 에러에서도 CORS 헤더가 필요)
|
||||||
return JSONResponse(
|
return JSONResponse(
|
||||||
status_code=500,
|
status_code=500,
|
||||||
content={"detail": f"Internal server error: {str(exc)}"}
|
content={"detail": f"Internal server error: {str(exc)}"},
|
||||||
|
headers={
|
||||||
|
"Access-Control-Allow-Origin": "*",
|
||||||
|
"Access-Control-Allow-Methods": "GET, POST, PUT, DELETE, OPTIONS",
|
||||||
|
"Access-Control-Allow-Headers": "*"
|
||||||
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|||||||
41
backend/migrate_add_photo_fields.py
Normal file
41
backend/migrate_add_photo_fields.py
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
"""
|
||||||
|
데이터베이스 마이그레이션: 사진 필드 추가
|
||||||
|
- 신고 사진 3, 4, 5 추가
|
||||||
|
- 완료 사진 2, 3, 4, 5 추가
|
||||||
|
"""
|
||||||
|
from sqlalchemy import create_engine, text
|
||||||
|
import os
|
||||||
|
|
||||||
|
# 데이터베이스 URL
|
||||||
|
DATABASE_URL = os.getenv("DATABASE_URL", "postgresql://postgres:postgres@db:5432/issue_tracker")
|
||||||
|
|
||||||
|
def run_migration():
|
||||||
|
engine = create_engine(DATABASE_URL)
|
||||||
|
|
||||||
|
with engine.connect() as conn:
|
||||||
|
print("마이그레이션 시작...")
|
||||||
|
|
||||||
|
try:
|
||||||
|
# 신고 사진 필드 추가
|
||||||
|
print("신고 사진 필드 추가 중...")
|
||||||
|
conn.execute(text("ALTER TABLE issues ADD COLUMN IF NOT EXISTS photo_path3 VARCHAR"))
|
||||||
|
conn.execute(text("ALTER TABLE issues ADD COLUMN IF NOT EXISTS photo_path4 VARCHAR"))
|
||||||
|
conn.execute(text("ALTER TABLE issues ADD COLUMN IF NOT EXISTS photo_path5 VARCHAR"))
|
||||||
|
|
||||||
|
# 완료 사진 필드 추가
|
||||||
|
print("완료 사진 필드 추가 중...")
|
||||||
|
conn.execute(text("ALTER TABLE issues ADD COLUMN IF NOT EXISTS completion_photo_path2 VARCHAR(500)"))
|
||||||
|
conn.execute(text("ALTER TABLE issues ADD COLUMN IF NOT EXISTS completion_photo_path3 VARCHAR(500)"))
|
||||||
|
conn.execute(text("ALTER TABLE issues ADD COLUMN IF NOT EXISTS completion_photo_path4 VARCHAR(500)"))
|
||||||
|
conn.execute(text("ALTER TABLE issues ADD COLUMN IF NOT EXISTS completion_photo_path5 VARCHAR(500)"))
|
||||||
|
|
||||||
|
conn.commit()
|
||||||
|
print("✅ 마이그레이션 완료!")
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
conn.rollback()
|
||||||
|
print(f"❌ 마이그레이션 실패: {e}")
|
||||||
|
raise
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
run_migration()
|
||||||
28
backend/migrations/018_add_deletion_log_table.sql
Normal file
28
backend/migrations/018_add_deletion_log_table.sql
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
-- 삭제 로그 테이블 추가
|
||||||
|
-- 생성일: 2025-11-08
|
||||||
|
-- 설명: 부적합 등 엔티티 삭제 시 로그를 보관하기 위한 테이블
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS deletion_logs (
|
||||||
|
id SERIAL PRIMARY KEY,
|
||||||
|
entity_type VARCHAR(50) NOT NULL,
|
||||||
|
entity_id INTEGER NOT NULL,
|
||||||
|
entity_data JSONB NOT NULL,
|
||||||
|
deleted_by_id INTEGER NOT NULL REFERENCES users(id),
|
||||||
|
deleted_at TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT (NOW() AT TIME ZONE 'Asia/Seoul'),
|
||||||
|
reason TEXT
|
||||||
|
);
|
||||||
|
|
||||||
|
-- 인덱스 추가
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_deletion_logs_entity_type ON deletion_logs(entity_type);
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_deletion_logs_entity_id ON deletion_logs(entity_id);
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_deletion_logs_deleted_by ON deletion_logs(deleted_by_id);
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_deletion_logs_deleted_at ON deletion_logs(deleted_at);
|
||||||
|
|
||||||
|
-- 테이블 코멘트
|
||||||
|
COMMENT ON TABLE deletion_logs IS '엔티티 삭제 로그 - 삭제된 데이터의 백업 및 추적';
|
||||||
|
COMMENT ON COLUMN deletion_logs.entity_type IS '삭제된 엔티티 타입 (issue, project, daily_work 등)';
|
||||||
|
COMMENT ON COLUMN deletion_logs.entity_id IS '삭제된 엔티티의 ID';
|
||||||
|
COMMENT ON COLUMN deletion_logs.entity_data IS '삭제 시점의 엔티티 전체 데이터 (JSON)';
|
||||||
|
COMMENT ON COLUMN deletion_logs.deleted_by_id IS '삭제 실행자 ID';
|
||||||
|
COMMENT ON COLUMN deletion_logs.deleted_at IS '삭제 시각 (KST)';
|
||||||
|
COMMENT ON COLUMN deletion_logs.reason IS '삭제 사유 (선택사항)';
|
||||||
181
backend/migrations/021_add_5_photo_support.sql
Normal file
181
backend/migrations/021_add_5_photo_support.sql
Normal file
@@ -0,0 +1,181 @@
|
|||||||
|
-- 021_add_5_photo_support.sql
|
||||||
|
-- 5장 사진 지원을 위한 추가 컬럼들
|
||||||
|
-- 작성일: 2025-11-08
|
||||||
|
-- 목적: photo_path3, photo_path4, photo_path5 및 completion_photo_path2~5 컬럼 추가
|
||||||
|
|
||||||
|
BEGIN;
|
||||||
|
|
||||||
|
DO $$
|
||||||
|
DECLARE
|
||||||
|
migration_name VARCHAR(255) := '021_add_5_photo_support.sql';
|
||||||
|
migration_notes TEXT := '5장 사진 지원: photo_path3~5, completion_photo_path2~5 컬럼 추가';
|
||||||
|
current_status VARCHAR(50);
|
||||||
|
BEGIN
|
||||||
|
-- migration_log 테이블이 없으면 생성 (멱등성)
|
||||||
|
CREATE TABLE IF NOT EXISTS migration_log (
|
||||||
|
id SERIAL PRIMARY KEY,
|
||||||
|
migration_file VARCHAR(255) NOT NULL UNIQUE,
|
||||||
|
executed_at TIMESTAMP WITH TIME ZONE DEFAULT NOW(),
|
||||||
|
status VARCHAR(50),
|
||||||
|
notes TEXT,
|
||||||
|
created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW(),
|
||||||
|
started_at TIMESTAMP WITH TIME ZONE,
|
||||||
|
completed_at TIMESTAMP WITH TIME ZONE
|
||||||
|
);
|
||||||
|
|
||||||
|
SELECT status INTO current_status FROM migration_log WHERE migration_file = migration_name;
|
||||||
|
|
||||||
|
IF current_status IS NULL THEN
|
||||||
|
RAISE NOTICE '--- 마이그레이션 % 시작 ---', migration_name;
|
||||||
|
|
||||||
|
-- 기본 사진 경로 3~5 추가
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'issues' AND column_name = 'photo_path3') THEN
|
||||||
|
ALTER TABLE issues ADD COLUMN photo_path3 VARCHAR(500);
|
||||||
|
RAISE NOTICE '✅ issues.photo_path3 컬럼이 추가되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ issues.photo_path3 컬럼이 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'issues' AND column_name = 'photo_path4') THEN
|
||||||
|
ALTER TABLE issues ADD COLUMN photo_path4 VARCHAR(500);
|
||||||
|
RAISE NOTICE '✅ issues.photo_path4 컬럼이 추가되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ issues.photo_path4 컬럼이 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'issues' AND column_name = 'photo_path5') THEN
|
||||||
|
ALTER TABLE issues ADD COLUMN photo_path5 VARCHAR(500);
|
||||||
|
RAISE NOTICE '✅ issues.photo_path5 컬럼이 추가되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ issues.photo_path5 컬럼이 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
-- 완료 사진 경로 2~5 추가
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'issues' AND column_name = 'completion_photo_path2') THEN
|
||||||
|
ALTER TABLE issues ADD COLUMN completion_photo_path2 VARCHAR(500);
|
||||||
|
RAISE NOTICE '✅ issues.completion_photo_path2 컬럼이 추가되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ issues.completion_photo_path2 컬럼이 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'issues' AND column_name = 'completion_photo_path3') THEN
|
||||||
|
ALTER TABLE issues ADD COLUMN completion_photo_path3 VARCHAR(500);
|
||||||
|
RAISE NOTICE '✅ issues.completion_photo_path3 컬럼이 추가되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ issues.completion_photo_path3 컬럼이 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'issues' AND column_name = 'completion_photo_path4') THEN
|
||||||
|
ALTER TABLE issues ADD COLUMN completion_photo_path4 VARCHAR(500);
|
||||||
|
RAISE NOTICE '✅ issues.completion_photo_path4 컬럼이 추가되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ issues.completion_photo_path4 컬럼이 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'issues' AND column_name = 'completion_photo_path5') THEN
|
||||||
|
ALTER TABLE issues ADD COLUMN completion_photo_path5 VARCHAR(500);
|
||||||
|
RAISE NOTICE '✅ issues.completion_photo_path5 컬럼이 추가되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ issues.completion_photo_path5 컬럼이 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
-- 추가 필드들 (최신 버전에서 필요한 것들)
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'issues' AND column_name = 'completion_rejected_at') THEN
|
||||||
|
ALTER TABLE issues ADD COLUMN completion_rejected_at TIMESTAMP WITH TIME ZONE;
|
||||||
|
RAISE NOTICE '✅ issues.completion_rejected_at 컬럼이 추가되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ issues.completion_rejected_at 컬럼이 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'issues' AND column_name = 'completion_rejected_by_id') THEN
|
||||||
|
ALTER TABLE issues ADD COLUMN completion_rejected_by_id INTEGER REFERENCES users(id);
|
||||||
|
RAISE NOTICE '✅ issues.completion_rejected_by_id 컬럼이 추가되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ issues.completion_rejected_by_id 컬럼이 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'issues' AND column_name = 'completion_rejection_reason') THEN
|
||||||
|
ALTER TABLE issues ADD COLUMN completion_rejection_reason TEXT;
|
||||||
|
RAISE NOTICE '✅ issues.completion_rejection_reason 컬럼이 추가되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ issues.completion_rejection_reason 컬럼이 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'issues' AND column_name = 'last_exported_at') THEN
|
||||||
|
ALTER TABLE issues ADD COLUMN last_exported_at TIMESTAMP WITH TIME ZONE;
|
||||||
|
RAISE NOTICE '✅ issues.last_exported_at 컬럼이 추가되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ issues.last_exported_at 컬럼이 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'issues' AND column_name = 'export_count') THEN
|
||||||
|
ALTER TABLE issues ADD COLUMN export_count INTEGER DEFAULT 0;
|
||||||
|
RAISE NOTICE '✅ issues.export_count 컬럼이 추가되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ issues.export_count 컬럼이 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
-- 인덱스 추가 (성능 향상)
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM pg_indexes WHERE tablename = 'issues' AND indexname = 'idx_issues_photo_path3') THEN
|
||||||
|
CREATE INDEX idx_issues_photo_path3 ON issues (photo_path3) WHERE photo_path3 IS NOT NULL;
|
||||||
|
RAISE NOTICE '✅ idx_issues_photo_path3 인덱스가 생성되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ idx_issues_photo_path3 인덱스가 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM pg_indexes WHERE tablename = 'issues' AND indexname = 'idx_issues_photo_path4') THEN
|
||||||
|
CREATE INDEX idx_issues_photo_path4 ON issues (photo_path4) WHERE photo_path4 IS NOT NULL;
|
||||||
|
RAISE NOTICE '✅ idx_issues_photo_path4 인덱스가 생성되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ idx_issues_photo_path4 인덱스가 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM pg_indexes WHERE tablename = 'issues' AND indexname = 'idx_issues_photo_path5') THEN
|
||||||
|
CREATE INDEX idx_issues_photo_path5 ON issues (photo_path5) WHERE photo_path5 IS NOT NULL;
|
||||||
|
RAISE NOTICE '✅ idx_issues_photo_path5 인덱스가 생성되었습니다.';
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE 'ℹ️ idx_issues_photo_path5 인덱스가 이미 존재합니다.';
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
-- 마이그레이션 검증
|
||||||
|
DECLARE
|
||||||
|
col_count INTEGER;
|
||||||
|
idx_count INTEGER;
|
||||||
|
BEGIN
|
||||||
|
SELECT COUNT(*) INTO col_count FROM information_schema.columns
|
||||||
|
WHERE table_name = 'issues' AND column_name IN (
|
||||||
|
'photo_path3', 'photo_path4', 'photo_path5',
|
||||||
|
'completion_photo_path2', 'completion_photo_path3', 'completion_photo_path4', 'completion_photo_path5',
|
||||||
|
'completion_rejected_at', 'completion_rejected_by_id', 'completion_rejection_reason',
|
||||||
|
'last_exported_at', 'export_count'
|
||||||
|
);
|
||||||
|
|
||||||
|
SELECT COUNT(*) INTO idx_count FROM pg_indexes
|
||||||
|
WHERE tablename = 'issues' AND indexname IN (
|
||||||
|
'idx_issues_photo_path3', 'idx_issues_photo_path4', 'idx_issues_photo_path5'
|
||||||
|
);
|
||||||
|
|
||||||
|
RAISE NOTICE '=== 마이그레이션 검증 결과 ===';
|
||||||
|
RAISE NOTICE '추가된 컬럼: %/11개', col_count;
|
||||||
|
RAISE NOTICE '생성된 인덱스: %/3개', idx_count;
|
||||||
|
|
||||||
|
IF col_count = 11 AND idx_count = 3 THEN
|
||||||
|
RAISE NOTICE '✅ 마이그레이션이 성공적으로 완료되었습니다!';
|
||||||
|
INSERT INTO migration_log (migration_file, status, notes, completed_at) VALUES (migration_name, 'SUCCESS', migration_notes, NOW());
|
||||||
|
ELSE
|
||||||
|
RAISE EXCEPTION '❌ 마이그레이션 검증 실패!';
|
||||||
|
END IF;
|
||||||
|
END;
|
||||||
|
|
||||||
|
ELSIF current_status = 'SUCCESS' THEN
|
||||||
|
RAISE NOTICE 'ℹ️ 마이그레이션 %는 이미 성공적으로 실행되었습니다. 스킵합니다.', migration_name;
|
||||||
|
ELSE
|
||||||
|
RAISE NOTICE '⚠️ 마이그레이션 %는 이전에 실패했습니다. 수동 확인이 필요합니다.', migration_name;
|
||||||
|
END IF;
|
||||||
|
END $$;
|
||||||
|
|
||||||
|
COMMIT;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -145,10 +145,6 @@ async def preview_daily_report(
|
|||||||
):
|
):
|
||||||
"""일일보고서 미리보기 - 추출될 항목 목록"""
|
"""일일보고서 미리보기 - 추출될 항목 목록"""
|
||||||
|
|
||||||
# 권한 확인
|
|
||||||
if current_user.role != UserRole.admin:
|
|
||||||
raise HTTPException(status_code=403, detail="품질팀만 접근 가능합니다")
|
|
||||||
|
|
||||||
# 프로젝트 확인
|
# 프로젝트 확인
|
||||||
project = db.query(Project).filter(Project.id == project_id).first()
|
project = db.query(Project).filter(Project.id == project_id).first()
|
||||||
if not project:
|
if not project:
|
||||||
@@ -169,7 +165,7 @@ async def preview_daily_report(
|
|||||||
issues = issues_query.all()
|
issues = issues_query.all()
|
||||||
|
|
||||||
# 정렬: 지연 -> 진행중 -> 완료됨 순으로, 같은 상태 내에서는 신고일 최신순
|
# 정렬: 지연 -> 진행중 -> 완료됨 순으로, 같은 상태 내에서는 신고일 최신순
|
||||||
issues = sorted(issues, key=lambda x: (get_issue_priority(x), -x.report_date.timestamp() if x.report_date else 0))
|
issues = sorted(issues, key=lambda x: (get_issue_priority(x), -get_timestamp(x.report_date)))
|
||||||
|
|
||||||
# 통계 계산
|
# 통계 계산
|
||||||
stats = calculate_project_stats(issues)
|
stats = calculate_project_stats(issues)
|
||||||
@@ -190,11 +186,7 @@ async def export_daily_report(
|
|||||||
current_user: User = Depends(get_current_user),
|
current_user: User = Depends(get_current_user),
|
||||||
db: Session = Depends(get_db)
|
db: Session = Depends(get_db)
|
||||||
):
|
):
|
||||||
"""품질팀용 일일보고서 엑셀 내보내기"""
|
"""일일보고서 엑셀 내보내기"""
|
||||||
|
|
||||||
# 권한 확인 (품질팀만 접근 가능)
|
|
||||||
if current_user.role != UserRole.admin:
|
|
||||||
raise HTTPException(status_code=403, detail="품질팀만 접근 가능합니다")
|
|
||||||
|
|
||||||
# 프로젝트 확인
|
# 프로젝트 확인
|
||||||
project = db.query(Project).filter(Project.id == request.project_id).first()
|
project = db.query(Project).filter(Project.id == request.project_id).first()
|
||||||
@@ -223,8 +215,18 @@ async def export_daily_report(
|
|||||||
not_exported_after_completion.append(issue)
|
not_exported_after_completion.append(issue)
|
||||||
elif issue.actual_completion_date:
|
elif issue.actual_completion_date:
|
||||||
# actual_completion_date가 있는 경우: 완료일과 마지막 추출일 비교
|
# actual_completion_date가 있는 경우: 완료일과 마지막 추출일 비교
|
||||||
completion_date = issue.actual_completion_date.replace(tzinfo=None) if issue.actual_completion_date.tzinfo else issue.actual_completion_date
|
# actual_completion_date는 date 또는 datetime일 수 있음
|
||||||
export_date = issue.last_exported_at.replace(tzinfo=None) if issue.last_exported_at.tzinfo else issue.last_exported_at
|
if isinstance(issue.actual_completion_date, datetime):
|
||||||
|
completion_date = issue.actual_completion_date.replace(tzinfo=None) if issue.actual_completion_date.tzinfo else issue.actual_completion_date
|
||||||
|
else:
|
||||||
|
# date 타입인 경우 datetime으로 변환
|
||||||
|
completion_date = datetime.combine(issue.actual_completion_date, datetime.min.time())
|
||||||
|
|
||||||
|
if isinstance(issue.last_exported_at, datetime):
|
||||||
|
export_date = issue.last_exported_at.replace(tzinfo=None) if issue.last_exported_at.tzinfo else issue.last_exported_at
|
||||||
|
else:
|
||||||
|
export_date = datetime.combine(issue.last_exported_at, datetime.min.time())
|
||||||
|
|
||||||
if completion_date > export_date:
|
if completion_date > export_date:
|
||||||
# 완료일이 마지막 추출일보다 나중 -> 완료 후 아직 추출 안됨 -> 진행 중 시트에 표시
|
# 완료일이 마지막 추출일보다 나중 -> 완료 후 아직 추출 안됨 -> 진행 중 시트에 표시
|
||||||
not_exported_after_completion.append(issue)
|
not_exported_after_completion.append(issue)
|
||||||
@@ -236,14 +238,14 @@ async def export_daily_report(
|
|||||||
in_progress_issues = in_progress_only + not_exported_after_completion
|
in_progress_issues = in_progress_only + not_exported_after_completion
|
||||||
|
|
||||||
# 진행 중 시트 정렬: 지연중 -> 진행중 -> 완료됨 순서
|
# 진행 중 시트 정렬: 지연중 -> 진행중 -> 완료됨 순서
|
||||||
in_progress_issues = sorted(in_progress_issues, key=lambda x: (get_issue_priority(x), -x.report_date.timestamp() if x.report_date else 0))
|
in_progress_issues = sorted(in_progress_issues, key=lambda x: (get_issue_priority(x), -get_timestamp(x.report_date)))
|
||||||
|
|
||||||
# "완료됨" 시트용: 완료 항목 중 "완료 후 추출된 것"만 (진행 중 시트에 표시되는 것 제외)
|
# "완료됨" 시트용: 완료 항목 중 "완료 후 추출된 것"만 (진행 중 시트에 표시되는 것 제외)
|
||||||
not_exported_ids = {issue.id for issue in not_exported_after_completion}
|
not_exported_ids = {issue.id for issue in not_exported_after_completion}
|
||||||
completed_issues = [issue for issue in all_completed if issue.id not in not_exported_ids]
|
completed_issues = [issue for issue in all_completed if issue.id not in not_exported_ids]
|
||||||
|
|
||||||
# 완료됨 시트도 정렬 (완료일 최신순)
|
# 완료됨 시트도 정렬 (완료일 최신순)
|
||||||
completed_issues = sorted(completed_issues, key=lambda x: -x.actual_completion_date.timestamp() if x.actual_completion_date else 0)
|
completed_issues = sorted(completed_issues, key=lambda x: -get_timestamp(x.actual_completion_date))
|
||||||
|
|
||||||
# 웹과 동일한 로직: 진행중 + 완료를 함께 정렬하여 순번 할당
|
# 웹과 동일한 로직: 진행중 + 완료를 함께 정렬하여 순번 할당
|
||||||
# (웹에서는 in_progress와 completed를 함께 가져와서 전체를 reviewed_at 순으로 정렬 후 순번 매김)
|
# (웹에서는 in_progress와 completed를 함께 가져와서 전체를 reviewed_at 순으로 정렬 후 순번 매김)
|
||||||
@@ -871,3 +873,12 @@ def get_issue_status_header_color(issue: Issue) -> str:
|
|||||||
elif priority == 3: # 완료
|
elif priority == 3: # 완료
|
||||||
return "92D050" # 진한 초록색
|
return "92D050" # 진한 초록색
|
||||||
return "4472C4" # 기본 파란색
|
return "4472C4" # 기본 파란색
|
||||||
|
|
||||||
|
def get_timestamp(dt) -> float:
|
||||||
|
"""date 또는 datetime 객체에서 timestamp 반환"""
|
||||||
|
if dt is None:
|
||||||
|
return 0
|
||||||
|
if isinstance(dt, datetime):
|
||||||
|
return dt.timestamp()
|
||||||
|
# date 타입인 경우 datetime으로 변환
|
||||||
|
return datetime.combine(dt, datetime.min.time()).timestamp()
|
||||||
|
|||||||
87
backup_script.sh
Executable file
87
backup_script.sh
Executable file
@@ -0,0 +1,87 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# M 프로젝트 자동 백업 스크립트
|
||||||
|
# 사용법: ./backup_script.sh
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
# 백업 디렉토리 설정
|
||||||
|
BACKUP_DIR="/Users/hyungi/M-Project/backups"
|
||||||
|
DATE=$(date +%Y%m%d_%H%M%S)
|
||||||
|
BACKUP_FOLDER="$BACKUP_DIR/$DATE"
|
||||||
|
|
||||||
|
echo "🚀 M 프로젝트 백업 시작: $DATE"
|
||||||
|
|
||||||
|
# 백업 폴더 생성
|
||||||
|
mkdir -p "$BACKUP_FOLDER"
|
||||||
|
|
||||||
|
# 1. 데이터베이스 백업 (가장 중요!)
|
||||||
|
echo "📊 데이터베이스 백업 중..."
|
||||||
|
docker exec m-project-db pg_dump -U mproject mproject > "$BACKUP_FOLDER/database_backup.sql"
|
||||||
|
echo "✅ 데이터베이스 백업 완료"
|
||||||
|
|
||||||
|
# 2. Docker 볼륨 백업
|
||||||
|
echo "💾 Docker 볼륨 백업 중..."
|
||||||
|
docker run --rm -v m-project_postgres_data:/data -v "$BACKUP_FOLDER":/backup alpine tar czf /backup/postgres_volume.tar.gz -C /data .
|
||||||
|
docker run --rm -v m-project_uploads:/data -v "$BACKUP_FOLDER":/backup alpine tar czf /backup/uploads_volume.tar.gz -C /data .
|
||||||
|
echo "✅ Docker 볼륨 백업 완료"
|
||||||
|
|
||||||
|
# 3. 설정 파일 백업
|
||||||
|
echo "⚙️ 설정 파일 백업 중..."
|
||||||
|
cp docker-compose.yml "$BACKUP_FOLDER/"
|
||||||
|
cp -r nginx/ "$BACKUP_FOLDER/"
|
||||||
|
cp -r backend/migrations/ "$BACKUP_FOLDER/"
|
||||||
|
echo "✅ 설정 파일 백업 완료"
|
||||||
|
|
||||||
|
# 4. 백업 정보 기록
|
||||||
|
echo "📝 백업 정보 기록 중..."
|
||||||
|
cat > "$BACKUP_FOLDER/backup_info.txt" << EOF
|
||||||
|
M 프로젝트 백업 정보
|
||||||
|
===================
|
||||||
|
백업 일시: $(date)
|
||||||
|
백업 타입: 전체 백업
|
||||||
|
백업 위치: $BACKUP_FOLDER
|
||||||
|
|
||||||
|
포함된 내용:
|
||||||
|
- database_backup.sql: PostgreSQL 데이터베이스 덤프
|
||||||
|
- postgres_volume.tar.gz: PostgreSQL 데이터 볼륨
|
||||||
|
- uploads_volume.tar.gz: 업로드 파일 볼륨
|
||||||
|
- docker-compose.yml: Docker 설정
|
||||||
|
- nginx/: Nginx 설정
|
||||||
|
- migrations/: 데이터베이스 마이그레이션 파일
|
||||||
|
|
||||||
|
복구 방법:
|
||||||
|
1. ./restore_script.sh $(pwd)
|
||||||
|
|
||||||
|
또는 수동 복구:
|
||||||
|
1. docker-compose down
|
||||||
|
2. docker volume rm m-project_postgres_data m-project_uploads
|
||||||
|
3. docker-compose up -d db
|
||||||
|
4. docker exec -i m-project-db psql -U mproject mproject < database_backup.sql
|
||||||
|
5. docker-compose up -d
|
||||||
|
|
||||||
|
백업 정책:
|
||||||
|
- 최신 10개 백업만 유지 (용량 절약)
|
||||||
|
- 매일 오후 9시 자동 백업
|
||||||
|
- 매주 일요일 오후 9시 30분 추가 백업
|
||||||
|
EOF
|
||||||
|
|
||||||
|
# 5. 백업 크기 확인
|
||||||
|
BACKUP_SIZE=$(du -sh "$BACKUP_FOLDER" | cut -f1)
|
||||||
|
echo "📏 백업 크기: $BACKUP_SIZE"
|
||||||
|
|
||||||
|
# 6. 오래된 백업 정리 (최신 10개만 유지)
|
||||||
|
echo "🧹 오래된 백업 정리 중..."
|
||||||
|
BACKUP_COUNT=$(find "$BACKUP_DIR" -type d -name "20*" | wc -l)
|
||||||
|
if [ $BACKUP_COUNT -gt 10 ]; then
|
||||||
|
REMOVE_COUNT=$((BACKUP_COUNT - 10))
|
||||||
|
echo "📊 현재 백업 개수: $BACKUP_COUNT개, 삭제할 개수: $REMOVE_COUNT개"
|
||||||
|
find "$BACKUP_DIR" -type d -name "20*" | sort | head -n $REMOVE_COUNT | xargs rm -rf
|
||||||
|
echo "✅ 오래된 백업 $REMOVE_COUNT개 삭제 완료"
|
||||||
|
else
|
||||||
|
echo "📊 현재 백업 개수: $BACKUP_COUNT개 (정리 불필요)"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "🎉 백업 완료!"
|
||||||
|
echo "📁 백업 위치: $BACKUP_FOLDER"
|
||||||
|
echo "📏 백업 크기: $BACKUP_SIZE"
|
||||||
106
deploy/deploy.sh
Executable file
106
deploy/deploy.sh
Executable file
@@ -0,0 +1,106 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# =============================================================================
|
||||||
|
# M-Project Synology NAS 배포 스크립트
|
||||||
|
# =============================================================================
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
echo "=========================================="
|
||||||
|
echo "M-Project (부적합관리) 배포 시작"
|
||||||
|
echo "=========================================="
|
||||||
|
|
||||||
|
# 1. 환경 변수 파일 확인
|
||||||
|
if [ ! -f .env ]; then
|
||||||
|
echo "❌ .env 파일이 없습니다."
|
||||||
|
echo " .env.synology 파일을 복사하고 값을 수정하세요:"
|
||||||
|
echo " cp .env.synology .env"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 비밀번호 미설정 확인
|
||||||
|
if grep -q "변경필수" .env; then
|
||||||
|
echo "❌ .env 파일에 기본값이 남아있습니다."
|
||||||
|
echo " 모든 '변경필수' 항목을 실제 값으로 수정하세요."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 2. Docker 이미지 빌드
|
||||||
|
echo ""
|
||||||
|
echo "🔨 Docker 이미지 빌드 중..."
|
||||||
|
docker-compose -f docker-compose.synology.yml build --no-cache
|
||||||
|
|
||||||
|
# 3. 기존 컨테이너 중지
|
||||||
|
echo ""
|
||||||
|
echo "🛑 기존 컨테이너 중지 중..."
|
||||||
|
docker-compose -f docker-compose.synology.yml down 2>/dev/null || true
|
||||||
|
|
||||||
|
# 4. 컨테이너 시작
|
||||||
|
echo ""
|
||||||
|
echo "🚀 컨테이너 시작 중..."
|
||||||
|
docker-compose -f docker-compose.synology.yml up -d
|
||||||
|
|
||||||
|
# 5. DB 초기화 대기
|
||||||
|
echo ""
|
||||||
|
echo "⏳ 데이터베이스 초기화 대기 중 (15초)..."
|
||||||
|
sleep 15
|
||||||
|
|
||||||
|
# 6. DB 마이그레이션 실행
|
||||||
|
echo ""
|
||||||
|
echo "📦 DB 마이그레이션 실행 중..."
|
||||||
|
for sql_file in ./init-db/*.sql; do
|
||||||
|
if [ -f "$sql_file" ]; then
|
||||||
|
echo " 실행: $(basename "$sql_file")"
|
||||||
|
docker exec -i m-project-db psql -U "${POSTGRES_USER:-mproject}" "${POSTGRES_DB:-mproject}" < "$sql_file" 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
# 7. 데이터베이스 복원 (백업 파일이 있는 경우)
|
||||||
|
BACKUP_FILE=$(ls -t backup_*.sql 2>/dev/null | head -1)
|
||||||
|
if [ -n "$BACKUP_FILE" ]; then
|
||||||
|
echo ""
|
||||||
|
read -p "📦 DB 백업 발견: $BACKUP_FILE - 복원하시겠습니까? (y/N) " -n 1 -r
|
||||||
|
echo ""
|
||||||
|
if [[ $REPLY =~ ^[Yy]$ ]]; then
|
||||||
|
echo "📦 데이터베이스 복원 중..."
|
||||||
|
docker exec -i m-project-db psql -U "${POSTGRES_USER:-mproject}" "${POSTGRES_DB:-mproject}" < "$BACKUP_FILE"
|
||||||
|
echo "✅ 데이터베이스 복원 완료"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 8. 상태 확인
|
||||||
|
echo ""
|
||||||
|
echo "=========================================="
|
||||||
|
echo "📊 컨테이너 상태"
|
||||||
|
echo "=========================================="
|
||||||
|
docker-compose -f docker-compose.synology.yml ps
|
||||||
|
|
||||||
|
# 9. 헬스체크
|
||||||
|
echo ""
|
||||||
|
echo "🔍 서비스 확인 중..."
|
||||||
|
sleep 5
|
||||||
|
|
||||||
|
check_service() {
|
||||||
|
local name="$1"
|
||||||
|
local url="$2"
|
||||||
|
printf " %-20s " "$name"
|
||||||
|
status=$(curl -s -o /dev/null -w "%{http_code}" --connect-timeout 3 "$url" 2>/dev/null || echo "000")
|
||||||
|
if [ "$status" -ge 200 ] && [ "$status" -lt 400 ]; then
|
||||||
|
echo "✅ OK ($status)"
|
||||||
|
else
|
||||||
|
echo "❌ FAIL ($status)"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
check_service "Backend API" "http://localhost:16000/api/health"
|
||||||
|
check_service "Frontend" "http://localhost:16080/"
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "=========================================="
|
||||||
|
echo "✅ 배포 완료!"
|
||||||
|
echo "=========================================="
|
||||||
|
echo ""
|
||||||
|
echo "접속 URL:"
|
||||||
|
echo " - 웹 UI: http://NAS_IP:16080"
|
||||||
|
echo " - API: http://NAS_IP:16000"
|
||||||
|
echo " - DB: NAS_IP:16432 (PostgreSQL)"
|
||||||
|
echo ""
|
||||||
80
deploy/docker-compose.synology.yml
Normal file
80
deploy/docker-compose.synology.yml
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
version: '3.8'
|
||||||
|
|
||||||
|
services:
|
||||||
|
# PostgreSQL 데이터베이스
|
||||||
|
db:
|
||||||
|
image: postgres:15-alpine
|
||||||
|
container_name: m-project-db
|
||||||
|
restart: unless-stopped
|
||||||
|
environment:
|
||||||
|
POSTGRES_USER: ${POSTGRES_USER:-mproject}
|
||||||
|
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
|
||||||
|
POSTGRES_DB: ${POSTGRES_DB:-mproject}
|
||||||
|
TZ: Asia/Seoul
|
||||||
|
PGTZ: Asia/Seoul
|
||||||
|
volumes:
|
||||||
|
- postgres_data:/var/lib/postgresql/data
|
||||||
|
- ./init-db:/docker-entrypoint-initdb.d
|
||||||
|
ports:
|
||||||
|
- "16432:5432"
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD-SHELL", "pg_isready -U ${POSTGRES_USER:-mproject}"]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 5s
|
||||||
|
retries: 5
|
||||||
|
networks:
|
||||||
|
- m-project-network
|
||||||
|
|
||||||
|
# FastAPI 백엔드
|
||||||
|
backend:
|
||||||
|
build: ./backend
|
||||||
|
container_name: m-project-backend
|
||||||
|
restart: unless-stopped
|
||||||
|
environment:
|
||||||
|
DATABASE_URL: postgresql://${POSTGRES_USER:-mproject}:${POSTGRES_PASSWORD}@db:5432/${POSTGRES_DB:-mproject}
|
||||||
|
SECRET_KEY: ${SECRET_KEY}
|
||||||
|
ALGORITHM: HS256
|
||||||
|
ACCESS_TOKEN_EXPIRE_MINUTES: 10080
|
||||||
|
ADMIN_USERNAME: ${ADMIN_USERNAME:-hyungi}
|
||||||
|
ADMIN_PASSWORD: ${ADMIN_PASSWORD}
|
||||||
|
TZ: Asia/Seoul
|
||||||
|
volumes:
|
||||||
|
- uploads:/app/uploads
|
||||||
|
ports:
|
||||||
|
- "16000:8000"
|
||||||
|
depends_on:
|
||||||
|
db:
|
||||||
|
condition: service_healthy
|
||||||
|
logging:
|
||||||
|
driver: "json-file"
|
||||||
|
options:
|
||||||
|
max-size: "10m"
|
||||||
|
max-file: "3"
|
||||||
|
networks:
|
||||||
|
- m-project-network
|
||||||
|
|
||||||
|
# Nginx 프론트엔드
|
||||||
|
nginx:
|
||||||
|
build: ./nginx
|
||||||
|
container_name: m-project-nginx
|
||||||
|
restart: unless-stopped
|
||||||
|
ports:
|
||||||
|
- "16080:80"
|
||||||
|
volumes:
|
||||||
|
- ./frontend:/usr/share/nginx/html:ro
|
||||||
|
- uploads:/app/uploads
|
||||||
|
depends_on:
|
||||||
|
- backend
|
||||||
|
networks:
|
||||||
|
- m-project-network
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
postgres_data:
|
||||||
|
driver: local
|
||||||
|
uploads:
|
||||||
|
driver: local
|
||||||
|
|
||||||
|
networks:
|
||||||
|
m-project-network:
|
||||||
|
driver: bridge
|
||||||
|
name: m-project-network
|
||||||
83
deploy/package.sh
Executable file
83
deploy/package.sh
Executable file
@@ -0,0 +1,83 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# =============================================================================
|
||||||
|
# M-Project 배포 패키지 생성 스크립트
|
||||||
|
# =============================================================================
|
||||||
|
|
||||||
|
set -e
|
||||||
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
||||||
|
PROJECT_DIR="$(dirname "$SCRIPT_DIR")"
|
||||||
|
DEPLOY_DIR="$SCRIPT_DIR"
|
||||||
|
PACKAGE_DIR="$DEPLOY_DIR/mproject-package"
|
||||||
|
|
||||||
|
echo "=========================================="
|
||||||
|
echo "M-Project 배포 패키지 생성"
|
||||||
|
echo "=========================================="
|
||||||
|
|
||||||
|
# 기존 패키지 삭제
|
||||||
|
rm -rf "$PACKAGE_DIR"
|
||||||
|
mkdir -p "$PACKAGE_DIR"
|
||||||
|
|
||||||
|
# 1. Docker 설정 파일
|
||||||
|
echo "📦 Docker 설정 복사..."
|
||||||
|
cp "$DEPLOY_DIR/docker-compose.synology.yml" "$PACKAGE_DIR/docker-compose.yml"
|
||||||
|
cp "$DEPLOY_DIR/.env.synology" "$PACKAGE_DIR/.env.example"
|
||||||
|
cp "$DEPLOY_DIR/deploy.sh" "$PACKAGE_DIR/"
|
||||||
|
chmod +x "$PACKAGE_DIR/deploy.sh"
|
||||||
|
|
||||||
|
# 2. 데이터베이스 백업 생성
|
||||||
|
echo "📦 DB 백업 시도..."
|
||||||
|
if docker exec m-project-db pg_dump -U mproject mproject > "$PACKAGE_DIR/backup_$(date +%Y%m%d_%H%M%S).sql" 2>/dev/null; then
|
||||||
|
echo " ✅ DB 백업 완료"
|
||||||
|
else
|
||||||
|
echo " ⚠️ DB 백업 건너뜀 (컨테이너 미실행)"
|
||||||
|
rm -f "$PACKAGE_DIR"/backup_*.sql
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 3. 소스 코드 복사
|
||||||
|
echo "📦 소스 코드 복사..."
|
||||||
|
|
||||||
|
# Backend
|
||||||
|
mkdir -p "$PACKAGE_DIR/backend"
|
||||||
|
rsync -a --exclude='__pycache__' --exclude='.git' --exclude='venv' --exclude='*.pyc' \
|
||||||
|
"$PROJECT_DIR/backend/" "$PACKAGE_DIR/backend/"
|
||||||
|
|
||||||
|
# Frontend
|
||||||
|
mkdir -p "$PACKAGE_DIR/frontend"
|
||||||
|
rsync -a --exclude='.git' --exclude='uploads' \
|
||||||
|
"$PROJECT_DIR/frontend/" "$PACKAGE_DIR/frontend/"
|
||||||
|
|
||||||
|
# Nginx
|
||||||
|
mkdir -p "$PACKAGE_DIR/nginx"
|
||||||
|
rsync -a "$PROJECT_DIR/nginx/" "$PACKAGE_DIR/nginx/"
|
||||||
|
|
||||||
|
# 4. init-db 폴더 (마이그레이션 스크립트)
|
||||||
|
mkdir -p "$PACKAGE_DIR/init-db"
|
||||||
|
if [ -d "$PROJECT_DIR/backend/migrations" ]; then
|
||||||
|
cp "$PROJECT_DIR/backend/migrations"/*.sql "$PACKAGE_DIR/init-db/" 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 5. 압축
|
||||||
|
echo "📦 압축 중..."
|
||||||
|
cd "$DEPLOY_DIR"
|
||||||
|
TIMESTAMP=$(date +%Y%m%d_%H%M%S)
|
||||||
|
tar -czf "mproject-deploy-$TIMESTAMP.tar.gz" -C "$DEPLOY_DIR" mproject-package
|
||||||
|
|
||||||
|
# 크기 확인
|
||||||
|
echo ""
|
||||||
|
echo "=========================================="
|
||||||
|
echo "✅ 패키지 생성 완료!"
|
||||||
|
echo "=========================================="
|
||||||
|
ls -lh "$DEPLOY_DIR/mproject-deploy-$TIMESTAMP.tar.gz"
|
||||||
|
echo ""
|
||||||
|
echo "파일: $DEPLOY_DIR/mproject-deploy-$TIMESTAMP.tar.gz"
|
||||||
|
echo ""
|
||||||
|
echo "Synology NAS로 전송:"
|
||||||
|
echo " scp $DEPLOY_DIR/mproject-deploy-$TIMESTAMP.tar.gz admin@NAS_IP:/volume1/docker/"
|
||||||
|
echo ""
|
||||||
|
echo "NAS에서 설치:"
|
||||||
|
echo " cd /volume1/docker/"
|
||||||
|
echo " tar -xzf mproject-deploy-$TIMESTAMP.tar.gz"
|
||||||
|
echo " cd mproject-package"
|
||||||
|
echo " cp .env.example .env && vi .env"
|
||||||
|
echo " bash deploy.sh"
|
||||||
|
echo ""
|
||||||
@@ -339,30 +339,26 @@
|
|||||||
async function loadProjects() {
|
async function loadProjects() {
|
||||||
try {
|
try {
|
||||||
// API에서 최신 프로젝트 데이터 가져오기
|
// API에서 최신 프로젝트 데이터 가져오기
|
||||||
const apiUrl = window.API_BASE_URL || 'http://localhost:16080/api';
|
const apiUrl = window.API_BASE_URL || (() => {
|
||||||
const response = await fetch(`${apiUrl}/projects/`, {
|
const hostname = window.location.hostname;
|
||||||
headers: {
|
const protocol = window.location.protocol;
|
||||||
'Authorization': `Bearer ${localStorage.getItem('access_token')}`,
|
const port = window.location.port;
|
||||||
'Content-Type': 'application/json'
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (response.ok) {
|
if ((hostname === 'localhost' || hostname === '127.0.0.1') && port) {
|
||||||
projects = await response.json();
|
return `${protocol}//${hostname}:${port}/api`;
|
||||||
console.log('프로젝트 로드 완료:', projects.length, '개');
|
|
||||||
console.log('활성 프로젝트:', projects.filter(p => p.is_active).length, '개');
|
|
||||||
|
|
||||||
// localStorage에도 캐시 저장
|
|
||||||
localStorage.setItem('work-report-projects', JSON.stringify(projects));
|
|
||||||
} else {
|
|
||||||
console.error('프로젝트 로드 실패:', response.status);
|
|
||||||
// 실패 시 localStorage에서 로드
|
|
||||||
const saved = localStorage.getItem('work-report-projects');
|
|
||||||
if (saved) {
|
|
||||||
projects = JSON.parse(saved);
|
|
||||||
console.log('캐시에서 프로젝트 로드:', projects.length, '개');
|
|
||||||
}
|
}
|
||||||
}
|
if (hostname === 'm.hyungi.net') {
|
||||||
|
return 'https://m-api.hyungi.net/api';
|
||||||
|
}
|
||||||
|
return '/api';
|
||||||
|
})();
|
||||||
|
// ProjectsAPI 사용 (모든 프로젝트 로드)
|
||||||
|
projects = await ProjectsAPI.getAll(false);
|
||||||
|
console.log('프로젝트 로드 완료:', projects.length, '개');
|
||||||
|
console.log('활성 프로젝트:', projects.filter(p => p.is_active).length, '개');
|
||||||
|
|
||||||
|
// localStorage에도 캐시 저장
|
||||||
|
localStorage.setItem('work-report-projects', JSON.stringify(projects));
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('프로젝트 로드 오류:', error);
|
console.error('프로젝트 로드 오류:', error);
|
||||||
// 오류 시 localStorage에서 로드
|
// 오류 시 localStorage에서 로드
|
||||||
|
|||||||
@@ -287,7 +287,19 @@
|
|||||||
// 프로젝트 로드
|
// 프로젝트 로드
|
||||||
async function loadProjects() {
|
async function loadProjects() {
|
||||||
try {
|
try {
|
||||||
const apiUrl = window.API_BASE_URL || 'http://localhost:16080/api';
|
const apiUrl = window.API_BASE_URL || (() => {
|
||||||
|
const hostname = window.location.hostname;
|
||||||
|
const protocol = window.location.protocol;
|
||||||
|
const port = window.location.port;
|
||||||
|
|
||||||
|
if ((hostname === 'localhost' || hostname === '127.0.0.1') && port) {
|
||||||
|
return `${protocol}//${hostname}:${port}/api`;
|
||||||
|
}
|
||||||
|
if (hostname === 'm.hyungi.net') {
|
||||||
|
return 'https://m-api.hyungi.net/api';
|
||||||
|
}
|
||||||
|
return '/api';
|
||||||
|
})();
|
||||||
const response = await fetch(`${apiUrl}/projects/`, {
|
const response = await fetch(`${apiUrl}/projects/`, {
|
||||||
headers: {
|
headers: {
|
||||||
'Authorization': `Bearer ${localStorage.getItem('access_token')}`,
|
'Authorization': `Bearer ${localStorage.getItem('access_token')}`,
|
||||||
|
|||||||
@@ -323,20 +323,22 @@
|
|||||||
// 데이터 로드 함수들
|
// 데이터 로드 함수들
|
||||||
async function loadProjects() {
|
async function loadProjects() {
|
||||||
try {
|
try {
|
||||||
const apiUrl = window.API_BASE_URL || 'http://localhost:16080/api';
|
const apiUrl = window.API_BASE_URL || (() => {
|
||||||
const response = await fetch(`${apiUrl}/projects/`, {
|
const hostname = window.location.hostname;
|
||||||
headers: {
|
const protocol = window.location.protocol;
|
||||||
'Authorization': `Bearer ${localStorage.getItem('access_token')}`,
|
const port = window.location.port;
|
||||||
'Content-Type': 'application/json'
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (response.ok) {
|
if ((hostname === 'localhost' || hostname === '127.0.0.1') && port) {
|
||||||
projects = await response.json();
|
return `${protocol}//${hostname}:${port}/api`;
|
||||||
updateProjectFilter();
|
}
|
||||||
} else {
|
if (hostname === 'm.hyungi.net') {
|
||||||
throw new Error('프로젝트 목록을 불러올 수 없습니다.');
|
return 'https://m-api.hyungi.net/api';
|
||||||
}
|
}
|
||||||
|
return '/api';
|
||||||
|
})();
|
||||||
|
// ProjectsAPI 사용
|
||||||
|
projects = await ProjectsAPI.getAll(false);
|
||||||
|
updateProjectFilter();
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('프로젝트 로드 실패:', error);
|
console.error('프로젝트 로드 실패:', error);
|
||||||
}
|
}
|
||||||
@@ -344,21 +346,24 @@
|
|||||||
|
|
||||||
async function loadInProgressIssues() {
|
async function loadInProgressIssues() {
|
||||||
try {
|
try {
|
||||||
const response = await fetch('/api/issues/admin/all', {
|
const apiUrl = window.API_BASE_URL || (() => {
|
||||||
headers: {
|
const hostname = window.location.hostname;
|
||||||
'Authorization': `Bearer ${localStorage.getItem('access_token')}`,
|
const protocol = window.location.protocol;
|
||||||
'Content-Type': 'application/json'
|
const port = window.location.port;
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (response.ok) {
|
if ((hostname === 'localhost' || hostname === '127.0.0.1') && port) {
|
||||||
const allData = await response.json();
|
return `${protocol}//${hostname}:${port}/api`;
|
||||||
// 진행 중 상태만 필터링
|
}
|
||||||
allIssues = allData.filter(issue => issue.review_status === 'in_progress');
|
if (hostname === 'm.hyungi.net') {
|
||||||
filteredIssues = [...allIssues];
|
return 'https://m-api.hyungi.net/api';
|
||||||
} else {
|
}
|
||||||
throw new Error('부적합 목록을 불러올 수 없습니다.');
|
return '/api';
|
||||||
}
|
})();
|
||||||
|
// ManagementAPI 사용하여 관리함 이슈 로드
|
||||||
|
const managementData = await ManagementAPI.getAll();
|
||||||
|
// 진행 중 상태만 필터링
|
||||||
|
allIssues = managementData.filter(issue => issue.review_status === 'in_progress');
|
||||||
|
filteredIssues = [...allIssues];
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('부적합 로드 실패:', error);
|
console.error('부적합 로드 실패:', error);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -668,24 +668,24 @@
|
|||||||
async function loadProjects() {
|
async function loadProjects() {
|
||||||
console.log('🔄 프로젝트 로드 시작');
|
console.log('🔄 프로젝트 로드 시작');
|
||||||
try {
|
try {
|
||||||
const apiUrl = window.API_BASE_URL || 'http://localhost:16080/api';
|
const apiUrl = window.API_BASE_URL || (() => {
|
||||||
const response = await fetch(`${apiUrl}/projects/`, {
|
const hostname = window.location.hostname;
|
||||||
headers: {
|
const protocol = window.location.protocol;
|
||||||
'Authorization': `Bearer ${localStorage.getItem('access_token')}`,
|
const port = window.location.port;
|
||||||
'Content-Type': 'application/json'
|
|
||||||
|
if ((hostname === 'localhost' || hostname === '127.0.0.1') && port) {
|
||||||
|
return `${protocol}//${hostname}:${port}/api`;
|
||||||
}
|
}
|
||||||
});
|
if (hostname === 'm.hyungi.net') {
|
||||||
|
return 'https://m-api.hyungi.net/api';
|
||||||
console.log('📡 프로젝트 API 응답 상태:', response.status);
|
}
|
||||||
|
return '/api';
|
||||||
if (response.ok) {
|
})();
|
||||||
projects = await response.json();
|
// ProjectsAPI 사용 (모든 프로젝트 로드)
|
||||||
console.log('✅ 프로젝트 로드 성공:', projects.length, '개');
|
projects = await ProjectsAPI.getAll(false);
|
||||||
console.log('📋 프로젝트 목록:', projects);
|
console.log('✅ 프로젝트 로드 성공:', projects.length, '개');
|
||||||
updateProjectFilter();
|
console.log('📋 프로젝트 목록:', projects);
|
||||||
} else {
|
updateProjectFilter();
|
||||||
console.error('❌ 프로젝트 API 응답 실패:', response.status, response.statusText);
|
|
||||||
}
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('❌ 프로젝트 로드 실패:', error);
|
console.error('❌ 프로젝트 로드 실패:', error);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -172,13 +172,14 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.collapse-content {
|
.collapse-content {
|
||||||
max-height: 1000px;
|
max-height: none;
|
||||||
overflow: hidden;
|
overflow: visible;
|
||||||
transition: max-height 0.3s ease-out;
|
transition: max-height 0.3s ease-out;
|
||||||
}
|
}
|
||||||
|
|
||||||
.collapse-content.collapsed {
|
.collapse-content.collapsed {
|
||||||
max-height: 0;
|
max-height: 0;
|
||||||
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 진행 중 카드 스타일 */
|
/* 진행 중 카드 스타일 */
|
||||||
@@ -472,66 +473,48 @@
|
|||||||
// 프로젝트 로드
|
// 프로젝트 로드
|
||||||
async function loadProjects() {
|
async function loadProjects() {
|
||||||
try {
|
try {
|
||||||
const apiUrl = window.API_BASE_URL || 'http://localhost:16080/api';
|
// ProjectsAPI 사용 (모든 프로젝트 로드)
|
||||||
const response = await fetch(`${apiUrl}/projects/`, {
|
projects = await ProjectsAPI.getAll(false);
|
||||||
headers: {
|
updateProjectFilter();
|
||||||
'Authorization': `Bearer ${localStorage.getItem('access_token')}`,
|
|
||||||
'Content-Type': 'application/json'
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (response.ok) {
|
|
||||||
projects = await response.json();
|
|
||||||
updateProjectFilter();
|
|
||||||
}
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('프로젝트 로드 실패:', error);
|
console.error('프로젝트 로드 실패:', error);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 부적합 목록 로드 (관리자는 모든 부적합 조회)
|
// 부적합 목록 로드 (관리함 API 사용)
|
||||||
async function loadIssues() {
|
async function loadIssues() {
|
||||||
try {
|
try {
|
||||||
let endpoint = '/api/issues/admin/all';
|
// ManagementAPI 사용
|
||||||
|
const managementIssues = await ManagementAPI.getAll();
|
||||||
|
|
||||||
const response = await fetch(endpoint, {
|
console.log('🔍 관리함 이슈 로드 완료:', managementIssues.length, '개');
|
||||||
headers: {
|
console.log('📊 상태별 분포:', {
|
||||||
'Authorization': `Bearer ${localStorage.getItem('access_token')}`,
|
in_progress: managementIssues.filter(i => i.review_status === 'in_progress').length,
|
||||||
'Content-Type': 'application/json'
|
completed: managementIssues.filter(i => i.review_status === 'completed').length,
|
||||||
}
|
other: managementIssues.filter(i => !['in_progress', 'completed'].includes(i.review_status)).length
|
||||||
});
|
});
|
||||||
|
|
||||||
if (response.ok) {
|
// 수신함에서 넘어온 순서대로 No. 재할당 (reviewed_at 기준)
|
||||||
const allIssues = await response.json();
|
managementIssues.sort((a, b) => new Date(a.reviewed_at) - new Date(b.reviewed_at));
|
||||||
// 관리함에서는 진행 중(in_progress)과 완료됨(completed) 상태만 표시
|
|
||||||
let filteredIssues = allIssues.filter(issue =>
|
|
||||||
issue.review_status === 'in_progress' || issue.review_status === 'completed'
|
|
||||||
);
|
|
||||||
|
|
||||||
// 수신함에서 넘어온 순서대로 No. 재할당 (reviewed_at 기준)
|
// 프로젝트별로 그룹화하여 No. 재할당
|
||||||
filteredIssues.sort((a, b) => new Date(a.reviewed_at) - new Date(b.reviewed_at));
|
const projectGroups = {};
|
||||||
|
managementIssues.forEach(issue => {
|
||||||
|
if (!projectGroups[issue.project_id]) {
|
||||||
|
projectGroups[issue.project_id] = [];
|
||||||
|
}
|
||||||
|
projectGroups[issue.project_id].push(issue);
|
||||||
|
});
|
||||||
|
|
||||||
// 프로젝트별로 그룹화하여 No. 재할당
|
// 각 프로젝트별로 순번 재할당
|
||||||
const projectGroups = {};
|
Object.keys(projectGroups).forEach(projectId => {
|
||||||
filteredIssues.forEach(issue => {
|
projectGroups[projectId].forEach((issue, index) => {
|
||||||
if (!projectGroups[issue.project_id]) {
|
issue.project_sequence_no = index + 1;
|
||||||
projectGroups[issue.project_id] = [];
|
|
||||||
}
|
|
||||||
projectGroups[issue.project_id].push(issue);
|
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
|
||||||
// 각 프로젝트별로 순번 재할당
|
issues = managementIssues;
|
||||||
Object.keys(projectGroups).forEach(projectId => {
|
filterIssues();
|
||||||
projectGroups[projectId].forEach((issue, index) => {
|
|
||||||
issue.project_sequence_no = index + 1;
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
issues = filteredIssues;
|
|
||||||
filterIssues();
|
|
||||||
} else {
|
|
||||||
throw new Error('부적합 목록을 불러올 수 없습니다.');
|
|
||||||
}
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('부적합 로드 실패:', error);
|
console.error('부적합 로드 실패:', error);
|
||||||
alert('부적합 목록을 불러오는데 실패했습니다.');
|
alert('부적합 목록을 불러오는데 실패했습니다.');
|
||||||
@@ -592,9 +575,22 @@
|
|||||||
function filterIssues() {
|
function filterIssues() {
|
||||||
const projectFilter = document.getElementById('projectFilter').value;
|
const projectFilter = document.getElementById('projectFilter').value;
|
||||||
|
|
||||||
|
console.log('🔍 필터링 시작:', {
|
||||||
|
currentTab: currentTab,
|
||||||
|
projectFilter: projectFilter,
|
||||||
|
totalIssues: issues.length
|
||||||
|
});
|
||||||
|
|
||||||
filteredIssues = issues.filter(issue => {
|
filteredIssues = issues.filter(issue => {
|
||||||
// 현재 탭에 따른 상태 필터링
|
// 현재 탭에 따른 상태 필터링
|
||||||
if (issue.review_status !== currentTab) return false;
|
let statusMatch = false;
|
||||||
|
if (currentTab === 'in_progress') {
|
||||||
|
statusMatch = issue.review_status === 'in_progress';
|
||||||
|
} else if (currentTab === 'completed') {
|
||||||
|
statusMatch = issue.review_status === 'completed';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!statusMatch) return false;
|
||||||
|
|
||||||
// 프로젝트 필터링
|
// 프로젝트 필터링
|
||||||
if (projectFilter && issue.project_id != projectFilter) return false;
|
if (projectFilter && issue.project_id != projectFilter) return false;
|
||||||
@@ -602,6 +598,11 @@
|
|||||||
return true;
|
return true;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
console.log('✅ 필터링 결과:', {
|
||||||
|
filteredCount: filteredIssues.length,
|
||||||
|
tab: currentTab
|
||||||
|
});
|
||||||
|
|
||||||
sortIssues();
|
sortIssues();
|
||||||
displayIssues();
|
displayIssues();
|
||||||
updateStatistics(); // 통계 업데이트 추가
|
updateStatistics(); // 통계 업데이트 추가
|
||||||
@@ -658,6 +659,8 @@
|
|||||||
const issues = groupedByDate[date];
|
const issues = groupedByDate[date];
|
||||||
const groupId = `group-${date.replace(/\./g, '-')}`;
|
const groupId = `group-${date.replace(/\./g, '-')}`;
|
||||||
|
|
||||||
|
console.log(`📅 날짜 그룹 [${date}]: ${issues.length}개 이슈`);
|
||||||
|
|
||||||
return `
|
return `
|
||||||
<div class="date-group">
|
<div class="date-group">
|
||||||
<div class="date-header flex items-center justify-between p-3 bg-gray-50 rounded-lg"
|
<div class="date-header flex items-center justify-between p-3 bg-gray-50 rounded-lg"
|
||||||
@@ -682,6 +685,18 @@
|
|||||||
}).join('');
|
}).join('');
|
||||||
|
|
||||||
container.innerHTML = dateGroups;
|
container.innerHTML = dateGroups;
|
||||||
|
|
||||||
|
// 모든 날짜 그룹을 기본적으로 펼쳐진 상태로 초기화
|
||||||
|
Object.keys(groupedByDate).forEach(date => {
|
||||||
|
const groupId = `group-${date.replace(/\./g, '-')}`;
|
||||||
|
const content = document.getElementById(groupId);
|
||||||
|
const icon = document.getElementById(`icon-${groupId}`);
|
||||||
|
|
||||||
|
if (content && icon) {
|
||||||
|
content.classList.remove('collapsed');
|
||||||
|
icon.style.transform = 'rotate(0deg)';
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// 이슈 행 생성 함수
|
// 이슈 행 생성 함수
|
||||||
|
|||||||
@@ -234,7 +234,19 @@
|
|||||||
// 프로젝트 목록 로드
|
// 프로젝트 목록 로드
|
||||||
async function loadProjects() {
|
async function loadProjects() {
|
||||||
try {
|
try {
|
||||||
const apiUrl = window.API_BASE_URL || 'http://localhost:16080/api';
|
const apiUrl = window.API_BASE_URL || (() => {
|
||||||
|
const hostname = window.location.hostname;
|
||||||
|
const protocol = window.location.protocol;
|
||||||
|
const port = window.location.port;
|
||||||
|
|
||||||
|
if ((hostname === 'localhost' || hostname === '127.0.0.1') && port) {
|
||||||
|
return `${protocol}//${hostname}:${port}/api`;
|
||||||
|
}
|
||||||
|
if (hostname === 'm.hyungi.net') {
|
||||||
|
return 'https://m-api.hyungi.net/api';
|
||||||
|
}
|
||||||
|
return '/api';
|
||||||
|
})();
|
||||||
|
|
||||||
const response = await fetch(`${apiUrl}/projects/`, {
|
const response = await fetch(`${apiUrl}/projects/`, {
|
||||||
headers: {
|
headers: {
|
||||||
@@ -302,7 +314,19 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const apiUrl = window.API_BASE_URL || 'http://localhost:16080/api';
|
const apiUrl = window.API_BASE_URL || (() => {
|
||||||
|
const hostname = window.location.hostname;
|
||||||
|
const protocol = window.location.protocol;
|
||||||
|
const port = window.location.port;
|
||||||
|
|
||||||
|
if ((hostname === 'localhost' || hostname === '127.0.0.1') && port) {
|
||||||
|
return `${protocol}//${hostname}:${port}/api`;
|
||||||
|
}
|
||||||
|
if (hostname === 'm.hyungi.net') {
|
||||||
|
return 'https://m-api.hyungi.net/api';
|
||||||
|
}
|
||||||
|
return '/api';
|
||||||
|
})();
|
||||||
const response = await fetch(`${apiUrl}/reports/daily-preview?project_id=${selectedProjectId}`, {
|
const response = await fetch(`${apiUrl}/reports/daily-preview?project_id=${selectedProjectId}`, {
|
||||||
headers: {
|
headers: {
|
||||||
'Authorization': `Bearer ${localStorage.getItem('access_token')}`
|
'Authorization': `Bearer ${localStorage.getItem('access_token')}`
|
||||||
@@ -427,7 +451,19 @@
|
|||||||
button.innerHTML = '<i class="fas fa-spinner fa-spin mr-2"></i>생성 중...';
|
button.innerHTML = '<i class="fas fa-spinner fa-spin mr-2"></i>생성 중...';
|
||||||
button.disabled = true;
|
button.disabled = true;
|
||||||
|
|
||||||
const apiUrl = window.API_BASE_URL || 'http://localhost:16080/api';
|
const apiUrl = window.API_BASE_URL || (() => {
|
||||||
|
const hostname = window.location.hostname;
|
||||||
|
const protocol = window.location.protocol;
|
||||||
|
const port = window.location.port;
|
||||||
|
|
||||||
|
if ((hostname === 'localhost' || hostname === '127.0.0.1') && port) {
|
||||||
|
return `${protocol}//${hostname}:${port}/api`;
|
||||||
|
}
|
||||||
|
if (hostname === 'm.hyungi.net') {
|
||||||
|
return 'https://m-api.hyungi.net/api';
|
||||||
|
}
|
||||||
|
return '/api';
|
||||||
|
})();
|
||||||
const response = await fetch(`${apiUrl}/reports/daily-export`, {
|
const response = await fetch(`${apiUrl}/reports/daily-export`, {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: {
|
headers: {
|
||||||
|
|||||||
@@ -6,8 +6,8 @@ const API_BASE_URL = (() => {
|
|||||||
|
|
||||||
console.log('🔧 API URL 생성 - hostname:', hostname, 'protocol:', protocol, 'port:', port);
|
console.log('🔧 API URL 생성 - hostname:', hostname, 'protocol:', protocol, 'port:', port);
|
||||||
|
|
||||||
// 로컬 환경 (포트 있음)
|
// 로컬 환경 (localhost 또는 127.0.0.1이고 포트 있음)
|
||||||
if (port === '16080') {
|
if ((hostname === 'localhost' || hostname === '127.0.0.1') && port) {
|
||||||
const url = `${protocol}//${hostname}:${port}/api`;
|
const url = `${protocol}//${hostname}:${port}/api`;
|
||||||
console.log('🏠 로컬 환경 URL:', url);
|
console.log('🏠 로컬 환경 URL:', url);
|
||||||
return url;
|
return url;
|
||||||
@@ -256,6 +256,41 @@ const DailyWorkAPI = {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Management API
|
||||||
|
const ManagementAPI = {
|
||||||
|
getAll: () => apiRequest('/management/'),
|
||||||
|
|
||||||
|
update: (issueId, updateData) => apiRequest(`/management/${issueId}`, {
|
||||||
|
method: 'PUT',
|
||||||
|
body: JSON.stringify(updateData)
|
||||||
|
}),
|
||||||
|
|
||||||
|
updateAdditionalInfo: (issueId, additionalInfo) => apiRequest(`/management/${issueId}/additional-info`, {
|
||||||
|
method: 'PUT',
|
||||||
|
body: JSON.stringify(additionalInfo)
|
||||||
|
})
|
||||||
|
};
|
||||||
|
|
||||||
|
// Inbox API
|
||||||
|
const InboxAPI = {
|
||||||
|
getAll: () => apiRequest('/inbox/'),
|
||||||
|
|
||||||
|
review: (issueId, reviewData) => apiRequest(`/inbox/${issueId}/review`, {
|
||||||
|
method: 'PUT',
|
||||||
|
body: JSON.stringify(reviewData)
|
||||||
|
}),
|
||||||
|
|
||||||
|
dispose: (issueId, disposeData) => apiRequest(`/inbox/${issueId}/dispose`, {
|
||||||
|
method: 'PUT',
|
||||||
|
body: JSON.stringify(disposeData)
|
||||||
|
}),
|
||||||
|
|
||||||
|
updateAdditionalInfo: (issueId, additionalInfo) => apiRequest(`/inbox/${issueId}/additional-info`, {
|
||||||
|
method: 'PUT',
|
||||||
|
body: JSON.stringify(additionalInfo)
|
||||||
|
})
|
||||||
|
};
|
||||||
|
|
||||||
// Reports API
|
// Reports API
|
||||||
const ReportsAPI = {
|
const ReportsAPI = {
|
||||||
getSummary: (startDate, endDate) => apiRequest('/reports/summary', {
|
getSummary: (startDate, endDate) => apiRequest('/reports/summary', {
|
||||||
|
|||||||
@@ -46,7 +46,13 @@ class PagePermissionManager {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
// API에서 사용자별 페이지 권한 가져오기
|
// API에서 사용자별 페이지 권한 가져오기
|
||||||
const apiUrl = window.API_BASE_URL || 'http://localhost:16080/api';
|
const apiUrl = window.API_BASE_URL || (() => {
|
||||||
|
const hostname = window.location.hostname;
|
||||||
|
if (hostname === 'm.hyungi.net') {
|
||||||
|
return 'https://m-api.hyungi.net/api';
|
||||||
|
}
|
||||||
|
return '/api';
|
||||||
|
})();
|
||||||
const response = await fetch(`${apiUrl}/users/${this.currentUser.id}/page-permissions`, {
|
const response = await fetch(`${apiUrl}/users/${this.currentUser.id}/page-permissions`, {
|
||||||
headers: {
|
headers: {
|
||||||
'Authorization': `Bearer ${localStorage.getItem('access_token')}`
|
'Authorization': `Bearer ${localStorage.getItem('access_token')}`
|
||||||
@@ -199,7 +205,19 @@ class PagePermissionManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const apiUrl = window.API_BASE_URL || 'http://localhost:16080/api';
|
const apiUrl = window.API_BASE_URL || (() => {
|
||||||
|
const hostname = window.location.hostname;
|
||||||
|
const protocol = window.location.protocol;
|
||||||
|
const port = window.location.port;
|
||||||
|
|
||||||
|
if ((hostname === 'localhost' || hostname === '127.0.0.1') && port) {
|
||||||
|
return `${protocol}//${hostname}:${port}/api`;
|
||||||
|
}
|
||||||
|
if (hostname === 'm.hyungi.net') {
|
||||||
|
return 'https://m-api.hyungi.net/api';
|
||||||
|
}
|
||||||
|
return '/api';
|
||||||
|
})();
|
||||||
const response = await fetch(`${apiUrl}/page-permissions/grant`, {
|
const response = await fetch(`${apiUrl}/page-permissions/grant`, {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: {
|
headers: {
|
||||||
@@ -232,7 +250,19 @@ class PagePermissionManager {
|
|||||||
*/
|
*/
|
||||||
async getUserPagePermissions(userId) {
|
async getUserPagePermissions(userId) {
|
||||||
try {
|
try {
|
||||||
const apiUrl = window.API_BASE_URL || 'http://localhost:16080/api';
|
const apiUrl = window.API_BASE_URL || (() => {
|
||||||
|
const hostname = window.location.hostname;
|
||||||
|
const protocol = window.location.protocol;
|
||||||
|
const port = window.location.port;
|
||||||
|
|
||||||
|
if ((hostname === 'localhost' || hostname === '127.0.0.1') && port) {
|
||||||
|
return `${protocol}//${hostname}:${port}/api`;
|
||||||
|
}
|
||||||
|
if (hostname === 'm.hyungi.net') {
|
||||||
|
return 'https://m-api.hyungi.net/api';
|
||||||
|
}
|
||||||
|
return '/api';
|
||||||
|
})();
|
||||||
const response = await fetch(`${apiUrl}/users/${userId}/page-permissions`, {
|
const response = await fetch(`${apiUrl}/users/${userId}/page-permissions`, {
|
||||||
headers: {
|
headers: {
|
||||||
'Authorization': `Bearer ${localStorage.getItem('access_token')}`
|
'Authorization': `Bearer ${localStorage.getItem('access_token')}`
|
||||||
|
|||||||
105
restore_script.sh
Executable file
105
restore_script.sh
Executable file
@@ -0,0 +1,105 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# M 프로젝트 복구 스크립트
|
||||||
|
# 사용법: ./restore_script.sh /path/to/backup/folder
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
if [ -z "$1" ]; then
|
||||||
|
echo "❌ 사용법: $0 <백업폴더경로>"
|
||||||
|
echo "예시: $0 /Users/hyungi/M-Project/backups/20251108_152538"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
BACKUP_FOLDER="$1"
|
||||||
|
|
||||||
|
if [ ! -d "$BACKUP_FOLDER" ]; then
|
||||||
|
echo "❌ 백업 폴더가 존재하지 않습니다: $BACKUP_FOLDER"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "🔄 M 프로젝트 복구 시작"
|
||||||
|
echo "📁 백업 폴더: $BACKUP_FOLDER"
|
||||||
|
|
||||||
|
# 백업 정보 확인
|
||||||
|
if [ -f "$BACKUP_FOLDER/backup_info.txt" ]; then
|
||||||
|
echo "📋 백업 정보:"
|
||||||
|
cat "$BACKUP_FOLDER/backup_info.txt"
|
||||||
|
echo ""
|
||||||
|
fi
|
||||||
|
|
||||||
|
read -p "⚠️ 기존 데이터가 모두 삭제됩니다. 계속하시겠습니까? (y/N): " -n 1 -r
|
||||||
|
echo
|
||||||
|
if [[ ! $REPLY =~ ^[Yy]$ ]]; then
|
||||||
|
echo "❌ 복구가 취소되었습니다."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 1. 서비스 중지
|
||||||
|
echo "🛑 서비스 중지 중..."
|
||||||
|
cd /Users/hyungi/M-Project
|
||||||
|
docker-compose down
|
||||||
|
|
||||||
|
# 2. 기존 볼륨 삭제
|
||||||
|
echo "🗑️ 기존 볼륨 삭제 중..."
|
||||||
|
docker volume rm m-project_postgres_data m-project_uploads 2>/dev/null || true
|
||||||
|
|
||||||
|
# 3. 데이터베이스 컨테이너만 시작
|
||||||
|
echo "🚀 데이터베이스 컨테이너 시작 중..."
|
||||||
|
docker-compose up -d db
|
||||||
|
|
||||||
|
# 데이터베이스 준비 대기
|
||||||
|
echo "⏳ 데이터베이스 준비 대기 중..."
|
||||||
|
sleep 10
|
||||||
|
|
||||||
|
# 4. 데이터베이스 복구
|
||||||
|
if [ -f "$BACKUP_FOLDER/database_backup.sql" ]; then
|
||||||
|
echo "📊 데이터베이스 복구 중..."
|
||||||
|
docker exec -i m-project-db psql -U mproject mproject < "$BACKUP_FOLDER/database_backup.sql"
|
||||||
|
echo "✅ 데이터베이스 복구 완료"
|
||||||
|
else
|
||||||
|
echo "❌ 데이터베이스 백업 파일을 찾을 수 없습니다."
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 5. Docker 볼륨 복구
|
||||||
|
if [ -f "$BACKUP_FOLDER/postgres_volume.tar.gz" ]; then
|
||||||
|
echo "💾 PostgreSQL 볼륨 복구 중..."
|
||||||
|
docker run --rm -v m-project_postgres_data:/data -v "$BACKUP_FOLDER":/backup alpine tar xzf /backup/postgres_volume.tar.gz -C /data
|
||||||
|
echo "✅ PostgreSQL 볼륨 복구 완료"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -f "$BACKUP_FOLDER/uploads_volume.tar.gz" ]; then
|
||||||
|
echo "📁 업로드 볼륨 복구 중..."
|
||||||
|
docker run --rm -v m-project_uploads:/data -v "$BACKUP_FOLDER":/backup alpine tar xzf /backup/uploads_volume.tar.gz -C /data
|
||||||
|
echo "✅ 업로드 볼륨 복구 완료"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 6. 설정 파일 복구
|
||||||
|
if [ -f "$BACKUP_FOLDER/docker-compose.yml" ]; then
|
||||||
|
echo "⚙️ 설정 파일 복구 중..."
|
||||||
|
cp "$BACKUP_FOLDER/docker-compose.yml" ./
|
||||||
|
if [ -d "$BACKUP_FOLDER/nginx" ]; then
|
||||||
|
cp -r "$BACKUP_FOLDER/nginx/" ./
|
||||||
|
fi
|
||||||
|
if [ -d "$BACKUP_FOLDER/migrations" ]; then
|
||||||
|
cp -r "$BACKUP_FOLDER/migrations/" ./backend/
|
||||||
|
fi
|
||||||
|
echo "✅ 설정 파일 복구 완료"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 7. 전체 서비스 시작
|
||||||
|
echo "🚀 전체 서비스 시작 중..."
|
||||||
|
docker-compose up -d
|
||||||
|
|
||||||
|
# 8. 서비스 상태 확인
|
||||||
|
echo "⏳ 서비스 시작 대기 중..."
|
||||||
|
sleep 15
|
||||||
|
|
||||||
|
echo "🔍 서비스 상태 확인 중..."
|
||||||
|
docker-compose ps
|
||||||
|
|
||||||
|
echo "🎉 복구 완료!"
|
||||||
|
echo "🌐 프론트엔드: http://localhost:16080"
|
||||||
|
echo "🔗 백엔드 API: http://localhost:16000"
|
||||||
|
echo "📊 데이터베이스: localhost:16432"
|
||||||
|
|
||||||
52
setup_auto_backup.sh
Executable file
52
setup_auto_backup.sh
Executable file
@@ -0,0 +1,52 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# M 프로젝트 자동 백업 설정 스크립트
|
||||||
|
|
||||||
|
echo "🔧 M 프로젝트 자동 백업 설정"
|
||||||
|
|
||||||
|
# 현재 crontab 백업
|
||||||
|
crontab -l > /tmp/current_crontab 2>/dev/null || touch /tmp/current_crontab
|
||||||
|
|
||||||
|
# M 프로젝트 백업 작업이 이미 있는지 확인
|
||||||
|
if grep -q "M-Project backup" /tmp/current_crontab; then
|
||||||
|
echo "⚠️ M 프로젝트 백업 작업이 이미 설정되어 있습니다."
|
||||||
|
echo "기존 설정:"
|
||||||
|
grep "M-Project backup" /tmp/current_crontab
|
||||||
|
read -p "기존 설정을 덮어쓰시겠습니까? (y/N): " -n 1 -r
|
||||||
|
echo
|
||||||
|
if [[ ! $REPLY =~ ^[Yy]$ ]]; then
|
||||||
|
echo "❌ 설정이 취소되었습니다."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
# 기존 M 프로젝트 백업 작업 제거
|
||||||
|
grep -v "M-Project backup" /tmp/current_crontab > /tmp/new_crontab
|
||||||
|
mv /tmp/new_crontab /tmp/current_crontab
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 새로운 백업 작업 추가
|
||||||
|
cat >> /tmp/current_crontab << 'EOF'
|
||||||
|
|
||||||
|
# M-Project backup - 매일 오후 9시에 실행
|
||||||
|
0 21 * * * /Users/hyungi/M-Project/backup_script.sh >> /Users/hyungi/M-Project/backup.log 2>&1
|
||||||
|
|
||||||
|
# M-Project backup - 매주 일요일 오후 9시 30분에 전체 백업 (추가 보안)
|
||||||
|
30 21 * * 0 /Users/hyungi/M-Project/backup_script.sh >> /Users/hyungi/M-Project/backup.log 2>&1
|
||||||
|
EOF
|
||||||
|
|
||||||
|
# 새로운 crontab 적용
|
||||||
|
crontab /tmp/current_crontab
|
||||||
|
|
||||||
|
# 정리
|
||||||
|
rm /tmp/current_crontab
|
||||||
|
|
||||||
|
echo "✅ 자동 백업 설정 완료!"
|
||||||
|
echo ""
|
||||||
|
echo "📅 백업 스케줄:"
|
||||||
|
echo " - 매일 오후 9시: 자동 백업"
|
||||||
|
echo " - 매주 일요일 오후 9시 30분: 추가 백업"
|
||||||
|
echo ""
|
||||||
|
echo "📋 현재 crontab 설정:"
|
||||||
|
crontab -l | grep -A2 -B2 "M-Project"
|
||||||
|
echo ""
|
||||||
|
echo "📄 백업 로그 위치: /Users/hyungi/M-Project/backup.log"
|
||||||
|
echo "📁 백업 저장 위치: /Users/hyungi/M-Project/backups/"
|
||||||
Reference in New Issue
Block a user