Files

9.8 KiB
Raw Permalink Blame History

Context

目前 query-tool 僅提供單點查詢 reject 資訊,沒有針對歷史趨勢、原因分布與績效指標的完整頁面。DW_MES_LOTREJECTHISTORY 存在同一 HISTORYMAINLINEID 對應多筆原因紀錄的特性,直接彙總 MOVEINQTY 會造成分母膨脹讓報廢率失真。另一方面現有語意中「reject 五欄合計」與 DEFECTQTY 曾被混用,導致跨頁面解讀不一致。新增可用資料表 ERP_PJ_WIP_SCRAP_REASONS_EXCLUDE 也帶來新政策需求:ENABLE_FLAG='Y' 的報廢原因應預設不納入良率計算,並允許使用者切換是否納入。

此變更需要跨越前端(新報表頁)、後端(新 API + service + SQL、與治理層route contract、drawer/page registry、coverage test屬於跨模組整合型設計。

Goals / Non-Goals

Goals:

  • 建立 /reject-history 專用報表頁提供篩選、KPI、趨勢、Pareto、明細與匯出。
  • 固化兩條指標語義並列:
    • 扣帳報廢 REJECT_TOTAL_QTY
    • 不扣帳報廢 DEFECT_QTY
  • ERP_PJ_WIP_SCRAP_REASONS_EXCLUDEENABLE_FLAG='Y')納入良率排除政策,且提供 UI/API 可切換納入模式。
  • 以事件層級去重 MOVEIN_QTY,避免因 HISTORYMAINLINEID 多筆造成比率失真。
  • 完整納入現有 pure Vite + portal-shell + route contract 治理流程。
  • 在視覺上清楚區分 reject 與 defect避免操作端誤判。

Non-Goals:

  • 不改寫既有 query-tool 與其 API 欄位契約。
  • 不新增第三方套件(沿用 Vue 3、ECharts、Flask、SQLLoader
  • 不在此變更內建立新的資料倉儲實體表(先以查詢層與 service 聚合實作)。
  • 不重構非 reject-history 相關頁面的 UI 風格。

Decisions

D1: 以單一「日粒度基底查詢」作為 API 共同資料來源

Decision:src/mes_dashboard/sql/reject_history/performance_daily.sql 作為基底查詢summary/trend/pareto/list/export 都由 service 在同一語義上二次聚合。

Why:

  • 可避免各 endpoint 重複實作計算邏輯造成語意漂移。
  • 已在 SQL 中明確定義五欄合計與 DEFECTQTY 分離,符合業務規則。
  • 可先解決語義一致性,再視流量評估是否做 materialization。

Alternatives considered:

  • 各 API 各自寫 SQL開發快但長期高風險易出現欄位定義不一致。
  • 先建資料表再查詢:效能可控但前置成本高,超出本次提案節奏。

D2: MOVEIN_QTY 以事件去重策略統一計算

Decision:HISTORYMAINLINEID 為首選去重鍵;若缺值,退回 CONTAINERID + TXNDATE(second) + SPECID 組合鍵;只對事件首筆計入 MOVEIN_QTY

Why:

  • 原始資料存在一對多原因拆分,若不去重分母會被重複計算。
  • 去重規則集中在基底 SQL前後端不用再各自補丁。

Alternatives considered:

  • CONTAINERID 去重:會誤合併同 lot 不同事件。
  • 不去重:報表比率不可用。

D3: API 採「頁面導向」端點設計

Decision: 提供 summarytrendreason-paretolistexport 五個端點,參數模型一致(日期 + 維度過濾),由 route 層統一做驗證與邊界控制。

Why:

  • 前端可平行載入、獨立重刷局部區塊。
  • 可針對高成本端點list/export單獨做 rate limit。
  • hold-history/resource-history 現有模式一致。

D4: 前端視覺採「雙軸語義敘事」布局

Decision: 在單頁內明確分離「扣帳報廢」與「不扣帳報廢」兩條視覺敘事線,避免混讀。

Visual structure:

  • Header漸層標題區顯示頁名、資料更新時間、語義說明 badge扣帳/不扣帳)。
  • Filter Card至少提供三個核心篩選器並含查詢/清除動作:
    • 時間篩選(start_date/end_date
    • 原因篩選(LOSSREASONNAME
    • WORKCENTER_GROUP 篩選(沿用既有頁面篩選體驗與資料來源)
  • KPI Row8 卡):MOVEIN_QTYREJECT_TOTAL_QTYDEFECT_QTY、兩種 rate、REJECT_SHARE_PCT、受影響 lot/workorder。
  • Trend Row
    • 左圖:REJECT_TOTAL_QTY vs DEFECT_QTY(量)
    • 右圖:REJECT_RATE_PCT vs DEFECT_RATE_PCT(率)
  • Pareto + Detail報廢量 vs 報廢原因 Pareto支援 metric mode與可分頁明細表。
    • Pareto 預設啟用「僅顯示累計前 80%」模式(以目前篩選後資料集計算)。
    • 提供開關切換完整 Pareto關閉前 80% 模式時,顯示篩選後的全部原因。
    • 視覺與互動可參考 WIP OVER VIEW 既有 Pareto 呈現方式,保持使用者認知一致。

Visual semantics:

  • Reject扣帳使用暖色語義紅/橘系)
  • Defect不扣帳使用冷色語義藍/青系)
  • MOVEIN_QTY、總計與背景採中性灰藍語義
  • 互動態hover/active/filter chip沿用既有 shared style token確保與現有報表視覺一致

