repoview 0.5.0 → 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.
Files changed (51) hide show
  1. package/CHANGELOG.md +53 -0
  2. package/CONTRIBUTING.md +4 -3
  3. package/DEVELOPMENT.md +70 -16
  4. package/README.md +36 -5
  5. package/dist/api.js +58 -0
  6. package/dist/api.js.map +1 -0
  7. package/dist/cli.js +224 -0
  8. package/dist/cli.js.map +1 -0
  9. package/dist/csv.js +64 -0
  10. package/dist/csv.js.map +1 -0
  11. package/dist/format.js +25 -0
  12. package/dist/format.js.map +1 -0
  13. package/dist/git.js +67 -0
  14. package/dist/git.js.map +1 -0
  15. package/dist/gitignore.js +34 -0
  16. package/dist/gitignore.js.map +1 -0
  17. package/dist/linkcheck.js +310 -0
  18. package/dist/linkcheck.js.map +1 -0
  19. package/dist/markdown.js +493 -0
  20. package/dist/markdown.js.map +1 -0
  21. package/dist/net.js +10 -0
  22. package/dist/net.js.map +1 -0
  23. package/dist/paths.js +59 -0
  24. package/dist/paths.js.map +1 -0
  25. package/dist/reload.js +36 -0
  26. package/dist/reload.js.map +1 -0
  27. package/dist/repo-context.js +73 -0
  28. package/dist/repo-context.js.map +1 -0
  29. package/dist/repo-router.js +801 -0
  30. package/dist/repo-router.js.map +1 -0
  31. package/dist/review-cli.js +228 -0
  32. package/dist/review-cli.js.map +1 -0
  33. package/dist/server.js +116 -0
  34. package/dist/server.js.map +1 -0
  35. package/dist/session.js +86 -0
  36. package/dist/session.js.map +1 -0
  37. package/dist/types.js +2 -0
  38. package/dist/types.js.map +1 -0
  39. package/dist/views.js +633 -0
  40. package/dist/views.js.map +1 -0
  41. package/package.json +22 -6
  42. package/public/app.css +842 -0
  43. package/public/app.js +35 -2
  44. package/public/review.js +587 -0
  45. package/public/session.js +61 -0
  46. package/src/cli.js +0 -73
  47. package/src/gitignore.js +0 -34
  48. package/src/linkcheck.js +0 -312
  49. package/src/markdown.js +0 -364
  50. package/src/server.js +0 -760
  51. package/src/views.js +0 -479
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
@@ -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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"}