18 KiB
Enum Metadata 최적화 - 코딩 계획
작성일: 2026-05-08 상태: 진행 중 기반 문서:
plans/enum-metadata-optimization.md목적:state0~7descriptor제거,desc/area만 유지, pv 값 파싱 로직 추가
작업 Todo 리스트
각 단계는 독립적으로 완료 여부를 추적할 수 있다. 체크박스를 사용하여 진행 상황을 기록한다.
- STEP 1 — 백업: 수정 대상 파일들을
.rooBackup/에 백업 - STEP 2 —
MetadataLoaderService.cs:MetaAttributes배열에서 state0~7descriptor 제거 - STEP 3 —
MetadataLoaderService.cs: 빌드 검증 - STEP 4 —
ExperionDbContext.cs:v_tag_summary뷰에서 state descriptor JOIN 제거 - STEP 5 —
ExperionDbContext.cs: 빌드 검증 - STEP 6 —
app.js: pv 값 파싱 헬퍼 함수parseEnumPv()추가 - STEP 7 —
app.js: pv 값 표시 관련 코드 모두parseEnumPv()적용 - STEP 8 — End-to-End 검증: 전체 빌드 + UI 테스트
- STEP 9 — git 커밋 및 정리
변경 대상 파일 목록
| # | 파일 | 변경 내용 | 영향 범위 |
|---|---|---|---|
| 1 | src/Infrastructure/OpcUa/MetadataLoaderService.cs |
MetaAttributes 배열 수정 |
메타데이터 로딩 |
| 2 | src/Infrastructure/Database/ExperionDbContext.cs |
v_tag_summary 뷰 단순화 |
DB 뷰 |
| 3 | src/Web/wwwroot/js/app.js |
pv 파싱 헬퍼 + 표시 로직 변경 | 프론트엔드 UI |
각 단계 상세 계획
STEP 1 — 백업: 수정 대상 파일들을 .rooBackup/에 백업
목적: 수정 전 원본 보존. 실패 시 복원 가능.
실행 명령:
TIMESTAMP=$(date +%Y%m%d%H%M)
mkdir -p .rooBackup/enum-opt-$TIMESTAMP/src/Infrastructure/OpcUa
mkdir -p .rooBackup/enum-opt-$TIMESTAMP/src/Infrastructure/Database
mkdir -p .rooBackup/enum-opt-$TIMESTAMP/src/Web/wwwroot/js
cp src/Infrastructure/OpcUa/MetadataLoaderService.cs .rooBackup/enum-opt-$TIMESTAMP/src/Infrastructure/OpcUa/
cp src/Infrastructure/Database/ExperionDbContext.cs .rooBackup/enum-opt-$TIMESTAMP/src/Infrastructure/Database/
cp src/Web/wwwroot/js/app.js .rooBackup/enum-opt-$TIMESTAMP/src/Web/wwwroot/js/
검증 기준:
.rooBackup/enum-opt-YYYYMMDDHHMM/폴더가 생성되고 3개 파일이 복사됨- 원본 파일과 백업 파일의 체크섬이 일치함 (
md5sum비교)
enum-metadata-optimization.md 규칙 매핑:
- 섹션 3.3: 수정이 필요한 코드 —
MetadataLoaderService.cs,ExperionDbContext.cs,app.js
STEP 2 — MetadataLoaderService.cs: MetaAttributes 배열에서 state0~7descriptor 제거
파일: MetadataLoaderService.cs
변경 위치: 20~26줄 (MetaAttributes 배열 정의)
변경 전 코드:
// 로드할 메타데이터 속성 목록
private static readonly string[] MetaAttributes =
{
"desc", "area",
"state0descriptor", "state1descriptor", "state2descriptor",
"state3descriptor", "state4descriptor", "state5descriptor",
"state6descriptor", "state7descriptor"
};
변경 후 코드:
// 로드할 메타데이터 속성 목록 (state0~7descriptor 제거 — pv 값에서 파싱)
private static readonly string[] MetaAttributes =
{
"desc", "area"
};
diff:
-// 로드할 메타데이터 속성 목록
+// 로드할 메타데이터 속성 목록 (state0~7descriptor 제거 — pv 값에서 파싱)
private static readonly string[] MetaAttributes =
{
- "desc", "area",
- "state0descriptor", "state1descriptor", "state2descriptor",
- "state3descriptor", "state4descriptor", "state5descriptor",
- "state6descriptor", "state7descriptor"
+ "desc", "area"
};
변경 이유:
- pv 값
{코드 | DisplayName | }에 이미 state descriptor 정보가 포함되어 있으므로 중복 저장 제거 - OPC UA 읽기 요청 80% 감소 (10개 속성 → 2개 속성)
- tag_metadata 행 80% 감소 (태그당 10행 → 2행)
영향 분석:
LoadMetadataAsync()메서드:MetaAttributes.Contains(n.Name)으로node_map_master에서 필터링하므로 state descriptor 노드는 더 이상 조회되지 않음ReadTagsAsync(): 8개 menos의 nodeId로 배치 읽기 → 네트워크 트래픽 감소- UPSERT 쿼리: 8개 menos의 행 삽입 → DB 부하 감소
검증 기준:
MetaAttributes배열이["desc", "area"]두 개만 포함- 컴파일 오류 없음
LoadMetadataAsync()메서드의 로직 변경 없이 배열 변경만으로 동작
enum-metadata-optimization.md 규칙 매핑:
- 섹션 3.2: desc, area만 저장
- 섹션 3.3:
MetadataLoaderService.cs수정 —MetaAttributes배열에서 state0~7descriptor 제거 - 섹션 4.2: tag_metadata 저장 속성 — desc, area 만
STEP 3 — MetadataLoaderService.cs 변경 후 빌드 검증
목적: STEP 2 변경이 컴파일 오류 없이 통과하는지 확인
실행 명령:
dotnet build src/Web/ExperionCrawler.csproj --no-restore -v q
검증 기준:
- 빌드 성공 (exit code 0)
- 경고 없음 (또는 기존 경고만)
MetaAttributes배열 참조하는 다른 코드가 없는지 확인
실패 시 대응:
- 컴파일 오류가 발생하면 오류 메시지를 읽고 원인 분석
MetaAttributes길이에 의존하는 코드가 있으면 해당 코드도 수정
STEP 4 — ExperionDbContext.cs: v_tag_summary 뷰에서 state descriptor JOIN 제거
파일: ExperionDbContext.cs
변경 위치: 302~330줄 (v_tag_summary 뷰 생성 SQL)
변경 전 코드 (315~329줄):
desc_md.value AS description,
area_md.value AS area,
s0d_md.value AS state0_descriptor,
s1d_md.value AS state1_descriptor,
s2d_md.value AS state2_descriptor
FROM (SELECT DISTINCT split_part(tagname, '.', 1) AS base_tag FROM realtime_table) rt_base
LEFT JOIN realtime_table pv_rt ON pv_rt.tagname = rt_base.base_tag || '.pv'
LEFT JOIN realtime_table sp_rt ON sp_rt.tagname = rt_base.base_tag || '.sp'
LEFT JOIN realtime_table op_rt ON op_rt.tagname = rt_base.base_tag || '.op'
LEFT JOIN realtime_table instate0_rt ON instate0_rt.tagname = rt_base.base_tag || '.instate0'
LEFT JOIN realtime_table instate1_rt ON instate1_rt.tagname = rt_base.base_tag || '.instate1'
LEFT JOIN realtime_table instate2_rt ON instate2_rt.tagname = rt_base.base_tag || '.instate2'
LEFT JOIN tag_metadata desc_md ON desc_md.base_tag = rt_base.base_tag AND desc_md.attribute = 'desc'
LEFT JOIN tag_metadata area_md ON area_md.base_tag = rt_base.base_tag AND area_md.attribute = 'area'
LEFT JOIN tag_metadata s0d_md ON s0d_md.base_tag = rt_base.base_tag AND s0d_md.attribute = 'state0descriptor'
LEFT JOIN tag_metadata s1d_md ON s1d_md.base_tag = rt_base.base_tag AND s1d_md.attribute = 'state1descriptor'
LEFT JOIN tag_metadata s2d_md ON s2d_md.base_tag = rt_base.base_tag AND s2d_md.attribute = 'state2descriptor'
변경 후 코드:
desc_md.value AS description,
area_md.value AS area
FROM (SELECT DISTINCT split_part(tagname, '.', 1) AS base_tag FROM realtime_table) rt_base
LEFT JOIN realtime_table pv_rt ON pv_rt.tagname = rt_base.base_tag || '.pv'
LEFT JOIN realtime_table sp_rt ON sp_rt.tagname = rt_base.base_tag || '.sp'
LEFT JOIN realtime_table op_rt ON op_rt.tagname = rt_base.base_tag || '.op'
LEFT JOIN realtime_table instate0_rt ON instate0_rt.tagname = rt_base.base_tag || '.instate0'
LEFT JOIN realtime_table instate1_rt ON instate1_rt.tagname = rt_base.base_tag || '.instate1'
LEFT JOIN realtime_table instate2_rt ON instate2_rt.tagname = rt_base.base_tag || '.instate2'
LEFT JOIN tag_metadata desc_md ON desc_md.base_tag = rt_base.base_tag AND desc_md.attribute = 'desc'
LEFT JOIN tag_metadata area_md ON area_md.base_tag = rt_base.base_tag AND area_md.attribute = 'area'
diff:
desc_md.value AS description,
area_md.value AS area,
- s0d_md.value AS state0_descriptor,
- s1d_md.value AS state1_descriptor,
- s2d_md.value AS state2_descriptor
+ area_md.value AS area
FROM (SELECT DISTINCT split_part(tagname, '.', 1) AS base_tag FROM realtime_table) rt_base
LEFT JOIN realtime_table pv_rt ON pv_rt.tagname = rt_base.base_tag || '.pv'
LEFT JOIN realtime_table sp_rt ON sp_rt.tagname = rt_base.base_tag || '.sp'
LEFT JOIN realtime_table op_rt ON op_rt.tagname = rt_base.base_tag || '.op'
LEFT JOIN realtime_table instate0_rt ON instate0_rt.tagname = rt_base.base_tag || '.instate0'
LEFT JOIN realtime_table instate1_rt ON instate1_rt.tagname = rt_base.base_tag || '.instate1'
LEFT JOIN realtime_table instate2_rt ON instate2_rt.tagname = rt_base.base_tag || '.instate2'
LEFT JOIN tag_metadata desc_md ON desc_md.base_tag = rt_base.base_tag AND desc_md.attribute = 'desc'
LEFT JOIN tag_metadata area_md ON area_md.base_tag = rt_base.base_tag AND area_md.attribute = 'area'
- LEFT JOIN tag_metadata s0d_md ON s0d_md.base_tag = rt_base.base_tag AND s0d_md.attribute = 'state0descriptor'
- LEFT JOIN tag_metadata s1d_md ON s1d_md.base_tag = rt_base.base_tag AND s1d_md.attribute = 'state1descriptor'
- LEFT JOIN tag_metadata s2d_md ON s2d_md.base_tag = rt_base.base_tag AND s2d_md.attribute = 'state2descriptor'
""");
변경 이유:
- state0~2descriptor가 더 이상 tag_metadata에 저장되지 않으므로 JOIN 제거
- 뷰 조회 성능 향상 (3개 LEFT JOIN 제거)
영향 분석:
v_tag_summary뷰를 조회하는 코드가state0_descriptor,state1_descriptor,state2_descriptor컬럼을 참조하면 NULL 반환- 이 뷰를 사용하는 곳이 있는지 검색 필요 (현재는 DB 초기화 시에만 사용)
검증 기준:
v_tag_summary뷰 SQL에서 state descriptor 관련 JOIN 3개 제거됨area_md.value AS area뒤 쉼표 제거 (마지막 SELECT 컬럼)- SQL 문법 오류 없음
enum-metadata-optimization.md 규칙 매핑:
- 섹션 3.3:
ExperionDbContext.cs수정 — v_tag_summary 뷰에서 state descriptor JOIN 제거
STEP 5 — ExperionDbContext.cs 변경 후 빌드 검증
목적: STEP 4 변경이 컴파일 오류 없이 통과하는지 확인
실행 명령:
dotnet build src/Web/ExperionCrawler.csproj --no-restore -v q
검증 기준:
- 빌드 성공 (exit code 0)
- SQL 문자열 문법 오류 없음 (raw string literal 안에서 SQL 구문 확인)
- 쉼표 누락/과잉 없음
실패 시 대응:
- SQL 구문 오류가 발생하면 SELECT 컬럼 목록의 쉼표 확인
area_md.value AS area가 마지막 컬럼이므로 쉼표 제거했는지 확인
STEP 6 — app.js: pv 값 파싱 헬퍼 함수 parseEnumPv() 추가
파일: app.js
삽입 위치: fmtVal() 함수 바로 아래 (2132줄 이후)
추가할 코드:
/**
* OPC UA EnumValueType pv 값 파싱
* "{0 | L-STpv | }" → "L-STpv"
* "{0 | MID | }" → "MID"
* 일반 값은 그대로 반환
*/
function parseEnumPv(v) {
if (v == null) return v;
const s = String(v);
// "{코드 | DisplayName | }" 패턴 매칭
const m = s.match(/^\{\s*\d+\s*\|\s*([^|]+?)\s*\|\s*\}$/);
return m ? m[1].trim() : s;
}
삽입 위치 상세:
- 2132줄 (
}—fmtVal함수 종료) 바로 다음에 삽입 - 기존
fmtVal함수는 변경하지 않음
기능 설명:
- 입력: pv 값 문자열 (예:
"{0 | L-STpv | }") - 출력: DisplayName 부분만 (예:
"L-STpv") - EnumValueType 형식이 아닌 일반 값은 그대로 반환
정규식 설명:
^\{—{로 시작\s*\d+\s*— 정수 코드 (공백 허용)\|\s*—|구분자([^|]+?)— DisplayName (첫 번째 캡처 그룹)\s*\|\s*\}$—| }로 끝남
검증 기준:
parseEnumPv("{0 | L-STpv | }")→"L-STpv"반환parseEnumPv("{0 | MID | }")→"MID"반환parseEnumPv("123.45")→"123.45"반환 (변경 없음)parseEnumPv(null)→null반환- 브라우저 콘솔에서 수동 테스트 가능
enum-metadata-optimization.md 규칙 매핑:
- 섹션 3.2: pv 값 파싱 방법 —
{코드 | 이름 | }에서 DisplayName 추출 - 섹션 3.3:
app.js— pv 값 표시 시 파싱 로직 추가
STEP 7 — app.js: pv 값 표시 관련 코드 모두 parseEnumPv() 적용
파일: app.js
변경 위치: pbRender() 함수 내 liveValue 표시 부분 (633줄)
변경 전 코드 (633줄):
<td class="val">${p?.liveValue != null ? esc(String(fmtVal(p.liveValue))) : '<span style="color:var(--t3)">—</span>'}</td>
변경 후 코드:
<td class="val">${p?.liveValue != null ? esc(String(fmtVal(parseEnumPv(p.liveValue)))) : '<span style="color:var(--t3)">—</span>'}</td>
diff:
-<td class="val">${p?.liveValue != null ? esc(String(fmtVal(p.liveValue))) : '<span style="color:var(--t3)">—</span>'}</td>
+<td class="val">${p?.liveValue != null ? esc(String(fmtVal(parseEnumPv(p.liveValue)))) : '<span style="color:var(--t3)">—</span>'}</td>
변경 이유:
liveValue가 EnumValueType 형식({0 | L-STpv | })이면 DisplayName만 표시- 일반 숫자 값은
parseEnumPv()가 그대로 반환하므로 영향 없음 fmtVal()은 숫자 포맷팅만 담당,parseEnumPv()는 EnumValueType 파싱만 담당 (단일 책임)
다른 적용 위치 확인:
fmtVal()을 사용하는 모든 곳을 검색하여 pv 값 표시에 사용되는 곳에parseEnumPv()적용t2sRenderTable()(1606줄):fmtVal(val)— 시계열 데이터이므로 변경 불필요 (아날로그 값 중심)histQuery()결과 렌더링 (956줄):fmtVal(raw)— 이력 데이터이므로 변경 불필요
검증 기준:
- 포인트빌더 테이블에서 xv-3202.pv 값이
"{0 | L-MID | }"대신"L-MID"로 표시됨 - 일반 숫자 값(예: ficq-6113.pv = "123.45")은 여전히
"123.45"로 표시됨 - 브라우저 콘솔에서 JS 오류 없음
enum-metadata-optimization.md 규칙 매핑:
- 섹션 3.2: 프론트엔드에서 pv 값 표시 시
{와|를 파싱하여 상태 이름만 표시
STEP 8 — End-to-End 검증: 전체 빌드 + UI 테스트
목적: 모든 변경 사항이 함께 동작하는지 최종 확인
실행 명령:
# 1. 전체 솔루션 빌드
dotnet build ExperionCrawler.sln -v q
# 2. 애플리케이션 시작
dotnet run --project src/Web/ExperionCrawler.csproj
검증 체크리스트:
빌드 검증
dotnet build성공 (exit code 0)- 컴파일 오류 0개
- 경고 수 증가 없음
백엔드 검증
- 애플리케이션 시작 시 DB 초기화 성공
v_tag_summary뷰 생성 성공 (state descriptor JOIN 없음)- 메타데이터 로드 시
desc,area만 조회됨 (로그 확인) tag_metadata테이블에 state0~7descriptor 행 없음
프론트엔드 검증
- 브라우저 콘솔 JS 오류 없음
- 포인트빌더 테이블에서 digital 태그 pv 값이 DisplayName만 표시됨
- 예:
xv-3202.pv→"L-MID"(기존:"{0 | L-MID | }")
- 예:
- 아날로그 태그 pv 값은 정상 표시됨
- 예:
ficq-6113.pv→"123.45"
- 예:
- 메타데이터 조회 시
desc,area만 반환됨
성능 검증
- 메타데이터 로드 시간 감소 확인 (기존 대비)
tag_metadata테이블 행 수 감소 확인 (기존 10행/태그 → 2행/태그)
enum-metadata-optimization.md 규칙 매핑:
- 섹션 4.3: 기대 효과 — OPC UA 읽기 요청 80% 감소, tag_metadata 행 80% 감소
- 섹션 4.4: 다음 단계 — 메타데이터 로드 테스트 실행
STEP 9 — git 커밋 및 정리
목적: 변경 사항을 커밋하고 작업 기록 남기기
커밋 전략: 각 파일 변경을 별도 커밋으로 관리
# 1. MetadataLoaderService.cs 커밋
git add src/Infrastructure/OpcUa/MetadataLoaderService.cs
git commit -m "feat: MetaAttributes에서 state0~7descriptor 제거 (pv 값 파싱으로 대체)"
# 2. ExperionDbContext.cs 커밋
git add src/Infrastructure/Database/ExperionDbContext.cs
git commit -m "feat: v_tag_summary 뷰에서 state descriptor JOIN 제거"
# 3. app.js 커밋
git add src/Web/wwwroot/js/app.js
git commit -m "feat: pv 값 파싱 헬퍼 parseEnumPv() 추가, 포인트빌더 테이블 적용"
# 4. 계획 문서 커밋
git add plans/enum-metadata-optimize-coding-plan.md
git commit -m "docs: enum metadata 최적화 코딩 계획 작성"
검증 기준:
- 각 커밋 메시지가 변경 내용을 명확히 설명함
git log에서 커밋 순서가 논리적임.rooBackup/폴더에 gitignore 적용되어 있음 (백업 파일 커밋 제외)
부록: rollback 절차
작업 중 문제가 발생하면 백업 파일로 복원:
TIMESTAMP=$(ls -d .rooBackup/enum-opt-* | tail -1 | xargs basename)
cp .rooBackup/$TIMESTAMP/src/Infrastructure/OpcUa/MetadataLoaderService.cs src/Infrastructure/OpcUa/
cp .rooBackup/$TIMESTAMP/src/Infrastructure/Database/ExperionDbContext.cs src/Infrastructure/Database/
cp .rooBackup/$TIMESTAMP/src/Web/wwwroot/js/app.js src/Web/wwwroot/js/
검증 요약
각 STEP의 검증 기준을 한눈에 확인:
| STEP | 파일 | 핵심 검증 항목 |
|---|---|---|
| 1 | — | 백업 파일 3개 생성됨 |
| 2 | MetadataLoaderService.cs | MetaAttributes = ["desc", "area"] |
| 3 | — | 빌드 성공 |
| 4 | ExperionDbContext.cs | state descriptor JOIN 3개 제거됨 |
| 5 | — | 빌드 성공 |
| 6 | app.js | parseEnumPv() 함수 추가됨 |
| 7 | app.js | pbRender()에서 parseEnumPv() 적용됨 |
| 8 | 전체 | End-to-End 테스트 통과 |
| 9 | — | git 커밋 완료 |