Alternatives considered:

  • 只保留單圖單指標:無法傳達兩種語義並列。
  • 新建完全獨立視覺主題:與既有 portal 風格落差大、維運成本高。

D5: 新頁採 pure Vite entry + Flask static route + shell native loader

Decision: frontend/src/reject-history/index.html 作為 entrybuild 後由 Flask send_from_directory 提供 /reject-history,並註冊到 routeContractsnativeModuleRegistry

Why:

  • 與既有 in-scope page 架構一致,降低整合風險。
  • 保留 shell canonical route 與 direct-entry 相容策略。

D6: 匯出欄位以「語義明示」優先於歷史相容命名

Decision: CSV 欄位明確輸出 REJECT_TOTAL_QTYDEFECT_QTY,並可附五個 reject 組成欄位;不使用易混淆別名。

Why:

  • 報表是對外分析依據,語義清晰優先於短期縮寫便利。
  • 與 field-name-consistency 治理要求一致。

D7: 納入「不計良率報廢」政策並提供可切換模式

Decision:ERP_PJ_WIP_SCRAP_REASONS_EXCLUDEENABLE_FLAG='Y' 為政策清單,預設排除該類報廢於良率相關計算,並新增 include_excluded_scrap 參數(預設 false)讓使用者可選擇納入。

Policy scope:

  • 影響 summary/trend/reason-pareto/list/export 的同一套查詢語義。
  • 預設模式下,標記為排除的報廢原因不進入良率計算;切換納入後,回到完整資料集計算。
  • API 回應 meta 顯示目前是否啟用排除政策,前端在 filter 區顯示「納入不計良率報廢」切換開關。

Why:

  • 業務規則已明示 ENABLE_FLAG='Y' 代表不納入良率計算,應在報表層落地。
  • 提供切換能兼顧日常看板(排除模式)與稽核追查(納入模式)。

Alternatives considered:

  • 永久硬排除:簡單但不利追溯與跨單位對帳。
  • 完全不排除:違反既有良率定義。

D8: 排除清單採「L2 Redis + L1 記憶體」每日全表快取

Decision: 新增 scrap_reason_exclusion_cache 模組採現有快取分層模式Redis 作為跨 worker 共用快取L2process memory 作為快速讀取層L1每日全表刷新一次啟動時先載入。

Refresh policy:

  • 啟動時進行首次載入。
  • 每 24 小時刷新一次(可由環境變數覆寫)。
  • Redis 不可用時,自動退化為 in-memory 快取,並在健康檢查/日誌揭露降級狀態。

Why:

  • 表筆數小(目前約 36 筆),適合全表快取,不需每次 query join DB。
  • 共享式 Redis 可避免多 gunicorn worker 間資料不一致。
  • 延續專案既有快取策略,降低維運認知成本。

Alternatives considered:

  • 僅記憶體快取:實作最簡單,但多 worker 會各自持有版本。
  • 每次即時查表:邏輯單純,但額外增加 Oracle 往返成本。

Risks / Trade-offs

  • [基底 SQL 單一來源造成查詢負載偏高] → 先以日期與維度條件收斂、list/export 加 rate limit必要時再追加快取或物化。
  • [使用者沿用舊語意理解 defect] → UI 顯示語義說明 badge + tooltip匯出欄位採顯式命名。
  • [Pareto 指標切換造成理解成本] → 預設以 REJECT_TOTAL_QTY 顯示,並保留清楚的 toggle label。
  • [報廢原因對應鍵格式不一致] → 在 service 層加入 reason normalization 規則trim/大小寫一致化,必要時切取代碼前綴),並在測試覆蓋。
  • [排除政策切換導致跨報表數值差異爭議] → API/前端都回傳並顯示 include_excluded_scrap 狀態與政策提示文字。
  • [Redis 不可用導致快取行為不一致] → 採 L1 fallback並透過 health/admin 狀態揭露快取降級。
  • [路由治理漏登記導致 shell 無法導航] → contract parity test + page_status 驗證列為必做任務。
  • [明細資料量大造成前端卡頓] → 後端分頁、預設 per_page=50,並避免一次性全量載入。

Migration Plan

  1. 建立後端 SQL/service/routes先讓 API 可單獨驗證)。
  2. 建立 scrap_reason_exclusion_cache(全表快取 + 每日刷新 + fallback
  3. 建立前端 reject-history 頁面與元件(先接 summary/trend再接 pareto/list/export
  4. 整合 shell 治理資產:routeContractsnativeModuleRegistrypage_status、Flask page route。
  5. 補齊測試service、routes、cache、route-contract parity、前端 smoke。
  6. 先以 dev 狀態上線到抽屜,完成 UAT 後調整為 released

Rollback strategy:

  • /reject-history 從 page registry 標記為隱藏或 dev 並停用導航入口。
  • 保留已上線的既有頁面與 API不影響既有報表路徑。

Open Questions

  • 趨勢圖預設粒度是否固定「日」,或需同頁支援週/月切換?
  • Pareto 預設排序基準是否固定 REJECT_TOTAL_QTY,是否要允許切換為 DEFECT_QTY
  • 匯出是否要同時提供「彙總版」與「明細版」兩種檔案型態,或先只提供明細版?