Improvement #172
Updated by Chakkaphon Noinang (Jay) 25 days ago
# Bff * [x] 3000 = main (new) พี่ไปร์ท * [ ] 3001 = masterdata * [ ] 3002 = jobpost * [ ] 3003 = candidates * [ ] 3004 = users # Process * [ ] 4003 = candidates * [ ] 4006 = jobapplication * [ ] 4007 = batch # System * [ ] 5001 = masterdata พี่ไปร์ท * [ ] 5002 = jobpost พี่ไปร์ท * [ ] 5003 = candidates พี่เจ * [ ] 5004 = users พี่ไปร์ท * [ ] 5005 = document พี่ไปร์ท * [ ] 5006 = jobapplication พี่เจ * [ ] 5007 = batch พี่เจ * [ ] 5008 = candidate-consumer พี่เจ * [ ] 5009 = complete-candidate-consumer ก้อง * [ ] 5010 = recruit-dlq-consumer ก้อง * [ ] 5011 = job-appointment ก้อง * [ ] 5012 = notification ก้อง ================== ### Dockerfile please backup name (Dockerfile.backup) ### Dockerfile ``` # ---------------------------------------------------- # Stage 1: Build the Application # ---------------------------------------------------- FROM node:24.11.0-alpine3.22 AS builder # Install build dependencies in a single layer RUN apk update && \ apk add --no-cache \ tzdata \ busybox-extras \ openssl \ libssl3 \ libc6-compat && \ cp /usr/share/zoneinfo/Asia/Bangkok /etc/localtime && \ rm -rf /var/cache/apk/* WORKDIR /app # Copy package files for dependency installation COPY package*.json ./ # Install all dependencies (including devDependencies for build) RUN npm install && \ npm cache clean --force # Copy source code COPY . . # Build application RUN npm run build # ---------------------------------------------------- # Stage 2: Production Image (Minimal & Secure) # ---------------------------------------------------- FROM node:24.11.0-alpine3.22 AS production # Install runtime dependencies RUN apk update && \ apk add --no-cache \ tzdata \ openssl \ libssl3 \ libc6-compat \ dumb-init && \ cp /usr/share/zoneinfo/Asia/Bangkok /etc/localtime && \ rm -rf /var/cache/apk/* WORKDIR /app # Set NODE_ENV ARG NODE_ENV=prod ENV NODE_ENV=${NODE_ENV} # Copy package files COPY --from=builder /app/package*.json ./ # Install production dependencies only RUN npm install --omit=dev && \ npm cache clean --force # Copy compiled application COPY --from=builder /app/dist/src ./dist COPY --from=builder /app/dist/config ./config COPY --from=builder /app/env ./env # Set ownership RUN chown -R node:node /app # Use non-root user USER node # Set port configuration ARG APPLICATION_PORT=80 ENV PORT=${APPLICATION_PORT} EXPOSE ${APPLICATION_PORT} # Health check HEALTHCHECK --interval=30s --timeout=3s --start-period=40s --retries=3 \ CMD node -e "require('http').get('http://localhost:' + process.env.PORT + '/health', (r) => {process.exit(r.statusCode === 200 ? 0 : 1)})" # Use dumb-init for proper signal handling CMD ["dumb-init", "node", "dist/main"] ``` # package.json ## เพิ่ม ``` "start:pre": "cross-env NODE_ENV=pre nest start" ``` # config/env.ts ## แก้เป็น ``` export enum ENV { DEV = "dev", UAT = "uat", SIT = "sit", PROD = "prod", PRE = "pre", } ``` # env.dev ## ลบ prefix ให้เป็นตามในตัวอย่าง ``` SYSTEM_JOB_POST_BASE_URL=http://localhost:5002/v1 ``` # env.uat ## แก้ port=80 ## แก้ตาม prefix ให้ตรงตามตัวอย่าง ``` SYSTEM_JOB_POST_BASE_URL=http://hrr-system-jobpost-svc.development.svc.cluster.local ``` # env.sit ## copy env.uat มาเป็นตัวตั้งและเปลี่ยน NODE_ENV=sit # env.pre ## copy env.uat มาเป็นตัวตั้งและเปลี่ยน NODE_ENV=pre # env.prod ## copy env.uat มาเป็นตัวตั้งและเปลี่ยน NODE_ENV=prod # main.ts ## enableCors ตามตัวอย่าง ``` app.enableCors({ app.enableCors({ origin: CORS_ORIGIN, origin: CORS_ORIGIN, credentials: false, credentials: false, methods: ["POST", "OPTIONS"], methods: ["GET", "POST", "OPTIONS"], allowedHeaders: [ allowedHeaders: [ "Content-Type", "Content-Type", "Authorization", "Authorization", "X-Request-ID", "X-Request-ID", "sender", "sender", "refer", "refer", "forward", "forward", "senddate", "senddate", "clientid", "clientid", ], ], exposedHeaders: ["X-Request-ID"], exposedHeaders: ["X-Request-ID"], }); }); ``` # modules/health/controllers/health.controller.ts ## แก้ตามตัวอย่างทั้งไฟล์ ```javascript import { Controller, Get, VERSION_NEUTRAL } from "@nestjs/common"; import { SkipEnvelope } from "@shared/http/skip-envelope.decorator"; @Controller({ path: "health", version: "1" }) @Controller({ version: VERSION_NEUTRAL }) export class HealthController { export class HealthController { @Post("ping") @Get("health") ping(): { status: string } { @SkipEnvelope() health(): { status: string } { return { status: "ok" }; } @Get("alive") @SkipEnvelope() alive(): { status: string } { return { status: "ok" }; return { status: "ok" }; } } } } ```