feat: migrate ServerPC data to asset_pc, enhance filters with location, and standardize page headers
- 서버PC 자산을 asset_pc 테이블로 통합 마이그레이션 및 스키마 확장 (위치, IP 정보 복구 완료) - 하드웨어 자산 페이지의 구매법인 필터를 자산위치 필터로 교체 및 동적 데이터 바인딩 적용 - 모든 자산 리스트 페이지 상단에 설명(Description) 필드 추가 및 헤더 표준화 - 상세 모달 내 삭제 버튼 기능 구현 및 서버PC 용도 필드 노출 오류 수정 - 현 사용조직 필터 리스트가 비어있던 DOM 셀렉터 버그 수정
This commit is contained in:
@@ -1,50 +1,28 @@
|
||||
import * as XLSX from 'xlsx';
|
||||
import { ASSET_SCHEMA } from './schema';
|
||||
|
||||
/**
|
||||
* ITAM 엑셀 핸들러 (Database Synchronized Edition)
|
||||
* 데이터베이스 실제 스키마 컬럼과 엑셀 헤더를 1:1로 일치시킵니다.
|
||||
*/
|
||||
|
||||
export interface HardwareAsset {
|
||||
[key: string]: any;
|
||||
id: string;
|
||||
type: string;
|
||||
법인: string;
|
||||
자산코드: string;
|
||||
명칭: string;
|
||||
위치: string;
|
||||
관리자: string;
|
||||
IP주소: string;
|
||||
MACaddress: string;
|
||||
HW사양: string;
|
||||
OS: string;
|
||||
금액?: string;
|
||||
납품업체: string;
|
||||
품의서명: string;
|
||||
비고?: string;
|
||||
}
|
||||
|
||||
export interface SoftwareAsset {
|
||||
[key: string]: any;
|
||||
id: string;
|
||||
type: string;
|
||||
분야?: string;
|
||||
법인: string;
|
||||
부서?: string;
|
||||
제품명: string;
|
||||
금액: string;
|
||||
수량: number;
|
||||
계정명: string;
|
||||
납품업체: string;
|
||||
비고: string;
|
||||
}
|
||||
|
||||
export interface SWUser {
|
||||
id: string;
|
||||
sw_id: string;
|
||||
법인: string;
|
||||
부서: string;
|
||||
팀: string;
|
||||
직위: string;
|
||||
이름: string;
|
||||
사용기간: string;
|
||||
신청서명: string;
|
||||
userData?: any[];
|
||||
user_name: string;
|
||||
dept: string;
|
||||
corp: string;
|
||||
[key: string]: any;
|
||||
}
|
||||
|
||||
export interface HardwareLog {
|
||||
@@ -53,106 +31,193 @@ export interface HardwareLog {
|
||||
date: string;
|
||||
details: string;
|
||||
user: string;
|
||||
cost?: number;
|
||||
}
|
||||
|
||||
export interface MasterAssetData {
|
||||
pc: HardwareAsset[];
|
||||
server: HardwareAsset[];
|
||||
storage: HardwareAsset[];
|
||||
equip: HardwareAsset[];
|
||||
mobile: HardwareAsset[];
|
||||
subSw: SoftwareAsset[];
|
||||
permSw: SoftwareAsset[];
|
||||
network: HardwareAsset[];
|
||||
equipment: HardwareAsset[];
|
||||
survey: HardwareAsset[];
|
||||
pcParts: HardwareAsset[];
|
||||
swInternal: SoftwareAsset[];
|
||||
swExternal: SoftwareAsset[];
|
||||
cloud: SoftwareAsset[];
|
||||
domain?: any[];
|
||||
hw: HardwareAsset[];
|
||||
sw: SoftwareAsset[];
|
||||
domain: any[];
|
||||
vip: HardwareAsset[];
|
||||
officeSupplies: HardwareAsset[];
|
||||
cost: any[];
|
||||
swUsers: SWUser[];
|
||||
logs: HardwareLog[];
|
||||
[key: string]: any;
|
||||
}
|
||||
|
||||
const PC_HEADERS = ['법인', '자산코드', '구매연월', '사용자', '현사용조직', '이전사용조직', '위치', '담당자(정)', '담당자(부)', '모델명', 'OS', 'CPU', 'GPU', 'RAM', 'SSD1', 'SSD2', 'SSD3', '메인보드', 'IP주소', '금액', '납품업체', '품의서명', '비고'];
|
||||
const SERVER_HEADERS = ['법인', '자산코드', '구매연월', '유형', '용도', '상세내용', '현사용조직', '이전사용조직', '위치', '담당자(정)', '담당자(부)', 'IP 주소 1', 'IP 주소 2', '원격도구', '서버 ID', '서버 PW', '모델명', 'OS', 'CPU', 'RAM', 'GPU', 'Storage 1', 'Storage 2', 'Storage 3', '모니터링', '금액', '납품업체', '품의서명', '비고'];
|
||||
const STORAGE_HEADERS = ['법인', '유형', '자산코드', '명칭', '위치', '모델명', '용량', '담당자(정)', '담당자(부)', 'IP주소', 'MAC주소', '구매연월', '금액', '납품업체', '품의서명', '비고'];
|
||||
const EQUIP_HEADERS = ['법인', '비품유형', '자산코드', '명칭', '위치', '관리자', 'IP주소', 'MACaddress', 'HW사양', 'OS', '구매연월', '금액', '납품업체', '품의서명', '비고'];
|
||||
const MOBILE_HEADERS = ['법인', '자산코드', '명칭', '위치', '관리자', '기기유형', 'OS', '구매연월', '금액', '납품업체', '품의서명', '비고'];
|
||||
|
||||
const SUB_SW_HEADERS = ['분야', '법인', '제품명', '부서', '수량', '금액', '구매일', '납품업체', '시작일', '만료일', '라이선스유형', '계정명', '비고'];
|
||||
const PERM_SW_HEADERS = ['분야', '법인', '제품명', '부서', '수량', '금액', '구매일', '납품업체', '시작일', '만료일', '라이선스키', '계정명', '비고'];
|
||||
const CLOUD_HEADERS = ['플랫폼명', '법인', '제품명', '부서', '계정명', '결제수단', '결제일', '연결카드번호', '당월청구액', '비고'];
|
||||
|
||||
const DOMAIN_HEADERS = ['유형', '법인', '서비스명', '관리도메인', '시작일', '만료일', '금액', '담당자', '담당자(부)', '비고'];
|
||||
/**
|
||||
* DB 컬럼 순서 및 구성 정의 (실제 DB 스키마 dump 기준)
|
||||
*/
|
||||
const DB_MAPPING: Record<string, (keyof typeof ASSET_SCHEMA)[]> = {
|
||||
pc: [
|
||||
'ASSET_TYPE', 'HW_STATUS', 'CURRENT_DEPT', 'PREV_DEPT', 'USER_POSITION',
|
||||
'EMP_NO', 'CURRENT_USER',
|
||||
'CPU', 'RAM', 'GPU', 'SSD1', 'SSD2', 'HDD1', 'HDD2', 'HDD3', 'HDD4', 'MAC_ADDR',
|
||||
'MANAGER_MAIN', 'MANAGER_SUB', 'PURCHASE_CORP', 'PURCHASE_DATE', 'PURCHASE_AMOUNT',
|
||||
'PURCHASE_VENDOR', 'MEMO', 'MAINBOARD'
|
||||
],
|
||||
server: [
|
||||
'ASSET_TYPE', 'MODEL_NAME', 'ASSET_PURPOSE', 'HW_STATUS',
|
||||
'CURRENT_DEPT', 'CPU', 'RAM', 'GPU', 'SSD1', 'SSD2', 'HDD1', 'HDD2', 'IP_ADDR',
|
||||
'REMOTE_TOOL', 'REMOTE_ID', 'REMOTE_PW', 'LOCATION', 'LOC_DETAIL', 'MANAGER_MAIN',
|
||||
'PURCHASE_CORP', 'PURCHASE_DATE', 'PURCHASE_AMOUNT', 'PURCHASE_VENDOR',
|
||||
'MEMO', 'PREV_DEPT', 'MANAGER_SUB', 'IP_ADDR2', 'MONITORING', 'HDD3', 'HDD4', 'EMP_NO'
|
||||
],
|
||||
storage: [
|
||||
'ASSET_TYPE', 'HW_STATUS', 'VOLUME', 'MODEL_NAME',
|
||||
'EMP_NO', 'CURRENT_USER',
|
||||
'SERIAL_NUM', 'LOCATION', 'LOC_DETAIL', 'MANAGER_MAIN', 'MANAGER_SUB',
|
||||
'PURCHASE_CORP', 'PURCHASE_DATE', 'PURCHASE_AMOUNT', 'PURCHASE_VENDOR',
|
||||
'MEMO', 'CURRENT_DEPT', 'PREV_DEPT'
|
||||
],
|
||||
network: [
|
||||
'PURCHASE_CORP', 'HW_STATUS', 'CURRENT_DEPT', 'PREV_DEPT',
|
||||
'EMP_NO', 'CURRENT_USER',
|
||||
'ASSET_TYPE', 'ASSET_MFR', 'MODEL_NAME', 'LOCATION', 'LOC_DETAIL', 'MANAGER_MAIN',
|
||||
'MANAGER_SUB', 'PURCHASE_DATE', 'PURCHASE_AMOUNT', 'PURCHASE_VENDOR', 'MEMO'
|
||||
],
|
||||
survey: [ // asset_survey (공간정보장비)
|
||||
'HW_STATUS', 'ASSET_NAME', 'LOCATION', 'LOC_DETAIL',
|
||||
'EMP_NO', 'CURRENT_USER',
|
||||
'MANAGER_MAIN', 'MANAGER_SUB', 'PURCHASE_CORP', 'PURCHASE_DATE', 'PURCHASE_AMOUNT',
|
||||
'PURCHASE_VENDOR', 'MEMO'
|
||||
],
|
||||
pcParts: [
|
||||
'HW_STATUS', 'ASSET_TYPE', 'ASSET_MFR', 'MODEL_NAME', 'VOLUME',
|
||||
'EMP_NO', 'CURRENT_USER',
|
||||
'MONITOR_INCH', 'LOCATION', 'LOC_DETAIL', 'PURCHASE_CORP', 'PURCHASE_DATE',
|
||||
'PURCHASE_AMOUNT', 'PURCHASE_VENDOR', 'MEMO'
|
||||
],
|
||||
equipment: [
|
||||
'HW_STATUS', 'ASSET_STATUS', 'ASSET_TYPE', 'ASSET_MFR',
|
||||
'EMP_NO', 'CURRENT_USER',
|
||||
'MODEL_NAME', 'LOCATION', 'LOC_DETAIL', 'MANAGER_MAIN', 'MANAGER_SUB',
|
||||
'PURCHASE_CORP', 'PURCHASE_DATE', 'PURCHASE_AMOUNT', 'PURCHASE_VENDOR',
|
||||
'MEMO'
|
||||
],
|
||||
officeSupplies: [ // asset_office_supplies (시설자산)
|
||||
'HW_STATUS', 'ASSET_TYPE', 'ASSET_MFR', 'MODEL_NAME',
|
||||
'EMP_NO', 'CURRENT_USER',
|
||||
'ASSET_COUNT', 'LOCATION', 'LOC_DETAIL', 'MANAGER_MAIN', 'MANAGER_SUB',
|
||||
'PURCHASE_CORP', 'PURCHASE_DATE', 'PURCHASE_AMOUNT', 'PURCHASE_VENDOR',
|
||||
'MEMO'
|
||||
],
|
||||
swInternal: [
|
||||
'SW_FIELD', 'DEV_OBJ', 'SW_STATUS', 'SW_TYPE', 'MANAGER_MAIN',
|
||||
'DEV_MGR', 'PLANNING_MGR', 'SALES_MGR', 'PURCHASE_CORP', 'MEMO'
|
||||
],
|
||||
swExternal: [
|
||||
'PRODUCT_NAME', 'SW_TYPE', 'SW_STATUS', 'SW_FIELD', 'CURRENT_DEPT',
|
||||
'PREV_DEPT', 'MANAGER_MAIN', 'PURCHASE_CORP', 'PURCHASE_DATE', 'PURCHASE_AMOUNT',
|
||||
'PURCHASE_VENDOR', 'EMAIL_ACCOUNT', 'MEMO', 'EMP_NO', 'CURRENT_USER'
|
||||
],
|
||||
cloud: [
|
||||
'ASSET_PURPOSE', 'PURCHASE_METHOD', 'PURCHASE_VENDOR', 'PURCHASE_CORP',
|
||||
'PURCHASE_DATE', 'PURCHASE_AMOUNT', 'MANAGER_MAIN', 'MANAGER_SUB',
|
||||
'MEMO', 'SW_ID', 'SW_PW'
|
||||
],
|
||||
domain: [
|
||||
'DOMAIN_ADDR', 'ASSET_PURPOSE', 'PURCHASE_VENDOR', 'ASSET_TYPE',
|
||||
'PURCHASE_CORP', 'PURCHASE_DATE', 'PURCHASE_AMOUNT', 'MANAGER_MAIN', 'MANAGER_SUB',
|
||||
'MEMO'
|
||||
],
|
||||
cost: [
|
||||
'ASSET_TYPE', 'ASSET_PURPOSE', 'LOCATION', 'LOC_DETAIL', 'MANAGER_MAIN',
|
||||
'MANAGER_SUB', 'PURCHASE_CORP', 'PURCHASE_DATE', 'PURCHASE_AMOUNT', 'PURCHASE_VENDOR',
|
||||
'EMAIL_ACCOUNT', 'EMAIL_PW', 'MEMO', 'EMP_NO', 'CURRENT_USER'
|
||||
],
|
||||
vip: [ // asset_vip (선물)
|
||||
'ASSET_NAME', 'MODEL_NAME', 'LOCATION', 'LOC_DETAIL',
|
||||
'PURCHASE_CORP', 'PURCHASE_DATE', 'EXPIRED_DATE', 'PURCHASE_VENDOR', 'MEMO'
|
||||
]
|
||||
};
|
||||
|
||||
export function downloadTemplate() {
|
||||
const wb = XLSX.utils.book_new();
|
||||
|
||||
const tabConfigs = [
|
||||
{ name: '개인PC', headers: PC_HEADERS },
|
||||
{ name: '서버', headers: SERVER_HEADERS },
|
||||
{ name: '스토리지', headers: STORAGE_HEADERS },
|
||||
{ name: '전산비품', headers: EQUIP_HEADERS },
|
||||
{ name: '모바일기기', headers: MOBILE_HEADERS },
|
||||
{ name: '구독SW', headers: SUB_SW_HEADERS },
|
||||
{ name: '영구SW', headers: PERM_SW_HEADERS },
|
||||
{ name: '클라우드', headers: CLOUD_HEADERS },
|
||||
{ name: '도메인', headers: DOMAIN_HEADERS }
|
||||
{ name: 'PC', key: 'pc' },
|
||||
{ name: '서버', key: 'server' },
|
||||
{ name: '스토리지', key: 'storage' },
|
||||
{ name: '공간정보장비', key: 'survey' },
|
||||
{ name: 'PC부품', key: 'pcParts' },
|
||||
{ name: '네트워크', key: 'network' },
|
||||
{ name: '업무지원장비', key: 'equipment' },
|
||||
{ name: '내부SW', key: 'swInternal' },
|
||||
{ name: '외부SW', key: 'swExternal' },
|
||||
{ name: '클라우드', key: 'cloud' },
|
||||
{ name: '도메인', key: 'domain' },
|
||||
{ name: '비용관리', key: 'cost' },
|
||||
{ name: '선물', key: 'vip' },
|
||||
{ name: '시설자산', key: 'officeSupplies' }
|
||||
];
|
||||
|
||||
const sampleData: Record<string, any[]> = {
|
||||
'개인PC': ['(주)에이치엠', 'PC-24001', '202401', '홍길동', '기술팀', '-', '서울본사 7층', '김관리', '이부관', 'LG Gram 16', 'Windows 11', 'i7-1360P', 'RTX 3050', '16GB', '512GB', '-', '-', 'LG Mainboard', '192.168.0.10', '1500000', 'LG전자', '2024_상반기_PC구매.pdf', '신규 입사자 지급용'],
|
||||
'서버': ['(주)에이치엠', 'SRV-24001', '202401', '물리', '웹서버', '운영 웹 서버', '인프라팀', '-', 'IDC 센터 1-A', '박서버', '최백업', '10.0.0.1', '10.0.0.2', 'RDP', 'admin', '********', 'Dell PowerEdge R750', 'Ubuntu 22.04', 'Xeon Gold 6330', '128GB', '-', '1TB SSD', '1TB SSD', '2TB HDD', 'Zabbix', '8500000', '델테크놀로지스', '2024_IDC_확장품의.pdf', '운영 환경 전용'],
|
||||
'도메인': ['도메인', '(주)에이치엠', '대표홈페이지', 'hm-corp.com', '2024-01-01', '2025-01-01', '55000', '홍길동', '이부관', '가비아 자동갱신']
|
||||
};
|
||||
|
||||
tabConfigs.forEach(config => {
|
||||
const data = [config.headers];
|
||||
if (sampleData[config.name]) {
|
||||
data.push(sampleData[config.name]);
|
||||
}
|
||||
const ws = XLSX.utils.aoa_to_sheet(data);
|
||||
ws['!cols'] = Array(config.headers.length).fill({ wch: 20 });
|
||||
const keys = DB_MAPPING[config.key];
|
||||
const headers = keys.map(k => ASSET_SCHEMA[k].ui);
|
||||
const ws = XLSX.utils.aoa_to_sheet([headers]);
|
||||
ws['!cols'] = Array(headers.length).fill({ wch: 20 });
|
||||
XLSX.utils.book_append_sheet(wb, ws, config.name);
|
||||
});
|
||||
|
||||
XLSX.writeFile(wb, 'itam_assets_template.xlsx');
|
||||
XLSX.writeFile(wb, 'itam_template_db_aligned.xlsx');
|
||||
}
|
||||
|
||||
export function exportToExcel(masterData: MasterAssetData) {
|
||||
const wb = XLSX.utils.book_new();
|
||||
const exportMap = [
|
||||
{ tab: '개인PC', list: masterData.pc, headers: PC_HEADERS, map: (a: any) => [a.법인, a.자산코드, a.구매연월, a.사용자, a.현사용조직, a.이전사용조직, a.위치, a.담당자_정, a.담당자_부, a.모델명, a.OS, a.CPU, a.GPU, a.RAM, a.SSD1, a.SSD2, a.SSD3, a.메인보드, a.IP주소, a.금액, a.납품업체, a.품의서명, a.비고] },
|
||||
{ tab: '서버', list: masterData.server, headers: SERVER_HEADERS, map: (a: any) => [a.법인, a.자산코드, a.구매연월, a.type, a.상세용도, a.상세, a.현사용조직, a.이전사용조직, a.위치, a.담당자_정, a.담당자_부, a.IP주소, a.IP2, a.원격접속, a.서버ID, a.서버PW, a.모델명, a.OS, a.CPU, a.RAM, a.GPU, a.SSD1, a.SSD2, a.SSD3, a.모니터링, a.금액, a.납품업체, a.품의서명, a.비고] },
|
||||
{ tab: '스토리지', list: masterData.storage, headers: STORAGE_HEADERS, map: (a: any) => [a.법인, a.상세용도, a.자산코드, a.명칭, a.위치, a.모델명, a.용량, a.담당자_정, a.담당자_부, a.IP주소, a.MACaddress, a.구매연월, a.금액, a.납품업체, a.품의서명, a.비고] },
|
||||
{ tab: '전산비품', list: masterData.equip, headers: EQUIP_HEADERS, map: (a: any) => [a.법인, a.상세용도, a.자산코드, a.명칭, a.위치, a.관리자, a.IP주소, a.MACaddress, a.HW사양, a.OS, a.구매연월, a.금액, a.납품업체, a.품의서명, a.비고] },
|
||||
{ tab: '모바일기기', list: masterData.mobile, headers: MOBILE_HEADERS, map: (a: any) => [a.법인, a.자산코드, a.명칭, a.위치, a.관리자, a.상세용도, a.OS, a.구매연월, a.금액, a.납품업체, a.품의서명, a.비고] },
|
||||
{ tab: '구독SW', list: masterData.subSw, headers: SUB_SW_HEADERS, map: (a: any) => [a.분야, a.법인, a.제품명, a.부서, a.수량, a.금액, a.구매일, a.납품업체, a.시작일, a.만료일, a.라이선스유형, a.계정명, a.비고] },
|
||||
{ tab: '영구SW', list: masterData.permSw, headers: PERM_SW_HEADERS, map: (a: any) => [a.분야, a.법인, a.제품명, a.부서, a.수량, a.금액, a.구매일, a.납품업체, a.시작일, a.만료일, a.라이선스키, a.계정명, a.비고] },
|
||||
{ tab: '클라우드', list: masterData.cloud, headers: CLOUD_HEADERS, map: (a: any) => [a.플랫폼명, a.법인, a.제품명, a.부서, a.계정명, a.결제수단, a.결제일, a.연결카드번호, a.당월청구액, a.비고] },
|
||||
{ tab: '도메인', list: masterData.domain || [], headers: DOMAIN_HEADERS, map: (a: any) => [a.type, a.corp, a.service_name, a.domain_name, a.start_date, a.expiry_date, a.price, a.manager_main, a.manager_sub, a.remarks] }
|
||||
|
||||
const exportConfigs = [
|
||||
{ name: 'PC', list: masterData.pc, key: 'pc' },
|
||||
{ name: '서버', list: masterData.server, key: 'server' },
|
||||
{ name: '스토리지', list: masterData.storage, key: 'storage' },
|
||||
{ name: '공간정보장비', list: masterData.survey || [], key: 'survey' },
|
||||
{ name: 'PC부품', list: masterData.pcParts || [], key: 'pcParts' },
|
||||
{ name: '네트워크', list: masterData.network || [], key: 'network' },
|
||||
{ name: '업무지원장비', list: masterData.equipment || [], key: 'equipment' },
|
||||
{ name: '내부SW', list: masterData.swInternal, key: 'swInternal' },
|
||||
{ name: '외부SW', list: masterData.swExternal, key: 'swExternal' },
|
||||
{ name: '클라우드', list: masterData.cloud || [], key: 'cloud' },
|
||||
{ name: '도메인', list: masterData.domain || [], key: 'domain' },
|
||||
{ name: '비용관리', list: masterData.cost || [], key: 'cost' },
|
||||
{ name: '선물', list: masterData.vip || [], key: 'vip' },
|
||||
{ name: '시설자산', list: masterData.officeSupplies || [], key: 'officeSupplies' }
|
||||
];
|
||||
|
||||
exportMap.forEach(m => {
|
||||
const ws = XLSX.utils.aoa_to_sheet([m.headers, ...m.list.map(m.map)]);
|
||||
XLSX.utils.book_append_sheet(wb, ws, m.tab);
|
||||
exportConfigs.forEach(config => {
|
||||
const schemaKeys = DB_MAPPING[config.key];
|
||||
const headers = schemaKeys.map(k => ASSET_SCHEMA[k].ui);
|
||||
const rows = config.list.map(asset =>
|
||||
schemaKeys.map(k => {
|
||||
const dbField = ASSET_SCHEMA[k].db;
|
||||
return asset[dbField] || asset[ASSET_SCHEMA[k].key] || '';
|
||||
})
|
||||
);
|
||||
|
||||
const ws = XLSX.utils.aoa_to_sheet([headers, ...rows]);
|
||||
XLSX.utils.book_append_sheet(wb, ws, config.name);
|
||||
});
|
||||
XLSX.writeFile(wb, `itam_master_${new Date().toISOString().split('T')[0]}.xlsx`);
|
||||
|
||||
XLSX.writeFile(wb, `itam_export_${new Date().toISOString().split('T')[0]}.xlsx`);
|
||||
}
|
||||
|
||||
/**
|
||||
* 엑셀 날짜 데이터(숫자 또는 문자열)를 YYYY-MM-DD 형식의 문자열로 변환
|
||||
*/
|
||||
export function formatExcelDate(val: any): string {
|
||||
if (!val) return '';
|
||||
if (typeof val === 'number') {
|
||||
// 엑셀 날짜 숫자 (1899-12-30 기준 일수)
|
||||
const date = new Date(Math.round((val - 25569) * 86400 * 1000));
|
||||
return date.toISOString().split('T')[0];
|
||||
}
|
||||
// 이미 문자열인 경우 기호 통일 (YYYY.MM.DD -> YYYY-MM-DD)
|
||||
if (typeof val === 'string') {
|
||||
return val.replace(/\./g, '-').trim();
|
||||
}
|
||||
return val ? String(val) : '';
|
||||
return String(val);
|
||||
}
|
||||
|
||||
export async function parseExcel(file: File): Promise<any> {
|
||||
@@ -163,44 +228,41 @@ export async function parseExcel(file: File): Promise<any> {
|
||||
const workbook = XLSX.read(e.target?.result, { type: 'array' });
|
||||
const parsedData: any = {};
|
||||
|
||||
workbook.SheetNames.forEach(rawSheetName => {
|
||||
const sheetName = rawSheetName.trim();
|
||||
const ws = workbook.Sheets[rawSheetName];
|
||||
workbook.SheetNames.forEach(sheetName => {
|
||||
const ws = workbook.Sheets[sheetName];
|
||||
const rows = XLSX.utils.sheet_to_json(ws, { defval: "" }) as any[];
|
||||
const list: any[] = [];
|
||||
|
||||
rows.forEach(rawR => {
|
||||
// 헤더명에 공백이 포함된 경우 대비하여 키 정리 (trim)
|
||||
const r: any = {};
|
||||
Object.keys(rawR).forEach(k => { r[k.trim()] = rawR[k]; });
|
||||
rows.forEach(r => {
|
||||
const data: any = { id: Math.random().toString(36).substring(2, 9) };
|
||||
|
||||
const common = { id: Math.random().toString(36).substring(2, 9) };
|
||||
if (sheetName === '개인PC') {
|
||||
const purchaseYM = formatExcelDate(r['구매연월']).replace(/-/g, '').substring(0, 6);
|
||||
list.push({ ...common, type: '개인PC', 법인: r['법인']||'', 자산코드: r['자산코드']||'', 구매연월: purchaseYM, 사용자: r['사용자']||'', 현사용조직: r['현사용조직']||'', 이전사용조직: r['이전사용조직']||'', 위치: r['위치']||'', 담당자_정: r['담당자(정)']||'', 담당자_부: r['담당자(부)']||'', 모델명: r['모델명']||'', OS: r['OS']||'', CPU: r['CPU']||'', GPU: r['GPU']||'', RAM: r['RAM']||'', SSD1: r['SSD1']||'', SSD2: r['SSD2']||'', SSD3: r['SSD3']||'', 메인보드: r['메인보드']||'', IP주소: r['IP주소']||'', 금액: r['금액']||'', 납품업체: r['납품업체']||'', 품의서명: r['품의서명']||'', 비고: r['비고']||'' });
|
||||
} else if (sheetName === '서버') {
|
||||
const purchaseYM = formatExcelDate(r['구매연월']).replace(/-/g, '').substring(0, 6);
|
||||
list.push({ ...common, type: '서버', 법인: r['법인']||'', 자산코드: r['자산코드']||'', 구매연월: purchaseYM, 상세용도: r['용도']||'', 상세: r['상세내용']||'', 현사용조직: r['현사용조직']||'', 이전사용조직: r['이전사용조직']||'', 위치: r['위치']||'', 담당자_정: r['담당자(정)']||'', 담당자_부: r['담당자(부)']||'', IP주소: r['IP 주소 1']||'', IP2: r['IP 주소 2']||'', 원격접속: r['원격도구']||'', 서버ID: r['서버 ID']||'', 서버PW: r['서버 PW']||'', 모델명: r['모델명']||'', OS: r['OS']||'', CPU: r['CPU']||'', RAM: r['RAM']||'', GPU: r['GPU']||'', SSD1: r['Storage 1']||'', SSD2: r['Storage 2']||'', SSD3: r['Storage 3']||'', 모니터링: r['모니터링']||'', 금액: r['금액']||'', 납품업체: r['납품업체']||'', 품의서명: r['품의서명']||'', 비고: r['비고']||'', type2: r['유형']||'물리' });
|
||||
} else if (sheetName === '스토리지') {
|
||||
const purchaseYM = formatExcelDate(r['구매연월']).replace(/-/g, '').substring(0, 6);
|
||||
list.push({ ...common, type: '스토리지', 법인: r['법인']||'', storage유형: r['유형']||'', 자산코드: r['자산코드']||'', 명칭: r['명칭']||'', 위치: r['위치']||'', 모델명: r['모델명']||'', 용량: r['용량']||'', 담당자_정: r['담당자(정)']||'', 담당자_부: r['담당자(부)']||'', IP주소: r['IP주소']||'', MACaddress: r['MAC주소']||'', 구매연월: purchaseYM, 금액: r['금액']||'', 납품업체: r['납품업체']||'', 품의서명: r['품의서명']||'', 비고: r['비고']||'' });
|
||||
} else if (sheetName === '전산비품') {
|
||||
const purchaseYM = formatExcelDate(r['구매연월']).replace(/-/g, '').substring(0, 6);
|
||||
list.push({ ...common, type: '전산비품', 법인: r['법인']||'', 비품유형: r['비품유형']||'', 자산코드: r['자산코드']||'', 명칭: r['명칭']||'', 위치: r['위치']||'', 관리자: r['관리자']||'', IP주소: r['IP주소']||'', MACaddress: r['MACaddress']||'', HW사양: r['HW사양']||'', OS: r['OS']||'', 구매연월: purchaseYM, 금액: r['금액']||'', 납품업체: r['납품업체']||'', 품의서명: r['품의서명']||'', 비고: r['비고']||'' });
|
||||
} else if (sheetName === '모바일기기') {
|
||||
const purchaseYM = formatExcelDate(r['구매연월']).replace(/-/g, '').substring(0, 6);
|
||||
list.push({ ...common, type: '모바일기기', 법인: r['법인']||'', 자산코드: r['자산코드']||'', 명칭: r['명칭']||'', 위치: r['위치']||'', 관리자: r['관리자']||'', 기기유형: r['기기유형']||'', OS: r['OS']||'', 구매연월: purchaseYM, 금액: r['금액']||'', 납품업체: r['납품업체']||'', 품의서명: r['품의서명']||'', 비고: r['비고']||'' });
|
||||
} else if (sheetName === '구독SW') {
|
||||
list.push({ ...common, type: '구독SW', 분야: r['분야']||'', 법인: r['법인']||'', 부서: r['부서']||'', 제품명: r['제품명']||'', 구매일: formatExcelDate(r['구매일']), 시작일: formatExcelDate(r['시작일']), 만료일: formatExcelDate(r['만료일']), 라이선스유형: r['라이선스유형']||'', 금액: r['금액']||'', 수량: parseInt(r['수량']||'1'), 계정명: r['계정명']||'', 납품업체: r['납품업체']||'', 비고: r['비고']||'' });
|
||||
} else if (sheetName === '영구SW') {
|
||||
list.push({ ...common, type: '영구SW', 분야: r['분야']||'', 법인: r['법인']||'', 부서: r['부서']||'', 제품명: r['제품명']||'', 구매일: formatExcelDate(r['구매일']), 시작일: formatExcelDate(r['시작일']), 만료일: formatExcelDate(r['만료일']), 라이선스키: r['라이선스키']||'', 금액: r['금액']||'', 수량: parseInt(r['수량']||'1'), 계정명: r['계정명']||'', 납품업체: r['납품업체']||'', 비고: r['비고']||'' });
|
||||
} else if (sheetName === '클라우드') {
|
||||
list.push({ ...common, type: '클라우드', 플랫폼명: r['플랫폼명']||'', 법인: r['법인']||'', 부서: r['부서']||'', 제품명: r['제품명']||'', 계정명: r['계정명']||'', 결제수단: r['결제수단']||'', 결제일: r['결제일']||'', 연결카드번호: r['연결카드번호']||'', 당월청구액: r['당월청구액']||'', 비고: r['비고']||'' });
|
||||
} else if (sheetName === '도메인') {
|
||||
list.push({ ...common, type: r['유형']||'도메인', corp: r['법인']||'', service_name: r['서비스명']||'', domain_name: r['관리도메인']||'', start_date: formatExcelDate(r['시작일']), expiry_date: formatExcelDate(r['만료일']), price: r['금액']||'', manager_main: r['담당자']||'', manager_sub: r['담당자(부)']||'', remarks: r['비고']||'' });
|
||||
}
|
||||
// Set default category based on sheet name
|
||||
data['category'] = sheetName;
|
||||
|
||||
Object.keys(r).forEach(label => {
|
||||
const schemaEntry = Object.values(ASSET_SCHEMA).find(s => s.ui === label);
|
||||
const key = schemaEntry ? schemaEntry.db : label;
|
||||
let val = r[label];
|
||||
|
||||
if (label.includes('일자') || label.includes('연월') || label.includes('만료일') || label.includes('시작일')) {
|
||||
val = formatExcelDate(val);
|
||||
}
|
||||
data[key] = val;
|
||||
});
|
||||
|
||||
list.push(data);
|
||||
});
|
||||
if (list.length > 0) parsedData[sheetName] = list;
|
||||
|
||||
// Sheet Name Mapping back to state keys
|
||||
const nameMap: Record<string, string> = {
|
||||
'PC': 'pc', '서버': 'server', '스토리지': 'storage', '공간정보장비': 'survey',
|
||||
'PC부품': 'pcParts', '네트워크': 'network', '업무지원장비': 'equipment',
|
||||
'내부SW': 'swInternal', '외부SW': 'swExternal', '클라우드': 'cloud',
|
||||
'도메인': 'domain', '비용관리': 'cost', '선물': 'vip', '시설자산': 'officeSupplies'
|
||||
};
|
||||
|
||||
const stateKey = nameMap[sheetName] || sheetName;
|
||||
if (list.length > 0) parsedData[stateKey] = list;
|
||||
});
|
||||
resolve(parsedData);
|
||||
} catch (err) { reject(err); }
|
||||
|
||||
Reference in New Issue
Block a user