repoview 0.5.1 → 0.6.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/CHANGELOG.md +53 -0
- package/CONTRIBUTING.md +4 -3
- package/DEVELOPMENT.md +70 -16
- package/README.md +36 -5
- package/dist/api.js +58 -0
- package/dist/api.js.map +1 -0
- package/dist/cli.js +224 -0
- package/dist/cli.js.map +1 -0
- package/dist/csv.js +64 -0
- package/dist/csv.js.map +1 -0
- package/dist/format.js +25 -0
- package/dist/format.js.map +1 -0
- package/dist/git.js +67 -0
- package/dist/git.js.map +1 -0
- package/dist/gitignore.js +34 -0
- package/dist/gitignore.js.map +1 -0
- package/dist/linkcheck.js +310 -0
- package/dist/linkcheck.js.map +1 -0
- package/dist/markdown.js +493 -0
- package/dist/markdown.js.map +1 -0
- package/dist/net.js +10 -0
- package/dist/net.js.map +1 -0
- package/dist/paths.js +59 -0
- package/dist/paths.js.map +1 -0
- package/dist/reload.js +36 -0
- package/dist/reload.js.map +1 -0
- package/dist/repo-context.js +73 -0
- package/dist/repo-context.js.map +1 -0
- package/dist/repo-router.js +801 -0
- package/dist/repo-router.js.map +1 -0
- package/dist/review-cli.js +228 -0
- package/dist/review-cli.js.map +1 -0
- package/dist/server.js +116 -0
- package/dist/server.js.map +1 -0
- package/dist/session.js +86 -0
- package/dist/session.js.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/dist/views.js +633 -0
- package/dist/views.js.map +1 -0
- package/package.json +20 -9
- package/public/app.css +82 -0
- package/public/app.js +4 -2
- package/public/review.js +9 -6
- package/public/session.js +61 -0
- package/src/cli.js +0 -91
- package/src/gitignore.js +0 -34
- package/src/linkcheck.js +0 -312
- package/src/markdown.js +0 -518
- package/src/review-cli.js +0 -245
- package/src/server.js +0 -1126
- package/src/views.js +0 -657
package/dist/format.js
ADDED
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
export function formatBytes(bytes) {
|
|
2
|
+
if (!Number.isFinite(bytes))
|
|
3
|
+
return "";
|
|
4
|
+
if (bytes < 1024)
|
|
5
|
+
return `${bytes} B`;
|
|
6
|
+
const units = ["KB", "MB", "GB", "TB"];
|
|
7
|
+
let value = bytes / 1024;
|
|
8
|
+
let unit = 0;
|
|
9
|
+
while (value >= 1024 && unit < units.length - 1) {
|
|
10
|
+
value /= 1024;
|
|
11
|
+
unit++;
|
|
12
|
+
}
|
|
13
|
+
return `${value.toFixed(value < 10 ? 1 : 0)} ${units[unit]}`;
|
|
14
|
+
}
|
|
15
|
+
export function formatDate(ms) {
|
|
16
|
+
const d = new Date(ms);
|
|
17
|
+
return d.toLocaleString(undefined, {
|
|
18
|
+
year: "numeric",
|
|
19
|
+
month: "short",
|
|
20
|
+
day: "2-digit",
|
|
21
|
+
hour: "2-digit",
|
|
22
|
+
minute: "2-digit",
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
//# sourceMappingURL=format.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"format.js","sourceRoot":"","sources":["../src/format.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,WAAW,CAAC,KAAa;IACvC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC;QAAE,OAAO,EAAE,CAAC;IACvC,IAAI,KAAK,GAAG,IAAI;QAAE,OAAO,GAAG,KAAK,IAAI,CAAC;IACtC,MAAM,KAAK,GAAG,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;IACvC,IAAI,KAAK,GAAG,KAAK,GAAG,IAAI,CAAC;IACzB,IAAI,IAAI,GAAG,CAAC,CAAC;IACb,OAAO,KAAK,IAAI,IAAI,IAAI,IAAI,GAAG,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAChD,KAAK,IAAI,IAAI,CAAC;QACd,IAAI,EAAE,CAAC;IACT,CAAC;IACD,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC,KAAK,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC;AAC/D,CAAC;AAED,MAAM,UAAU,UAAU,CAAC,EAAU;IACnC,MAAM,CAAC,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC,CAAC;IACvB,OAAO,CAAC,CAAC,cAAc,CAAC,SAAS,EAAE;QACjC,IAAI,EAAE,SAAS;QACf,KAAK,EAAE,OAAO;QACd,GAAG,EAAE,SAAS;QACd,IAAI,EAAE,SAAS;QACf,MAAM,EAAE,SAAS;KAClB,CAAC,CAAC;AACL,CAAC"}
|
package/dist/git.js
ADDED
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
import fs from "node:fs/promises";
|
|
2
|
+
import path from "node:path";
|
|
3
|
+
import { spawn } from "node:child_process";
|
|
4
|
+
export function execGit(repoRootReal, args, maxBytes = 1024 * 1024) {
|
|
5
|
+
return new Promise((resolve) => {
|
|
6
|
+
const child = spawn("git", args, { cwd: repoRootReal });
|
|
7
|
+
let out = "";
|
|
8
|
+
let size = 0;
|
|
9
|
+
let killed = false;
|
|
10
|
+
child.stdout.on("data", (chunk) => {
|
|
11
|
+
size += chunk.length;
|
|
12
|
+
if (size > maxBytes) {
|
|
13
|
+
if (!killed) {
|
|
14
|
+
killed = true;
|
|
15
|
+
child.kill();
|
|
16
|
+
}
|
|
17
|
+
return;
|
|
18
|
+
}
|
|
19
|
+
out += String(chunk);
|
|
20
|
+
});
|
|
21
|
+
child.on("close", (code) => {
|
|
22
|
+
if (killed)
|
|
23
|
+
return resolve({ output: out, tooLarge: true, code });
|
|
24
|
+
resolve({ output: code === 0 ? out.trim() : null, tooLarge: false, code });
|
|
25
|
+
});
|
|
26
|
+
child.on("error", () => resolve({ output: null, tooLarge: false, code: -1 }));
|
|
27
|
+
});
|
|
28
|
+
}
|
|
29
|
+
export function validateGitRef(ref) {
|
|
30
|
+
if (!ref || typeof ref !== "string")
|
|
31
|
+
return false;
|
|
32
|
+
return /^[a-zA-Z0-9_.\/\-~^]+$/.test(ref);
|
|
33
|
+
}
|
|
34
|
+
export async function getGitBranches(repoRootReal) {
|
|
35
|
+
const { output } = await execGit(repoRootReal, ["branch", "--format=%(refname:short)"]);
|
|
36
|
+
if (!output)
|
|
37
|
+
return [];
|
|
38
|
+
return output.split("\n").filter(Boolean);
|
|
39
|
+
}
|
|
40
|
+
export async function getGitTags(repoRootReal) {
|
|
41
|
+
const { output } = await execGit(repoRootReal, ["tag", "-l"]);
|
|
42
|
+
if (!output)
|
|
43
|
+
return [];
|
|
44
|
+
return output.split("\n").filter(Boolean);
|
|
45
|
+
}
|
|
46
|
+
export async function getGitDiffRaw(repoRootReal, base) {
|
|
47
|
+
const maxBytes = 512 * 1024;
|
|
48
|
+
const { output, tooLarge } = await execGit(repoRootReal, ["diff", base], maxBytes);
|
|
49
|
+
return { raw: output || "", tooLarge };
|
|
50
|
+
}
|
|
51
|
+
export async function getGitInfo(repoRootReal) {
|
|
52
|
+
const gitDir = path.join(repoRootReal, ".git");
|
|
53
|
+
try {
|
|
54
|
+
await fs.stat(gitDir);
|
|
55
|
+
}
|
|
56
|
+
catch {
|
|
57
|
+
return { branch: null, commit: null };
|
|
58
|
+
}
|
|
59
|
+
const [branchResult, commitResult] = await Promise.all([
|
|
60
|
+
execGit(repoRootReal, ["rev-parse", "--abbrev-ref", "HEAD"]),
|
|
61
|
+
execGit(repoRootReal, ["rev-parse", "HEAD"]),
|
|
62
|
+
]);
|
|
63
|
+
const branch = branchResult.output;
|
|
64
|
+
const commit = commitResult.output;
|
|
65
|
+
return { branch: branch && branch !== "HEAD" ? branch : branch, commit };
|
|
66
|
+
}
|
|
67
|
+
//# sourceMappingURL=git.js.map
|
package/dist/git.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"git.js","sourceRoot":"","sources":["../src/git.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import fs from "node:fs/promises";
|
|
2
|
+
import path from "node:path";
|
|
3
|
+
import ignore from "ignore";
|
|
4
|
+
function toPosixPath(p) {
|
|
5
|
+
return p.split(path.sep).join("/");
|
|
6
|
+
}
|
|
7
|
+
export async function loadGitIgnoreMatcher(repoRootReal) {
|
|
8
|
+
const ig = ignore();
|
|
9
|
+
// Baseline ignores (never show these via toggle either).
|
|
10
|
+
ig.add([".git/"]);
|
|
11
|
+
try {
|
|
12
|
+
const content = await fs.readFile(path.join(repoRootReal, ".gitignore"), "utf8");
|
|
13
|
+
ig.add(content);
|
|
14
|
+
}
|
|
15
|
+
catch {
|
|
16
|
+
// No .gitignore or unreadable; ignore.
|
|
17
|
+
}
|
|
18
|
+
return {
|
|
19
|
+
ignores(relPathPosix, { isDir = false } = {}) {
|
|
20
|
+
const p = toPosixPath(String(relPathPosix || "").replace(/^\/+/, ""));
|
|
21
|
+
if (!p)
|
|
22
|
+
return false;
|
|
23
|
+
if (ig.ignores(p))
|
|
24
|
+
return true;
|
|
25
|
+
if (isDir) {
|
|
26
|
+
const withSlash = p.endsWith("/") ? p : `${p}/`;
|
|
27
|
+
if (ig.ignores(withSlash))
|
|
28
|
+
return true;
|
|
29
|
+
}
|
|
30
|
+
return false;
|
|
31
|
+
},
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
//# sourceMappingURL=gitignore.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"gitignore.js","sourceRoot":"","sources":["../src/gitignore.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAClC,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,MAAM,MAAM,QAAQ,CAAC;AAG5B,SAAS,WAAW,CAAC,CAAS;IAC5B,OAAO,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AACrC,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,oBAAoB,CAAC,YAAoB;IAC7D,MAAM,EAAE,GAAG,MAAM,EAAE,CAAC;IAEpB,yDAAyD;IACzD,EAAE,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;IAElB,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,YAAY,CAAC,EAAE,MAAM,CAAC,CAAC;QACjF,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAClB,CAAC;IAAC,MAAM,CAAC;QACP,uCAAuC;IACzC,CAAC;IAED,OAAO;QACL,OAAO,CAAC,YAAoB,EAAE,EAAE,KAAK,GAAG,KAAK,KAA0B,EAAE;YACvE,MAAM,CAAC,GAAG,WAAW,CAAC,MAAM,CAAC,YAAY,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,CAAC;YACtE,IAAI,CAAC,CAAC;gBAAE,OAAO,KAAK,CAAC;YACrB,IAAI,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC;gBAAE,OAAO,IAAI,CAAC;YAC/B,IAAI,KAAK,EAAE,CAAC;gBACV,MAAM,SAAS,GAAG,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC;gBAChD,IAAI,EAAE,CAAC,OAAO,CAAC,SAAS,CAAC;oBAAE,OAAO,IAAI,CAAC;YACzC,CAAC;YACD,OAAO,KAAK,CAAC;QACf,CAAC;KACF,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,310 @@
|
|
|
1
|
+
import fs from "node:fs/promises";
|
|
2
|
+
import path from "node:path";
|
|
3
|
+
function toPosixPath(p) {
|
|
4
|
+
return p.split(path.sep).join("/");
|
|
5
|
+
}
|
|
6
|
+
function isWithinRoot(rootReal, candidateReal) {
|
|
7
|
+
if (candidateReal === rootReal)
|
|
8
|
+
return true;
|
|
9
|
+
const rootWithSep = rootReal.endsWith(path.sep) ? rootReal : rootReal + path.sep;
|
|
10
|
+
return candidateReal.startsWith(rootWithSep);
|
|
11
|
+
}
|
|
12
|
+
async function safeResolveExisting(repoRootReal, relPosixPath) {
|
|
13
|
+
const stripped = String(relPosixPath || "").replace(/^\/+/, "");
|
|
14
|
+
const resolved = path.resolve(repoRootReal, stripped);
|
|
15
|
+
if (!isWithinRoot(repoRootReal, resolved)) {
|
|
16
|
+
return { ok: false, reason: "escape", resolved: null, type: null };
|
|
17
|
+
}
|
|
18
|
+
try {
|
|
19
|
+
await fs.lstat(resolved);
|
|
20
|
+
}
|
|
21
|
+
catch {
|
|
22
|
+
return { ok: false, reason: "missing", resolved: null, type: null };
|
|
23
|
+
}
|
|
24
|
+
let real;
|
|
25
|
+
try {
|
|
26
|
+
real = await fs.realpath(resolved);
|
|
27
|
+
}
|
|
28
|
+
catch {
|
|
29
|
+
return { ok: false, reason: "missing", resolved: null, type: null };
|
|
30
|
+
}
|
|
31
|
+
if (!isWithinRoot(repoRootReal, real)) {
|
|
32
|
+
return { ok: false, reason: "escape", resolved: null, type: null };
|
|
33
|
+
}
|
|
34
|
+
const stat = await fs.stat(real);
|
|
35
|
+
return {
|
|
36
|
+
ok: true,
|
|
37
|
+
reason: null,
|
|
38
|
+
resolved: real,
|
|
39
|
+
type: stat.isDirectory() ? "dir" : stat.isFile() ? "file" : "other",
|
|
40
|
+
};
|
|
41
|
+
}
|
|
42
|
+
function extractInternalUrlsFromHtml(html) {
|
|
43
|
+
const urls = [];
|
|
44
|
+
const re = /\b(?:href|src)=(["'])([^"']+)\1/gi;
|
|
45
|
+
let match;
|
|
46
|
+
while ((match = re.exec(html))) {
|
|
47
|
+
const raw = match[2].trim();
|
|
48
|
+
if (!raw || raw.startsWith("#"))
|
|
49
|
+
continue;
|
|
50
|
+
urls.push(raw);
|
|
51
|
+
}
|
|
52
|
+
return urls;
|
|
53
|
+
}
|
|
54
|
+
function decodePosixPathFromUrlPath(urlPathname, prefix) {
|
|
55
|
+
const rest = urlPathname.slice(prefix.length);
|
|
56
|
+
const stripped = rest.replace(/^\/+/, "");
|
|
57
|
+
const segments = stripped.split("/").filter(Boolean);
|
|
58
|
+
try {
|
|
59
|
+
return segments.map((s) => decodeURIComponent(s)).join("/");
|
|
60
|
+
}
|
|
61
|
+
catch {
|
|
62
|
+
return null;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
function isMarkdownFile(relPosix) {
|
|
66
|
+
const lower = relPosix.toLowerCase();
|
|
67
|
+
const base = path.posix.basename(lower);
|
|
68
|
+
if (base === "readme" || base.startsWith("readme."))
|
|
69
|
+
return true;
|
|
70
|
+
const ext = path.posix.extname(lower).replace(/^\./, "");
|
|
71
|
+
return new Set(["md", "markdown", "mdown", "mkd", "mkdn"]).has(ext);
|
|
72
|
+
}
|
|
73
|
+
async function listMarkdownFiles(repoRootReal, { maxFiles, isIgnored } = {}) {
|
|
74
|
+
const results = [];
|
|
75
|
+
const stack = [
|
|
76
|
+
{ abs: repoRootReal, relPosix: "" },
|
|
77
|
+
];
|
|
78
|
+
const ignoredNames = new Set([".git", "node_modules"]);
|
|
79
|
+
while (stack.length) {
|
|
80
|
+
const { abs, relPosix } = stack.pop();
|
|
81
|
+
let entries;
|
|
82
|
+
try {
|
|
83
|
+
entries = await fs.readdir(abs, { withFileTypes: true });
|
|
84
|
+
}
|
|
85
|
+
catch {
|
|
86
|
+
continue;
|
|
87
|
+
}
|
|
88
|
+
for (const e of entries) {
|
|
89
|
+
if (ignoredNames.has(e.name))
|
|
90
|
+
continue;
|
|
91
|
+
const childAbs = path.join(abs, e.name);
|
|
92
|
+
const childRel = relPosix ? `${relPosix}/${e.name}` : e.name;
|
|
93
|
+
const childRelPosix = toPosixPath(childRel);
|
|
94
|
+
if (typeof isIgnored === "function" && isIgnored(childRelPosix, { isDir: e.isDirectory() }))
|
|
95
|
+
continue;
|
|
96
|
+
if (e.isDirectory()) {
|
|
97
|
+
stack.push({ abs: childAbs, relPosix: childRelPosix });
|
|
98
|
+
}
|
|
99
|
+
else if (e.isFile()) {
|
|
100
|
+
if (isMarkdownFile(childRelPosix))
|
|
101
|
+
results.push(childRelPosix);
|
|
102
|
+
if (maxFiles && results.length >= maxFiles)
|
|
103
|
+
return results;
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
results.sort();
|
|
108
|
+
return results;
|
|
109
|
+
}
|
|
110
|
+
export function createRepoLinkScanner({ repoRootReal, markdownRenderer, isIgnored, }) {
|
|
111
|
+
let current = {
|
|
112
|
+
status: "idle",
|
|
113
|
+
lastResult: null,
|
|
114
|
+
lastError: null,
|
|
115
|
+
lastStartedAt: null,
|
|
116
|
+
lastFinishedAt: null,
|
|
117
|
+
};
|
|
118
|
+
let scanRunning = false;
|
|
119
|
+
let scanQueued = false;
|
|
120
|
+
async function scanOnce({ maxMarkdownFiles = 5000, maxBytesPerFile = 2 * 1024 * 1024, concurrency = 16, } = {}) {
|
|
121
|
+
const startedAt = Date.now();
|
|
122
|
+
current = { ...current, status: "running", lastError: null, lastStartedAt: startedAt };
|
|
123
|
+
const markdownFiles = await listMarkdownFiles(repoRootReal, {
|
|
124
|
+
maxFiles: maxMarkdownFiles,
|
|
125
|
+
isIgnored,
|
|
126
|
+
});
|
|
127
|
+
const broken = [];
|
|
128
|
+
let filesScanned = 0;
|
|
129
|
+
let urlsChecked = 0;
|
|
130
|
+
const queue = markdownFiles.slice();
|
|
131
|
+
const workers = Array.from({ length: concurrency }, async () => {
|
|
132
|
+
while (queue.length) {
|
|
133
|
+
const relPosix = queue.pop();
|
|
134
|
+
if (!relPosix)
|
|
135
|
+
return;
|
|
136
|
+
const abs = path.join(repoRootReal, relPosix);
|
|
137
|
+
let stat;
|
|
138
|
+
try {
|
|
139
|
+
stat = await fs.stat(abs);
|
|
140
|
+
}
|
|
141
|
+
catch {
|
|
142
|
+
continue;
|
|
143
|
+
}
|
|
144
|
+
if (stat.size > maxBytesPerFile)
|
|
145
|
+
continue;
|
|
146
|
+
let text;
|
|
147
|
+
try {
|
|
148
|
+
text = await fs.readFile(abs, "utf8");
|
|
149
|
+
}
|
|
150
|
+
catch {
|
|
151
|
+
continue;
|
|
152
|
+
}
|
|
153
|
+
filesScanned++;
|
|
154
|
+
const baseDirPosix = path.posix.dirname(relPosix);
|
|
155
|
+
const env = { baseDirPosix: baseDirPosix === "." ? "" : baseDirPosix };
|
|
156
|
+
let html = "";
|
|
157
|
+
try {
|
|
158
|
+
html = markdownRenderer.render(text, env);
|
|
159
|
+
}
|
|
160
|
+
catch {
|
|
161
|
+
continue;
|
|
162
|
+
}
|
|
163
|
+
const urls = extractInternalUrlsFromHtml(html);
|
|
164
|
+
for (const raw of urls) {
|
|
165
|
+
if (raw.startsWith("http://") || raw.startsWith("https://"))
|
|
166
|
+
continue;
|
|
167
|
+
if (raw.startsWith("//"))
|
|
168
|
+
continue;
|
|
169
|
+
if (raw.startsWith("mailto:") || raw.startsWith("tel:"))
|
|
170
|
+
continue;
|
|
171
|
+
if (raw.startsWith("data:"))
|
|
172
|
+
continue;
|
|
173
|
+
urlsChecked++;
|
|
174
|
+
let urlPath;
|
|
175
|
+
try {
|
|
176
|
+
urlPath = new URL(raw, "http://local").pathname;
|
|
177
|
+
}
|
|
178
|
+
catch {
|
|
179
|
+
broken.push({
|
|
180
|
+
source: relPosix,
|
|
181
|
+
url: raw,
|
|
182
|
+
kind: "url",
|
|
183
|
+
reason: "invalid_url",
|
|
184
|
+
});
|
|
185
|
+
continue;
|
|
186
|
+
}
|
|
187
|
+
if (urlPath === "/events" || urlPath.startsWith("/static/"))
|
|
188
|
+
continue;
|
|
189
|
+
if (urlPath === "/broken-links" || urlPath === "/broken-links.json")
|
|
190
|
+
continue;
|
|
191
|
+
let expected = null;
|
|
192
|
+
let expectType = null;
|
|
193
|
+
if (urlPath.startsWith("/blob/")) {
|
|
194
|
+
expected = decodePosixPathFromUrlPath(urlPath, "/blob/");
|
|
195
|
+
expectType = "blob";
|
|
196
|
+
}
|
|
197
|
+
else if (urlPath.startsWith("/tree/")) {
|
|
198
|
+
expected = decodePosixPathFromUrlPath(urlPath, "/tree/");
|
|
199
|
+
expectType = "tree";
|
|
200
|
+
}
|
|
201
|
+
else if (urlPath.startsWith("/raw/")) {
|
|
202
|
+
expected = decodePosixPathFromUrlPath(urlPath, "/raw/");
|
|
203
|
+
expectType = "raw";
|
|
204
|
+
}
|
|
205
|
+
else {
|
|
206
|
+
broken.push({
|
|
207
|
+
source: relPosix,
|
|
208
|
+
url: raw,
|
|
209
|
+
kind: "url",
|
|
210
|
+
reason: "unknown_route",
|
|
211
|
+
});
|
|
212
|
+
continue;
|
|
213
|
+
}
|
|
214
|
+
if (expected == null) {
|
|
215
|
+
broken.push({
|
|
216
|
+
source: relPosix,
|
|
217
|
+
url: raw,
|
|
218
|
+
kind: expectType,
|
|
219
|
+
reason: "bad_encoding",
|
|
220
|
+
});
|
|
221
|
+
continue;
|
|
222
|
+
}
|
|
223
|
+
if (typeof isIgnored === "function") {
|
|
224
|
+
if (expectType === "tree" && isIgnored(expected, { isDir: true }))
|
|
225
|
+
continue;
|
|
226
|
+
if (expectType !== "tree" && isIgnored(expected, { isDir: false }))
|
|
227
|
+
continue;
|
|
228
|
+
}
|
|
229
|
+
const resolved = await safeResolveExisting(repoRootReal, expected);
|
|
230
|
+
if (!resolved.ok) {
|
|
231
|
+
broken.push({
|
|
232
|
+
source: relPosix,
|
|
233
|
+
url: raw,
|
|
234
|
+
kind: expectType,
|
|
235
|
+
reason: resolved.reason,
|
|
236
|
+
target: expected,
|
|
237
|
+
});
|
|
238
|
+
continue;
|
|
239
|
+
}
|
|
240
|
+
if (expectType === "raw" && resolved.type !== "file") {
|
|
241
|
+
broken.push({
|
|
242
|
+
source: relPosix,
|
|
243
|
+
url: raw,
|
|
244
|
+
kind: expectType,
|
|
245
|
+
reason: "not_a_file",
|
|
246
|
+
target: expected,
|
|
247
|
+
});
|
|
248
|
+
}
|
|
249
|
+
else if (expectType === "tree" && resolved.type !== "dir") {
|
|
250
|
+
broken.push({
|
|
251
|
+
source: relPosix,
|
|
252
|
+
url: raw,
|
|
253
|
+
kind: expectType,
|
|
254
|
+
reason: "not_a_directory",
|
|
255
|
+
target: expected,
|
|
256
|
+
});
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
});
|
|
261
|
+
await Promise.all(workers);
|
|
262
|
+
const finishedAt = Date.now();
|
|
263
|
+
const result = {
|
|
264
|
+
startedAt,
|
|
265
|
+
finishedAt,
|
|
266
|
+
durationMs: finishedAt - startedAt,
|
|
267
|
+
filesScanned,
|
|
268
|
+
urlsChecked,
|
|
269
|
+
broken: broken.sort((a, b) => a.source.localeCompare(b.source) || a.url.localeCompare(b.url)),
|
|
270
|
+
};
|
|
271
|
+
current = {
|
|
272
|
+
status: "idle",
|
|
273
|
+
lastResult: result,
|
|
274
|
+
lastError: null,
|
|
275
|
+
lastStartedAt: startedAt,
|
|
276
|
+
lastFinishedAt: finishedAt,
|
|
277
|
+
};
|
|
278
|
+
return result;
|
|
279
|
+
}
|
|
280
|
+
async function triggerScan(options) {
|
|
281
|
+
if (scanRunning) {
|
|
282
|
+
scanQueued = true;
|
|
283
|
+
return current;
|
|
284
|
+
}
|
|
285
|
+
scanRunning = true;
|
|
286
|
+
try {
|
|
287
|
+
await scanOnce(options);
|
|
288
|
+
}
|
|
289
|
+
catch (e) {
|
|
290
|
+
current = {
|
|
291
|
+
...current,
|
|
292
|
+
status: "idle",
|
|
293
|
+
lastError: String(e?.message || e),
|
|
294
|
+
};
|
|
295
|
+
}
|
|
296
|
+
finally {
|
|
297
|
+
scanRunning = false;
|
|
298
|
+
if (scanQueued) {
|
|
299
|
+
scanQueued = false;
|
|
300
|
+
void triggerScan(options);
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
return current;
|
|
304
|
+
}
|
|
305
|
+
function getState() {
|
|
306
|
+
return current;
|
|
307
|
+
}
|
|
308
|
+
return { scanOnce, triggerScan, getState };
|
|
309
|
+
}
|
|
310
|
+
//# sourceMappingURL=linkcheck.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"linkcheck.js","sourceRoot":"","sources":["../src/linkcheck.ts"],"names":[],"mappings":"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"}
|