Files
distribution_erp/scripts/test-prod-full-e2e.mjs
T
chpark 6ac6807b1b
Deploy momo-erp / deploy (push) Failing after 11m45s
fix(deploy): DB IP를 121.156.99.3으로 갱신 — 운영 .env.production 자동 반영
- .gitea/workflows/deploy.yml: heredoc DATABASE_URL을 새 DB IP로
- CICD_SETUP.md / e2e 스크립트: 문서·테스트의 DB URL 일괄 갱신
- 이전엔 git push 후에도 deploy.yml의 hardcoded 구IP가 .env.production을
  덮어써서 운영이 옛 DB로 부팅됨 → 본 커밋으로 자동배포 시 신 DB 적용

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-30 13:55:46 +09:00

124 lines
4.7 KiB
JavaScript

// 운영 E2E: 가입(또는 기존 로그인) → 발주 → 관리자 승인 → 메일
// 모든 거래처 비밀번호: test1234 (AES 인코딩 일치)
// 관리자: plm_admin / qlalfqjsgh11 (FITO 마스터)
import pg from "pg";
const BASE = process.env.E2E_BASE || "https://momo.junggomoa.com";
const DB_URL = "postgresql://momo_app:qlalfqjsgh11@121.156.99.3:5432/distribution";
const log = (...a) => console.log("[e2e]", ...a);
const fail = (m) => { console.error("[e2e] ✖", m); process.exit(1); };
const apiClient = (jar) => async (path, init = {}) => {
const res = await fetch(`${BASE}${path}`, {
...init,
headers: {
"Content-Type": "application/json",
...(jar.value ? { Cookie: jar.value } : {}),
...(init.headers ?? {}),
},
redirect: "manual",
});
const sc = res.headers.getSetCookie?.() || [];
for (const c of sc) {
const kv = c.split(";")[0];
const k = kv.split("=")[0];
jar.value = jar.value
? jar.value.split("; ").filter((p) => !p.startsWith(k + "=")).concat(kv).join("; ")
: kv;
}
const text = await res.text();
let json;
try { json = JSON.parse(text); } catch { json = { raw: text.slice(0, 200) }; }
return { status: res.status, body: json };
};
// 1. 헬스체크
log("0. 헬스체크");
const h = await fetch(`${BASE}/`);
log(` ${h.status} ${BASE}/`);
// 2. DB 시드 검증
const db = new pg.Client({ connectionString: DB_URL });
await db.connect();
const itemRows = await db.query(`SELECT objid, item_name FROM momo_items WHERE item_code LIKE 'SAMP-%' ORDER BY item_code LIMIT 5`);
log(` 샘플 품목 ${itemRows.rowCount}개 확인`);
// 3. 거래처 로그인 (chpark@wace.me / test1234)
log("1. 거래처 로그인 — chpark@wace.me");
const userJar = { value: "" };
const userApi = apiClient(userJar);
const login = await userApi("/api/auth/login", {
method: "POST", body: JSON.stringify({ userId: "chpark@wace.me", password: "test1234" }),
});
if (!login.body.success) fail(`거래처 로그인 실패: ${JSON.stringify(login.body)}`);
log(` ✔ 로그인 성공, redirectTo=${login.body.redirectTo}`);
// 4. 품목 조회
log("2. 품목 검색");
const items = await userApi("/api/m/items/list", {
method: "POST", body: JSON.stringify({ keyword: "SAMP" }),
});
const visible = (items.body.RESULTLIST || []).filter((r) => r.ITEM_CODE?.startsWith("SAMP-"));
if (visible.length === 0) fail(`품목 조회 실패: ${JSON.stringify(items.body)}`);
log(`${visible.length}개 품목 노출 (재고 ${visible[0].STOCK_QTY})`);
// 5. 출고 요청
log("3. 출고 요청");
const order = await userApi("/api/m/orders/save", {
method: "POST",
body: JSON.stringify({
lines: [
{ itemObjid: visible[0].OBJID, qty: 5 },
{ itemObjid: visible[1]?.OBJID, qty: 3 },
].filter((l) => l.itemObjid),
memo: "최종 E2E 테스트",
}),
});
if (!order.body.success) fail(`발주 실패: ${JSON.stringify(order.body)}`);
log(` ✔ 발주번호 ${order.body.orderNo}`);
// 6. 관리자 로그인
log("4. 관리자 로그인 (plm_admin)");
const adminJar = { value: "" };
const adminApi = apiClient(adminJar);
const adminLogin = await adminApi("/api/auth/login", {
method: "POST", body: JSON.stringify({ userId: "plm_admin", password: "qlalfqjsgh11" }),
});
if (!adminLogin.body.success) fail(`관리자 로그인 실패: ${JSON.stringify(adminLogin.body)}`);
log(` ✔ 관리자 세션`);
// 7. 출고 관리 목록 조회
log("5. 출고 목록 조회 (관리자)");
const orderList = await adminApi("/api/m/orders/list", {
method: "POST", body: JSON.stringify({ status: "REQUESTED" }),
});
log(` 요청 상태 발주: ${orderList.body.RESULTLIST?.length ?? 0}`);
// 8. 발주 승인 + 메일 발송
log("6. 발주 승인 (메일 자동 발송)");
const approve = await adminApi("/api/m/orders/approve", {
method: "POST", body: JSON.stringify({ objid: order.body.objId }),
});
if (!approve.body.success) fail(`승인 실패: ${JSON.stringify(approve.body)}`);
log(` ✔ 승인 완료. mailSent=${approve.body.mailSent}, error=${approve.body.mailError ?? "(none)"}`);
// 9. 후속 검증
const finalRow = await db.query(
`SELECT order_no, status, total_amount, total_taxfree, total_taxable
FROM momo_orders WHERE objid = $1`,
[order.body.objId]
);
log(` 주문 ${finalRow.rows[0].order_no}: status=${finalRow.rows[0].status}, total=${finalRow.rows[0].total_amount}`);
const mailRow = await db.query(
`SELECT to_email, subject, status FROM momo_mail_logs WHERE ref_objid = $1 ORDER BY regdate DESC LIMIT 1`,
[order.body.objId]
);
if (mailRow.rowCount > 0) {
const m = mailRow.rows[0];
log(` 메일 → ${m.to_email}, status=${m.status}, subject=${m.subject}`);
}
await db.end();
log("✔ E2E 전체 통과");