@clazic/kordoc 2.4.19 → 2.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/chunk-AEWWERJ5.js +35 -0
- package/dist/chunk-AEWWERJ5.js.map +1 -0
- package/dist/chunk-CPTOBJJD.js +125 -0
- package/dist/chunk-CPTOBJJD.js.map +1 -0
- package/dist/{chunk-MZN7PLTZ.js → chunk-IJGNPAK2.js} +2 -2
- package/dist/{chunk-MZN7PLTZ.js.map → chunk-IJGNPAK2.js.map} +1 -1
- package/dist/{chunk-463YQ2WL.js → chunk-NKUNXGWI.js} +4 -6
- package/dist/chunk-NKUNXGWI.js.map +1 -0
- package/dist/chunk-THBLCND6.js +33 -0
- package/dist/chunk-THBLCND6.js.map +1 -0
- package/dist/cli.js +37 -5
- package/dist/cli.js.map +1 -1
- package/dist/doctor-SJ7NYSXC.js +126 -0
- package/dist/doctor-SJ7NYSXC.js.map +1 -0
- package/dist/index.cjs +2591 -82
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +1 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.js +2604 -94
- package/dist/index.js.map +1 -1
- package/dist/install-commands-P2KTEXQ4.js +11 -0
- package/dist/mcp.js +5 -2
- package/dist/mcp.js.map +1 -1
- package/dist/pm-7KGLH6MX.js +9 -0
- package/dist/pm-7KGLH6MX.js.map +1 -0
- package/dist/setup/doctor.cjs +308 -0
- package/dist/setup/doctor.js +288 -0
- package/dist/{utils-YUAT7LFD.js → utils-RBXHHCLI.js} +2 -2
- package/dist/utils-RBXHHCLI.js.map +1 -0
- package/dist/{watch-WEOFVVDO.js → watch-FRLS6FKE.js} +6 -3
- package/dist/{watch-WEOFVVDO.js.map → watch-FRLS6FKE.js.map} +1 -1
- package/package.json +7 -3
- package/scripts/postinstall.cjs +27 -0
- package/dist/chunk-463YQ2WL.js.map +0 -1
- /package/dist/{utils-YUAT7LFD.js.map → install-commands-P2KTEXQ4.js.map} +0 -0
|
@@ -0,0 +1,288 @@
|
|
|
1
|
+
var __require = /* @__PURE__ */ ((x) => typeof require !== "undefined" ? require : typeof Proxy !== "undefined" ? new Proxy(x, {
|
|
2
|
+
get: (a, b) => (typeof require !== "undefined" ? require : a)[b]
|
|
3
|
+
}) : x)(function(x) {
|
|
4
|
+
if (typeof require !== "undefined") return require.apply(this, arguments);
|
|
5
|
+
throw Error('Dynamic require of "' + x + '" is not supported');
|
|
6
|
+
});
|
|
7
|
+
|
|
8
|
+
// src/setup/detect.ts
|
|
9
|
+
import { spawnSync } from "child_process";
|
|
10
|
+
import { existsSync } from "fs";
|
|
11
|
+
import { join } from "path";
|
|
12
|
+
var isWin = process.platform === "win32";
|
|
13
|
+
function which(cmd) {
|
|
14
|
+
const finder = isWin ? "where" : "which";
|
|
15
|
+
const r = spawnSync(finder, [cmd], { encoding: "utf8", shell: isWin });
|
|
16
|
+
if (r.status !== 0) return void 0;
|
|
17
|
+
return r.stdout.split(/\r?\n/).find(Boolean)?.trim();
|
|
18
|
+
}
|
|
19
|
+
function resolveCmd(cmd, envKey) {
|
|
20
|
+
const override = process.env[envKey];
|
|
21
|
+
if (override) return override;
|
|
22
|
+
const found = which(cmd);
|
|
23
|
+
if (found) return found;
|
|
24
|
+
if (isWin) return findWinCandidate(cmd);
|
|
25
|
+
return void 0;
|
|
26
|
+
}
|
|
27
|
+
function findWinCandidate(cmd) {
|
|
28
|
+
const exe = cmd.endsWith(".exe") ? cmd : `${cmd}.exe`;
|
|
29
|
+
const userProfile = process.env["USERPROFILE"] ?? "C:\\Users\\Default";
|
|
30
|
+
const localAppData = process.env["LOCALAPPDATA"] ?? join(userProfile, "AppData", "Local");
|
|
31
|
+
const programFiles = process.env["ProgramFiles"] ?? "C:\\Program Files";
|
|
32
|
+
const programFilesX86 = process.env["ProgramFiles(x86)"] ?? "C:\\Program Files (x86)";
|
|
33
|
+
const isPopplerExe = ["pdftoppm.exe", "pdfinfo.exe", "pdftotext.exe"].includes(exe);
|
|
34
|
+
const isSoffice = exe === "soffice.exe";
|
|
35
|
+
if (isPopplerExe) return findPopplerWin(exe, { userProfile, localAppData, programFiles, programFilesX86 });
|
|
36
|
+
if (isSoffice) return findSofficeWin({ programFiles, programFilesX86, localAppData });
|
|
37
|
+
return void 0;
|
|
38
|
+
}
|
|
39
|
+
function findPopplerWin(exe, p) {
|
|
40
|
+
const scoopPath = join(p.userProfile, "scoop", "apps", "poppler", "current", "bin", exe);
|
|
41
|
+
if (existsSync(scoopPath)) return scoopPath;
|
|
42
|
+
const wingetBase = join(p.localAppData, "Microsoft", "WinGet", "Packages");
|
|
43
|
+
const wingetResult = globFirst(wingetBase, ["oschwartz10612.Poppler_*"], ["poppler-*", "Library", "bin", exe]);
|
|
44
|
+
if (wingetResult) return wingetResult;
|
|
45
|
+
const chocoBase = join("C:", "ProgramData", "chocolatey", "lib", "poppler", "tools");
|
|
46
|
+
const chocoResult = globFirst(chocoBase, ["poppler-*"], ["bin", exe]);
|
|
47
|
+
if (chocoResult) return chocoResult;
|
|
48
|
+
for (const base of [
|
|
49
|
+
join("C:", "poppler", "bin", exe),
|
|
50
|
+
join("C:", "tools", "poppler", "bin", exe),
|
|
51
|
+
join(p.programFiles, "poppler", "bin", exe),
|
|
52
|
+
join(p.programFilesX86, "poppler", "bin", exe),
|
|
53
|
+
join(p.localAppData, "Programs", "poppler", "bin", exe)
|
|
54
|
+
]) {
|
|
55
|
+
if (existsSync(base)) return base;
|
|
56
|
+
}
|
|
57
|
+
return void 0;
|
|
58
|
+
}
|
|
59
|
+
function findSofficeWin(p) {
|
|
60
|
+
for (const candidate of [
|
|
61
|
+
join(p.programFiles, "LibreOffice", "program", "soffice.exe"),
|
|
62
|
+
join(p.programFilesX86, "LibreOffice", "program", "soffice.exe"),
|
|
63
|
+
join(p.localAppData, "Programs", "LibreOffice", "program", "soffice.exe"),
|
|
64
|
+
// winget 설치는 Program Files와 동일
|
|
65
|
+
join(p.programFiles, "LibreOffice 24", "program", "soffice.exe"),
|
|
66
|
+
join(p.programFiles, "LibreOffice 25", "program", "soffice.exe")
|
|
67
|
+
]) {
|
|
68
|
+
if (existsSync(candidate)) return candidate;
|
|
69
|
+
}
|
|
70
|
+
return void 0;
|
|
71
|
+
}
|
|
72
|
+
function globFirst(base, wildcardSegments, rest) {
|
|
73
|
+
if (!existsSync(base)) return void 0;
|
|
74
|
+
let current = base;
|
|
75
|
+
for (const seg of wildcardSegments) {
|
|
76
|
+
if (!seg.includes("*")) {
|
|
77
|
+
current = join(current, seg);
|
|
78
|
+
if (!existsSync(current)) return void 0;
|
|
79
|
+
continue;
|
|
80
|
+
}
|
|
81
|
+
try {
|
|
82
|
+
const { readdirSync } = __require("fs");
|
|
83
|
+
const entries = readdirSync(current);
|
|
84
|
+
const pattern = seg.replace(/\*/g, ".*");
|
|
85
|
+
const re = new RegExp(`^${pattern}$`, "i");
|
|
86
|
+
const match = entries.find((e) => re.test(e));
|
|
87
|
+
if (!match) return void 0;
|
|
88
|
+
current = join(current, match);
|
|
89
|
+
} catch {
|
|
90
|
+
return void 0;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
const final = join(current, ...rest);
|
|
94
|
+
return existsSync(final) ? final : void 0;
|
|
95
|
+
}
|
|
96
|
+
function probe(cmd, envKey, versionArgs = ["--version"]) {
|
|
97
|
+
const resolvedPath = resolveCmd(cmd, envKey);
|
|
98
|
+
if (!resolvedPath) return { name: cmd, found: false };
|
|
99
|
+
const v = spawnSync(resolvedPath, versionArgs, { encoding: "utf8", shell: isWin });
|
|
100
|
+
const versionLine = (v.stdout || v.stderr).split(/\r?\n/)[0]?.trim();
|
|
101
|
+
return { name: cmd, found: true, path: resolvedPath, version: versionLine };
|
|
102
|
+
}
|
|
103
|
+
function probeOcrToolchain() {
|
|
104
|
+
return {
|
|
105
|
+
// poppler는 --version이 비표준이므로 -v 사용
|
|
106
|
+
pdftoppm: probe("pdftoppm", "KORDOC_PDFTOPPM_PATH", ["-v"]),
|
|
107
|
+
pdfinfo: probe("pdfinfo", "KORDOC_PDFINFO_PATH", ["-v"]),
|
|
108
|
+
pdftotext: probe("pdftotext", "KORDOC_PDFTOTEXT_PATH", ["-v"]),
|
|
109
|
+
soffice: probe("soffice", "KORDOC_SOFFICE_PATH", ["--version"])
|
|
110
|
+
};
|
|
111
|
+
}
|
|
112
|
+
function isWSL() {
|
|
113
|
+
if (process.platform !== "linux") return false;
|
|
114
|
+
try {
|
|
115
|
+
const { readFileSync } = __require("fs");
|
|
116
|
+
const version = readFileSync("/proc/version", "utf8");
|
|
117
|
+
return /microsoft|wsl/i.test(version);
|
|
118
|
+
} catch {
|
|
119
|
+
return false;
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
// src/setup/pm.ts
|
|
124
|
+
import { spawnSync as spawnSync2 } from "child_process";
|
|
125
|
+
var isWin2 = process.platform === "win32";
|
|
126
|
+
function which2(cmd) {
|
|
127
|
+
const finder = isWin2 ? "where" : "which";
|
|
128
|
+
const r = spawnSync2(finder, [cmd], { encoding: "utf8", shell: isWin2 });
|
|
129
|
+
return r.status === 0;
|
|
130
|
+
}
|
|
131
|
+
function detectPackageManager() {
|
|
132
|
+
if (process.platform === "darwin") {
|
|
133
|
+
return which2("brew") ? "brew" : "unknown";
|
|
134
|
+
}
|
|
135
|
+
if (isWin2) {
|
|
136
|
+
if (which2("winget")) return "winget";
|
|
137
|
+
if (which2("scoop")) return "scoop";
|
|
138
|
+
if (which2("choco")) return "choco";
|
|
139
|
+
return "unknown";
|
|
140
|
+
}
|
|
141
|
+
if (which2("apt-get") || which2("apt")) return "apt";
|
|
142
|
+
if (which2("dnf")) return "dnf";
|
|
143
|
+
if (which2("yum")) return "yum";
|
|
144
|
+
if (which2("pacman")) return "pacman";
|
|
145
|
+
if (which2("zypper")) return "zypper";
|
|
146
|
+
if (which2("apk")) return "apk";
|
|
147
|
+
return "unknown";
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
// src/setup/install-commands.ts
|
|
151
|
+
var POPPLER_RECIPES = {
|
|
152
|
+
brew: { pm: "brew", cmd: "brew install poppler", needsSudo: false },
|
|
153
|
+
winget: { pm: "winget", cmd: "winget install --id oschwartz10612.Poppler", needsSudo: false, notes: "\uC2E4\uC874 \uD655\uC778\uB428 (winget-pkgs 25.07.0-0)" },
|
|
154
|
+
scoop: { pm: "scoop", cmd: "scoop install poppler", needsSudo: false, notes: "main \uBC84\uD0B7 \uD3EC\uD568, PATH \uC790\uB3D9 \uB4F1\uB85D" },
|
|
155
|
+
choco: { pm: "choco", cmd: "choco install poppler -y", needsSudo: true, notes: "\uAD00\uB9AC\uC790 PowerShell \uD544\uC694" },
|
|
156
|
+
apt: { pm: "apt", cmd: "sudo apt-get install -y poppler-utils", needsSudo: true },
|
|
157
|
+
dnf: { pm: "dnf", cmd: "sudo dnf install -y poppler-utils", needsSudo: true },
|
|
158
|
+
yum: { pm: "yum", cmd: "sudo yum install -y poppler-utils", needsSudo: true },
|
|
159
|
+
pacman: { pm: "pacman", cmd: "sudo pacman -S --noconfirm poppler", needsSudo: true },
|
|
160
|
+
zypper: { pm: "zypper", cmd: "sudo zypper install -y poppler-tools", needsSudo: true },
|
|
161
|
+
apk: { pm: "apk", cmd: "apk add --no-cache poppler-utils", needsSudo: false, notes: "Alpine/Docker \uCE5C\uD654" },
|
|
162
|
+
unknown: null
|
|
163
|
+
};
|
|
164
|
+
var LIBREOFFICE_RECIPES = {
|
|
165
|
+
brew: { pm: "brew", cmd: "brew install --cask libreoffice", needsSudo: false },
|
|
166
|
+
winget: { pm: "winget", cmd: "winget install --id TheDocumentFoundation.LibreOffice", needsSudo: false, notes: "\uC2E4\uC874 \uD655\uC778\uB428" },
|
|
167
|
+
scoop: { pm: "scoop", cmd: "scoop bucket add extras && scoop install libreoffice", needsSudo: false },
|
|
168
|
+
choco: { pm: "choco", cmd: "choco install libreoffice-fresh -y", needsSudo: true, notes: "\uAD00\uB9AC\uC790 PowerShell \uD544\uC694" },
|
|
169
|
+
apt: { pm: "apt", cmd: "sudo apt-get install -y libreoffice", needsSudo: true },
|
|
170
|
+
dnf: { pm: "dnf", cmd: "sudo dnf install -y libreoffice", needsSudo: true },
|
|
171
|
+
yum: { pm: "yum", cmd: "sudo yum install -y libreoffice", needsSudo: true },
|
|
172
|
+
pacman: { pm: "pacman", cmd: "sudo pacman -S --noconfirm libreoffice-still", needsSudo: true },
|
|
173
|
+
zypper: { pm: "zypper", cmd: "sudo zypper install -y libreoffice", needsSudo: true },
|
|
174
|
+
apk: { pm: "apk", cmd: "apk add --no-cache libreoffice", needsSudo: false, notes: "Alpine \uBBF8\uC9C0\uC6D0 \uAC00\uB2A5 \u2014 node:20-slim(Debian) \uAD8C\uC7A5" },
|
|
175
|
+
unknown: null
|
|
176
|
+
};
|
|
177
|
+
|
|
178
|
+
// src/setup/doctor.ts
|
|
179
|
+
var hr = "\u2500".repeat(45);
|
|
180
|
+
function platform() {
|
|
181
|
+
const arch = process.arch;
|
|
182
|
+
if (isWSL()) return `linux/WSL (arch: ${arch})`;
|
|
183
|
+
return `${process.platform} (arch: ${arch})`;
|
|
184
|
+
}
|
|
185
|
+
function icon(found) {
|
|
186
|
+
return found ? "\u2705" : "\u274C";
|
|
187
|
+
}
|
|
188
|
+
function pad(s, n) {
|
|
189
|
+
return s.padEnd(n);
|
|
190
|
+
}
|
|
191
|
+
function formatTool(name, status) {
|
|
192
|
+
const ico = icon(status.found);
|
|
193
|
+
if (!status.found) return ` ${pad(name, 10)} ${ico} \uCC3E\uC744 \uC218 \uC5C6\uC74C`;
|
|
194
|
+
const ver = status.version ? ` (${status.version})` : "";
|
|
195
|
+
return ` ${pad(name, 10)} ${ico} ${status.path}${ver}`;
|
|
196
|
+
}
|
|
197
|
+
function runDoctor(opts = {}) {
|
|
198
|
+
const toolchain = probeOcrToolchain();
|
|
199
|
+
const pm = detectPackageManager();
|
|
200
|
+
const popplerOk = toolchain.pdftoppm.found && toolchain.pdfinfo.found;
|
|
201
|
+
const sofficeOk = toolchain.soffice.found;
|
|
202
|
+
const missing = [];
|
|
203
|
+
if (!popplerOk) missing.push("poppler");
|
|
204
|
+
if (!sofficeOk) missing.push("libreoffice");
|
|
205
|
+
const out = opts.postinstall ? process.stderr : process.stdout;
|
|
206
|
+
out.write(`${hr}
|
|
207
|
+
`);
|
|
208
|
+
out.write(`kordoc OCR \uC758\uC874\uC131 \uC810\uAC80
|
|
209
|
+
`);
|
|
210
|
+
out.write(`${hr}
|
|
211
|
+
`);
|
|
212
|
+
out.write(`\uD50C\uB7AB\uD3FC: ${platform()}
|
|
213
|
+
`);
|
|
214
|
+
out.write(`\uD328\uD0A4\uC9C0 \uB9E4\uB2C8\uC800: ${pm === "unknown" ? "\uAC10\uC9C0 \uC2E4\uD328" : pm}
|
|
215
|
+
`);
|
|
216
|
+
out.write(`Node.js: ${process.version}
|
|
217
|
+
`);
|
|
218
|
+
out.write(`
|
|
219
|
+
`);
|
|
220
|
+
out.write(`[OCR \uD30C\uC774\uD504\uB77C\uC778 \uC758\uC874\uC131]
|
|
221
|
+
`);
|
|
222
|
+
out.write(`${formatTool("pdftoppm", toolchain.pdftoppm)}
|
|
223
|
+
`);
|
|
224
|
+
out.write(`${formatTool("pdfinfo", toolchain.pdfinfo)}
|
|
225
|
+
`);
|
|
226
|
+
out.write(`${formatTool("pdftotext", toolchain.pdftotext)}
|
|
227
|
+
`);
|
|
228
|
+
out.write(`${formatTool("soffice", toolchain.soffice)}
|
|
229
|
+
`);
|
|
230
|
+
if (missing.length > 0) {
|
|
231
|
+
out.write(`
|
|
232
|
+
\uAD8C\uC7A5 \uC124\uCE58 \uBA85\uB839:
|
|
233
|
+
`);
|
|
234
|
+
if (!popplerOk) {
|
|
235
|
+
const recipe = POPPLER_RECIPES[pm];
|
|
236
|
+
if (recipe) {
|
|
237
|
+
out.write(` # poppler (pdftoppm/pdfinfo)
|
|
238
|
+
`);
|
|
239
|
+
out.write(` ${recipe.cmd}
|
|
240
|
+
`);
|
|
241
|
+
if (recipe.notes) out.write(` # ${recipe.notes}
|
|
242
|
+
`);
|
|
243
|
+
} else {
|
|
244
|
+
out.write(` poppler: OS\uC5D0 \uB9DE\uB294 \uBC29\uBC95\uC73C\uB85C \uC124\uCE58\uD558\uC138\uC694 (README \uCC38\uACE0)
|
|
245
|
+
`);
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
if (!sofficeOk) {
|
|
249
|
+
const recipe = LIBREOFFICE_RECIPES[pm];
|
|
250
|
+
if (recipe) {
|
|
251
|
+
out.write(`
|
|
252
|
+
# LibreOffice (soffice)
|
|
253
|
+
`);
|
|
254
|
+
out.write(` ${recipe.cmd}
|
|
255
|
+
`);
|
|
256
|
+
if (recipe.notes) out.write(` # ${recipe.notes}
|
|
257
|
+
`);
|
|
258
|
+
} else {
|
|
259
|
+
out.write(` LibreOffice: OS\uC5D0 \uB9DE\uB294 \uBC29\uBC95\uC73C\uB85C \uC124\uCE58\uD558\uC138\uC694 (README \uCC38\uACE0)
|
|
260
|
+
`);
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
if (process.platform === "win32") {
|
|
264
|
+
out.write(`
|
|
265
|
+
\u26A0\uFE0F \uC124\uCE58 \uD6C4 \uC0C8 \uD130\uBBF8\uB110\uC744 \uC5F4\uAC70\uB098 PATH\uB97C \uAC31\uC2E0\uD574\uC57C \uD569\uB2C8\uB2E4.
|
|
266
|
+
`);
|
|
267
|
+
out.write(` PowerShell: $env:Path = [System.Environment]::GetEnvironmentVariable("Path","Machine") + ";" + [System.Environment]::GetEnvironmentVariable("Path","User")
|
|
268
|
+
`);
|
|
269
|
+
}
|
|
270
|
+
out.write(`
|
|
271
|
+
\uD658\uACBD\uBCC0\uC218\uB85C \uACBD\uB85C\uB97C \uC9C1\uC811 \uC9C0\uC815\uD560 \uC218\uB3C4 \uC788\uC2B5\uB2C8\uB2E4:
|
|
272
|
+
`);
|
|
273
|
+
out.write(` KORDOC_PDFTOPPM_PATH, KORDOC_PDFINFO_PATH, KORDOC_SOFFICE_PATH
|
|
274
|
+
`);
|
|
275
|
+
out.write(`
|
|
276
|
+
\uC790\uC138\uD55C \uC810\uAC80: npx kordoc doctor
|
|
277
|
+
`);
|
|
278
|
+
}
|
|
279
|
+
out.write(`${hr}
|
|
280
|
+
`);
|
|
281
|
+
const summary = missing.length === 0 ? "\uBAA8\uB4E0 \uC758\uC874\uC131\uC774 \uC124\uCE58\uB418\uC5B4 \uC788\uC2B5\uB2C8\uB2E4 \u2705" : `${missing.length}\uAC1C \uACB0\uC190 \u2014 OCR \uBAA8\uB4DC \uC0AC\uC6A9 \uC2DC \uCC28\uB2E8\uB429\uB2C8\uB2E4 (${missing.join(", ")})`;
|
|
282
|
+
out.write(`\uC694\uC57D: ${summary}
|
|
283
|
+
`);
|
|
284
|
+
return { ok: missing.length === 0, missing, toolchain, pm };
|
|
285
|
+
}
|
|
286
|
+
export {
|
|
287
|
+
runDoctor
|
|
288
|
+
};
|
|
@@ -9,7 +9,7 @@ import {
|
|
|
9
9
|
sanitizeError,
|
|
10
10
|
sanitizeHref,
|
|
11
11
|
toArrayBuffer
|
|
12
|
-
} from "./chunk-
|
|
12
|
+
} from "./chunk-IJGNPAK2.js";
|
|
13
13
|
import "./chunk-ZWE3DS7E.js";
|
|
14
14
|
export {
|
|
15
15
|
KordocError,
|
|
@@ -22,4 +22,4 @@ export {
|
|
|
22
22
|
sanitizeHref,
|
|
23
23
|
toArrayBuffer
|
|
24
24
|
};
|
|
25
|
-
//# sourceMappingURL=utils-
|
|
25
|
+
//# sourceMappingURL=utils-RBXHHCLI.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
|
|
@@ -2,10 +2,13 @@
|
|
|
2
2
|
import {
|
|
3
3
|
detectFormat,
|
|
4
4
|
parse
|
|
5
|
-
} from "./chunk-
|
|
5
|
+
} from "./chunk-NKUNXGWI.js";
|
|
6
|
+
import "./chunk-CPTOBJJD.js";
|
|
7
|
+
import "./chunk-THBLCND6.js";
|
|
8
|
+
import "./chunk-AEWWERJ5.js";
|
|
6
9
|
import {
|
|
7
10
|
toArrayBuffer
|
|
8
|
-
} from "./chunk-
|
|
11
|
+
} from "./chunk-IJGNPAK2.js";
|
|
9
12
|
import "./chunk-MOL7MDBG.js";
|
|
10
13
|
import "./chunk-Y4WFKJ5P.js";
|
|
11
14
|
import "./chunk-YW5G6BCJ.js";
|
|
@@ -136,4 +139,4 @@ async function sendWebhook(url, payload) {
|
|
|
136
139
|
export {
|
|
137
140
|
watchDirectory
|
|
138
141
|
};
|
|
139
|
-
//# sourceMappingURL=watch-
|
|
142
|
+
//# sourceMappingURL=watch-FRLS6FKE.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/watch.ts"],"sourcesContent":["/** 디렉토리 감시 모드 — 새 문서 자동 변환 + Webhook 알림 */\n\nimport { watch, readFileSync, writeFileSync, mkdirSync, statSync, existsSync } from \"fs\"\nimport { basename, resolve, extname, sep } from \"path\"\nimport { parse, detectFormat } from \"./index.js\"\nimport { toArrayBuffer } from \"./utils.js\"\nimport type { WatchOptions } from \"./types.js\"\n\nconst SUPPORTED_EXTENSIONS = new Set([\".hwp\", \".hwpx\", \".pdf\", \".xlsx\", \".docx\"])\nconst DEBOUNCE_MS = 1000\n/** 파일 쓰기 완료 판정: 연속 2회 동일 크기 확인 간격 */\nconst STABLE_CHECK_MS = 300\nconst MAX_FILE_SIZE = 500 * 1024 * 1024\n\n/**\n * 디렉토리를 감시하여 새 문서 파일을 자동 변환.\n *\n * @example\n * ```bash\n * kordoc watch ./incoming -d ./output --webhook https://api.example.com/docs\n * ```\n */\nexport async function watchDirectory(options: WatchOptions): Promise<void> {\n const { dir, outDir, webhook, format = \"markdown\", pages, silent } = options\n\n if (!existsSync(dir)) throw new Error(`디렉토리를 찾을 수 없습니다: ${dir}`)\n if (webhook) validateWebhookUrl(webhook)\n if (outDir) mkdirSync(outDir, { recursive: true })\n\n const log = silent ? () => {} : (msg: string) => process.stderr.write(msg + \"\\n\")\n log(`[kordoc watch] 감시 시작: ${resolve(dir)}`)\n if (outDir) log(`[kordoc watch] 출력: ${resolve(outDir)}`)\n if (webhook) log(`[kordoc watch] 웹훅: ${webhook}`)\n\n // 디바운스 맵\n const pending = new Map<string, ReturnType<typeof setTimeout>>()\n\n /** 파일 크기가 안정화될 때까지 대기 (쓰기 완료 감지) */\n const waitForStableSize = async (absPath: string): Promise<number> => {\n let prevSize = statSync(absPath).size\n await new Promise(r => setTimeout(r, STABLE_CHECK_MS))\n if (!existsSync(absPath)) return 0\n const currSize = statSync(absPath).size\n if (currSize !== prevSize) {\n // 크기가 변했으면 한 번 더 대기\n await new Promise(r => setTimeout(r, STABLE_CHECK_MS))\n if (!existsSync(absPath)) return 0\n return statSync(absPath).size\n }\n return currSize\n }\n\n const processFile = async (filePath: string) => {\n const ext = extname(filePath).toLowerCase()\n if (!SUPPORTED_EXTENSIONS.has(ext)) return\n\n const fileName = basename(filePath)\n try {\n const absPath = resolve(dir, filePath)\n // 경로 순회 방지 — 감시 디렉토리 외부 파일 차단\n const realDir = resolve(dir)\n const dirWithSep = realDir.endsWith(sep) ? realDir : realDir + sep\n if (!absPath.startsWith(dirWithSep) && absPath !== realDir) return\n if (!existsSync(absPath)) return\n\n const fileSize = await waitForStableSize(absPath)\n if (fileSize > MAX_FILE_SIZE || fileSize === 0) return\n\n log(`[kordoc watch] 변환 중: ${fileName}`)\n\n const buffer = readFileSync(absPath)\n const arrayBuffer = toArrayBuffer(buffer)\n const parseOptions = pages ? { pages } : undefined\n const result = await parse(arrayBuffer, parseOptions)\n\n if (!result.success) {\n log(`[kordoc watch] 실패: ${fileName} — ${result.error}`)\n await sendWebhook(webhook, { file: fileName, format: detectFormat(arrayBuffer), success: false, error: result.error })\n return\n }\n\n const output = format === \"json\" ? JSON.stringify(result, null, 2) : result.markdown\n\n if (outDir) {\n const outExt = format === \"json\" ? \".json\" : \".md\"\n const outPath = resolve(outDir, fileName.replace(/\\.[^.]+$/, outExt))\n writeFileSync(outPath, output, \"utf-8\")\n log(`[kordoc watch] 완료: ${fileName} → ${basename(outPath)}`)\n } else {\n process.stdout.write(output + \"\\n\")\n }\n\n await sendWebhook(webhook, {\n file: fileName,\n format: result.fileType,\n success: true,\n markdown: format === \"markdown\" ? output.substring(0, 1000) : undefined,\n })\n } catch (err) {\n log(`[kordoc watch] 에러: ${fileName} — ${err instanceof Error ? err.message : err}`)\n }\n }\n\n const handler = (_event: string | null, filename: string | Buffer | null) => {\n if (!filename) return\n const filePath = filename.toString()\n const existing = pending.get(filePath)\n if (existing) clearTimeout(existing)\n pending.set(filePath, setTimeout(() => {\n pending.delete(filePath)\n processFile(filePath).catch(() => {})\n }, DEBOUNCE_MS))\n }\n\n // fs.watch recursive: Node 18+ macOS/Windows, Node 22+ Linux\n // Linux Node 22 미만은 recursive 미지원 → graceful degradation (최상위 디렉토리만 감시)\n try {\n watch(dir, { recursive: true }, handler)\n } catch {\n process.stderr.write(\"[kordoc watch] 경고: 하위 디렉토리 감시 미지원 (Node.js 22+ 또는 macOS/Windows 필요) — 최상위만 감시\\n\")\n watch(dir, handler)\n }\n\n // 프로세스 종료 방지 (Ctrl+C로 종료)\n return new Promise(() => {})\n}\n\n/** Webhook URL 검증 — SSRF 방지: http/https만 허용, localhost/private IP 차단 */\nfunction validateWebhookUrl(url: string): void {\n let parsed: URL\n try {\n parsed = new URL(url)\n } catch {\n throw new Error(`유효하지 않은 webhook URL: ${url}`)\n }\n if (parsed.protocol !== \"http:\" && parsed.protocol !== \"https:\") {\n throw new Error(`허용되지 않는 webhook 프로토콜: ${parsed.protocol}`)\n }\n const hostname = parsed.hostname.toLowerCase()\n if (\n hostname === \"localhost\" ||\n hostname === \"[::1]\" ||\n hostname.startsWith(\"127.\") ||\n hostname.startsWith(\"10.\") ||\n hostname.startsWith(\"192.168.\") ||\n /^172\\.(1[6-9]|2\\d|3[01])\\./.test(hostname) ||\n hostname === \"0.0.0.0\" ||\n hostname.startsWith(\"169.254.\") ||\n hostname.endsWith(\".local\") ||\n // IPv6 사설 대역\n hostname.startsWith(\"[fc\") ||\n hostname.startsWith(\"[fd\") ||\n hostname.startsWith(\"[fe80:\") ||\n hostname === \"[::0]\" ||\n hostname === \"[::]\" ||\n // 클라우드 메타데이터 엔드포인트\n hostname === \"metadata.google.internal\" ||\n hostname === \"metadata.google\" ||\n // 16진수/8진수 IP 인코딩 우회 방지\n /^0x[0-9a-f]+$/i.test(hostname) ||\n /^0[0-7]+$/.test(hostname)\n ) {\n throw new Error(`내부 네트워크 대상 webhook은 허용되지 않습니다: ${hostname}`)\n }\n}\n\nasync function sendWebhook(url: string | undefined, payload: Record<string, unknown>): Promise<void> {\n if (!url) return\n try {\n validateWebhookUrl(url)\n await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ ...payload, timestamp: new Date().toISOString() }),\n })\n } catch {\n // webhook 실패는 조용히 무시\n }\n}\n"],"mappings":"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","names":[]}
|
|
1
|
+
{"version":3,"sources":["../src/watch.ts"],"sourcesContent":["/** 디렉토리 감시 모드 — 새 문서 자동 변환 + Webhook 알림 */\n\nimport { watch, readFileSync, writeFileSync, mkdirSync, statSync, existsSync } from \"fs\"\nimport { basename, resolve, extname, sep } from \"path\"\nimport { parse, detectFormat } from \"./index.js\"\nimport { toArrayBuffer } from \"./utils.js\"\nimport type { WatchOptions } from \"./types.js\"\n\nconst SUPPORTED_EXTENSIONS = new Set([\".hwp\", \".hwpx\", \".pdf\", \".xlsx\", \".docx\"])\nconst DEBOUNCE_MS = 1000\n/** 파일 쓰기 완료 판정: 연속 2회 동일 크기 확인 간격 */\nconst STABLE_CHECK_MS = 300\nconst MAX_FILE_SIZE = 500 * 1024 * 1024\n\n/**\n * 디렉토리를 감시하여 새 문서 파일을 자동 변환.\n *\n * @example\n * ```bash\n * kordoc watch ./incoming -d ./output --webhook https://api.example.com/docs\n * ```\n */\nexport async function watchDirectory(options: WatchOptions): Promise<void> {\n const { dir, outDir, webhook, format = \"markdown\", pages, silent } = options\n\n if (!existsSync(dir)) throw new Error(`디렉토리를 찾을 수 없습니다: ${dir}`)\n if (webhook) validateWebhookUrl(webhook)\n if (outDir) mkdirSync(outDir, { recursive: true })\n\n const log = silent ? () => {} : (msg: string) => process.stderr.write(msg + \"\\n\")\n log(`[kordoc watch] 감시 시작: ${resolve(dir)}`)\n if (outDir) log(`[kordoc watch] 출력: ${resolve(outDir)}`)\n if (webhook) log(`[kordoc watch] 웹훅: ${webhook}`)\n\n // 디바운스 맵\n const pending = new Map<string, ReturnType<typeof setTimeout>>()\n\n /** 파일 크기가 안정화될 때까지 대기 (쓰기 완료 감지) */\n const waitForStableSize = async (absPath: string): Promise<number> => {\n let prevSize = statSync(absPath).size\n await new Promise(r => setTimeout(r, STABLE_CHECK_MS))\n if (!existsSync(absPath)) return 0\n const currSize = statSync(absPath).size\n if (currSize !== prevSize) {\n // 크기가 변했으면 한 번 더 대기\n await new Promise(r => setTimeout(r, STABLE_CHECK_MS))\n if (!existsSync(absPath)) return 0\n return statSync(absPath).size\n }\n return currSize\n }\n\n const processFile = async (filePath: string) => {\n const ext = extname(filePath).toLowerCase()\n if (!SUPPORTED_EXTENSIONS.has(ext)) return\n\n const fileName = basename(filePath)\n try {\n const absPath = resolve(dir, filePath)\n // 경로 순회 방지 — 감시 디렉토리 외부 파일 차단\n const realDir = resolve(dir)\n const dirWithSep = realDir.endsWith(sep) ? realDir : realDir + sep\n if (!absPath.startsWith(dirWithSep) && absPath !== realDir) return\n if (!existsSync(absPath)) return\n\n const fileSize = await waitForStableSize(absPath)\n if (fileSize > MAX_FILE_SIZE || fileSize === 0) return\n\n log(`[kordoc watch] 변환 중: ${fileName}`)\n\n const buffer = readFileSync(absPath)\n const arrayBuffer = toArrayBuffer(buffer)\n const parseOptions = pages ? { pages } : undefined\n const result = await parse(arrayBuffer, parseOptions)\n\n if (!result.success) {\n log(`[kordoc watch] 실패: ${fileName} — ${result.error}`)\n await sendWebhook(webhook, { file: fileName, format: detectFormat(arrayBuffer), success: false, error: result.error })\n return\n }\n\n const output = format === \"json\" ? JSON.stringify(result, null, 2) : result.markdown\n\n if (outDir) {\n const outExt = format === \"json\" ? \".json\" : \".md\"\n const outPath = resolve(outDir, fileName.replace(/\\.[^.]+$/, outExt))\n writeFileSync(outPath, output, \"utf-8\")\n log(`[kordoc watch] 완료: ${fileName} → ${basename(outPath)}`)\n } else {\n process.stdout.write(output + \"\\n\")\n }\n\n await sendWebhook(webhook, {\n file: fileName,\n format: result.fileType,\n success: true,\n markdown: format === \"markdown\" ? output.substring(0, 1000) : undefined,\n })\n } catch (err) {\n log(`[kordoc watch] 에러: ${fileName} — ${err instanceof Error ? err.message : err}`)\n }\n }\n\n const handler = (_event: string | null, filename: string | Buffer | null) => {\n if (!filename) return\n const filePath = filename.toString()\n const existing = pending.get(filePath)\n if (existing) clearTimeout(existing)\n pending.set(filePath, setTimeout(() => {\n pending.delete(filePath)\n processFile(filePath).catch(() => {})\n }, DEBOUNCE_MS))\n }\n\n // fs.watch recursive: Node 18+ macOS/Windows, Node 22+ Linux\n // Linux Node 22 미만은 recursive 미지원 → graceful degradation (최상위 디렉토리만 감시)\n try {\n watch(dir, { recursive: true }, handler)\n } catch {\n process.stderr.write(\"[kordoc watch] 경고: 하위 디렉토리 감시 미지원 (Node.js 22+ 또는 macOS/Windows 필요) — 최상위만 감시\\n\")\n watch(dir, handler)\n }\n\n // 프로세스 종료 방지 (Ctrl+C로 종료)\n return new Promise(() => {})\n}\n\n/** Webhook URL 검증 — SSRF 방지: http/https만 허용, localhost/private IP 차단 */\nfunction validateWebhookUrl(url: string): void {\n let parsed: URL\n try {\n parsed = new URL(url)\n } catch {\n throw new Error(`유효하지 않은 webhook URL: ${url}`)\n }\n if (parsed.protocol !== \"http:\" && parsed.protocol !== \"https:\") {\n throw new Error(`허용되지 않는 webhook 프로토콜: ${parsed.protocol}`)\n }\n const hostname = parsed.hostname.toLowerCase()\n if (\n hostname === \"localhost\" ||\n hostname === \"[::1]\" ||\n hostname.startsWith(\"127.\") ||\n hostname.startsWith(\"10.\") ||\n hostname.startsWith(\"192.168.\") ||\n /^172\\.(1[6-9]|2\\d|3[01])\\./.test(hostname) ||\n hostname === \"0.0.0.0\" ||\n hostname.startsWith(\"169.254.\") ||\n hostname.endsWith(\".local\") ||\n // IPv6 사설 대역\n hostname.startsWith(\"[fc\") ||\n hostname.startsWith(\"[fd\") ||\n hostname.startsWith(\"[fe80:\") ||\n hostname === \"[::0]\" ||\n hostname === \"[::]\" ||\n // 클라우드 메타데이터 엔드포인트\n hostname === \"metadata.google.internal\" ||\n hostname === \"metadata.google\" ||\n // 16진수/8진수 IP 인코딩 우회 방지\n /^0x[0-9a-f]+$/i.test(hostname) ||\n /^0[0-7]+$/.test(hostname)\n ) {\n throw new Error(`내부 네트워크 대상 webhook은 허용되지 않습니다: ${hostname}`)\n }\n}\n\nasync function sendWebhook(url: string | undefined, payload: Record<string, unknown>): Promise<void> {\n if (!url) return\n try {\n validateWebhookUrl(url)\n await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ ...payload, timestamp: new Date().toISOString() }),\n })\n } catch {\n // webhook 실패는 조용히 무시\n }\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;AAEA,SAAS,OAAO,cAAc,eAAe,WAAW,UAAU,kBAAkB;AACpF,SAAS,UAAU,SAAS,SAAS,WAAW;AAKhD,IAAM,uBAAuB,oBAAI,IAAI,CAAC,QAAQ,SAAS,QAAQ,SAAS,OAAO,CAAC;AAChF,IAAM,cAAc;AAEpB,IAAM,kBAAkB;AACxB,IAAM,gBAAgB,MAAM,OAAO;AAUnC,eAAsB,eAAe,SAAsC;AACzE,QAAM,EAAE,KAAK,QAAQ,SAAS,SAAS,YAAY,OAAO,OAAO,IAAI;AAErE,MAAI,CAAC,WAAW,GAAG,EAAG,OAAM,IAAI,MAAM,gFAAoB,GAAG,EAAE;AAC/D,MAAI,QAAS,oBAAmB,OAAO;AACvC,MAAI,OAAQ,WAAU,QAAQ,EAAE,WAAW,KAAK,CAAC;AAEjD,QAAM,MAAM,SAAS,MAAM;AAAA,EAAC,IAAI,CAAC,QAAgB,QAAQ,OAAO,MAAM,MAAM,IAAI;AAChF,MAAI,6CAAyB,QAAQ,GAAG,CAAC,EAAE;AAC3C,MAAI,OAAQ,KAAI,gCAAsB,QAAQ,MAAM,CAAC,EAAE;AACvD,MAAI,QAAS,KAAI,gCAAsB,OAAO,EAAE;AAGhD,QAAM,UAAU,oBAAI,IAA2C;AAG/D,QAAM,oBAAoB,OAAO,YAAqC;AACpE,QAAI,WAAW,SAAS,OAAO,EAAE;AACjC,UAAM,IAAI,QAAQ,OAAK,WAAW,GAAG,eAAe,CAAC;AACrD,QAAI,CAAC,WAAW,OAAO,EAAG,QAAO;AACjC,UAAM,WAAW,SAAS,OAAO,EAAE;AACnC,QAAI,aAAa,UAAU;AAEzB,YAAM,IAAI,QAAQ,OAAK,WAAW,GAAG,eAAe,CAAC;AACrD,UAAI,CAAC,WAAW,OAAO,EAAG,QAAO;AACjC,aAAO,SAAS,OAAO,EAAE;AAAA,IAC3B;AACA,WAAO;AAAA,EACT;AAEA,QAAM,cAAc,OAAO,aAAqB;AAC9C,UAAM,MAAM,QAAQ,QAAQ,EAAE,YAAY;AAC1C,QAAI,CAAC,qBAAqB,IAAI,GAAG,EAAG;AAEpC,UAAM,WAAW,SAAS,QAAQ;AAClC,QAAI;AACF,YAAM,UAAU,QAAQ,KAAK,QAAQ;AAErC,YAAM,UAAU,QAAQ,GAAG;AAC3B,YAAM,aAAa,QAAQ,SAAS,GAAG,IAAI,UAAU,UAAU;AAC/D,UAAI,CAAC,QAAQ,WAAW,UAAU,KAAK,YAAY,QAAS;AAC5D,UAAI,CAAC,WAAW,OAAO,EAAG;AAE1B,YAAM,WAAW,MAAM,kBAAkB,OAAO;AAChD,UAAI,WAAW,iBAAiB,aAAa,EAAG;AAEhD,UAAI,uCAAwB,QAAQ,EAAE;AAEtC,YAAM,SAAS,aAAa,OAAO;AACnC,YAAM,cAAc,cAAc,MAAM;AACxC,YAAM,eAAe,QAAQ,EAAE,MAAM,IAAI;AACzC,YAAM,SAAS,MAAM,MAAM,aAAa,YAAY;AAEpD,UAAI,CAAC,OAAO,SAAS;AACnB,YAAI,gCAAsB,QAAQ,WAAM,OAAO,KAAK,EAAE;AACtD,cAAM,YAAY,SAAS,EAAE,MAAM,UAAU,QAAQ,aAAa,WAAW,GAAG,SAAS,OAAO,OAAO,OAAO,MAAM,CAAC;AACrH;AAAA,MACF;AAEA,YAAM,SAAS,WAAW,SAAS,KAAK,UAAU,QAAQ,MAAM,CAAC,IAAI,OAAO;AAE5E,UAAI,QAAQ;AACV,cAAM,SAAS,WAAW,SAAS,UAAU;AAC7C,cAAM,UAAU,QAAQ,QAAQ,SAAS,QAAQ,YAAY,MAAM,CAAC;AACpE,sBAAc,SAAS,QAAQ,OAAO;AACtC,YAAI,gCAAsB,QAAQ,WAAM,SAAS,OAAO,CAAC,EAAE;AAAA,MAC7D,OAAO;AACL,gBAAQ,OAAO,MAAM,SAAS,IAAI;AAAA,MACpC;AAEA,YAAM,YAAY,SAAS;AAAA,QACzB,MAAM;AAAA,QACN,QAAQ,OAAO;AAAA,QACf,SAAS;AAAA,QACT,UAAU,WAAW,aAAa,OAAO,UAAU,GAAG,GAAI,IAAI;AAAA,MAChE,CAAC;AAAA,IACH,SAAS,KAAK;AACZ,UAAI,gCAAsB,QAAQ,WAAM,eAAe,QAAQ,IAAI,UAAU,GAAG,EAAE;AAAA,IACpF;AAAA,EACF;AAEA,QAAM,UAAU,CAAC,QAAuB,aAAqC;AAC3E,QAAI,CAAC,SAAU;AACf,UAAM,WAAW,SAAS,SAAS;AACnC,UAAM,WAAW,QAAQ,IAAI,QAAQ;AACrC,QAAI,SAAU,cAAa,QAAQ;AACnC,YAAQ,IAAI,UAAU,WAAW,MAAM;AACrC,cAAQ,OAAO,QAAQ;AACvB,kBAAY,QAAQ,EAAE,MAAM,MAAM;AAAA,MAAC,CAAC;AAAA,IACtC,GAAG,WAAW,CAAC;AAAA,EACjB;AAIA,MAAI;AACF,UAAM,KAAK,EAAE,WAAW,KAAK,GAAG,OAAO;AAAA,EACzC,QAAQ;AACN,YAAQ,OAAO,MAAM,yMAAiF;AACtG,UAAM,KAAK,OAAO;AAAA,EACpB;AAGA,SAAO,IAAI,QAAQ,MAAM;AAAA,EAAC,CAAC;AAC7B;AAGA,SAAS,mBAAmB,KAAmB;AAC7C,MAAI;AACJ,MAAI;AACF,aAAS,IAAI,IAAI,GAAG;AAAA,EACtB,QAAQ;AACN,UAAM,IAAI,MAAM,sDAAwB,GAAG,EAAE;AAAA,EAC/C;AACA,MAAI,OAAO,aAAa,WAAW,OAAO,aAAa,UAAU;AAC/D,UAAM,IAAI,MAAM,2EAAyB,OAAO,QAAQ,EAAE;AAAA,EAC5D;AACA,QAAM,WAAW,OAAO,SAAS,YAAY;AAC7C,MACE,aAAa,eACb,aAAa,WACb,SAAS,WAAW,MAAM,KAC1B,SAAS,WAAW,KAAK,KACzB,SAAS,WAAW,UAAU,KAC9B,6BAA6B,KAAK,QAAQ,KAC1C,aAAa,aACb,SAAS,WAAW,UAAU,KAC9B,SAAS,SAAS,QAAQ;AAAA,EAE1B,SAAS,WAAW,KAAK,KACzB,SAAS,WAAW,KAAK,KACzB,SAAS,WAAW,QAAQ,KAC5B,aAAa,WACb,aAAa;AAAA,EAEb,aAAa,8BACb,aAAa;AAAA,EAEb,iBAAiB,KAAK,QAAQ,KAC9B,YAAY,KAAK,QAAQ,GACzB;AACA,UAAM,IAAI,MAAM,uHAAkC,QAAQ,EAAE;AAAA,EAC9D;AACF;AAEA,eAAe,YAAY,KAAyB,SAAiD;AACnG,MAAI,CAAC,IAAK;AACV,MAAI;AACF,uBAAmB,GAAG;AACtB,UAAM,MAAM,KAAK;AAAA,MACf,QAAQ;AAAA,MACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,MAC9C,MAAM,KAAK,UAAU,EAAE,GAAG,SAAS,YAAW,oBAAI,KAAK,GAAE,YAAY,EAAE,CAAC;AAAA,IAC1E,CAAC;AAAA,EACH,QAAQ;AAAA,EAER;AACF;","names":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@clazic/kordoc",
|
|
3
|
-
"version": "2.
|
|
3
|
+
"version": "2.5.0",
|
|
4
4
|
"description": "Parse Korean documents (HWP, HWPX, PDF, XLSX, DOCX) to Markdown",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"exports": {
|
|
@@ -18,9 +18,11 @@
|
|
|
18
18
|
"kordoc-mcp": "dist/mcp.js"
|
|
19
19
|
},
|
|
20
20
|
"files": [
|
|
21
|
-
"dist"
|
|
21
|
+
"dist",
|
|
22
|
+
"scripts"
|
|
22
23
|
],
|
|
23
24
|
"scripts": {
|
|
25
|
+
"postinstall": "node scripts/postinstall.cjs",
|
|
24
26
|
"build": "tsup",
|
|
25
27
|
"dev": "tsup --watch",
|
|
26
28
|
"test": "node --import tsx --test tests/*.test.ts",
|
|
@@ -62,10 +64,12 @@
|
|
|
62
64
|
"commander": "^14.0.3",
|
|
63
65
|
"exceljs": "^4.4.0",
|
|
64
66
|
"jszip": "^3.10.1",
|
|
65
|
-
"libreoffice-convert": "^1.8.1",
|
|
66
67
|
"pdfjs-dist": "^5.6.205",
|
|
67
68
|
"zod": "^4.3.6"
|
|
68
69
|
},
|
|
70
|
+
"optionalDependencies": {
|
|
71
|
+
"libreoffice-convert": "^1.8.1"
|
|
72
|
+
},
|
|
69
73
|
"devDependencies": {
|
|
70
74
|
"@types/node": "^18.19.130",
|
|
71
75
|
"pdfjs-dist": "^5.6.205",
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
"use strict"
|
|
2
|
+
// postinstall — 검사·안내 전용. 절대 자동 설치하지 않으며 항상 exit 0.
|
|
3
|
+
const { spawnSync } = require("node:child_process")
|
|
4
|
+
const fs = require("node:fs")
|
|
5
|
+
const path = require("node:path")
|
|
6
|
+
|
|
7
|
+
// 건너뛰는 조건: CI, 프로덕션 모드, 스킵 플래그, 잘못된 lifecycle
|
|
8
|
+
const skip =
|
|
9
|
+
process.env.CI === "true" ||
|
|
10
|
+
process.env.KORDOC_SKIP_DOCTOR === "1" ||
|
|
11
|
+
process.env.npm_config_production === "true" ||
|
|
12
|
+
process.env.npm_lifecycle_event !== "postinstall"
|
|
13
|
+
|
|
14
|
+
if (skip) process.exit(0)
|
|
15
|
+
|
|
16
|
+
try {
|
|
17
|
+
// dist/setup/doctor.cjs가 없으면 (git clone 직후 등) 조용히 종료
|
|
18
|
+
const entry = path.join(__dirname, "..", "dist", "setup", "doctor.cjs")
|
|
19
|
+
if (!fs.existsSync(entry)) process.exit(0)
|
|
20
|
+
|
|
21
|
+
const result = spawnSync(process.execPath, [entry, "--postinstall"], { stdio: "inherit" })
|
|
22
|
+
// doctor 결과와 무관하게 항상 성공으로 끝낸다 — npm install 차단 금지
|
|
23
|
+
} catch (_e) {
|
|
24
|
+
// 어떤 오류도 npm install을 실패시키지 않는다
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
process.exit(0)
|