@ahkohd/yagami 0.1.2

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 (81) hide show
  1. package/.beads/.beads-credential-key +1 -0
  2. package/.beads/README.md +81 -0
  3. package/.beads/config.yaml +54 -0
  4. package/.beads/hooks/post-checkout +24 -0
  5. package/.beads/hooks/post-merge +24 -0
  6. package/.beads/hooks/pre-commit +24 -0
  7. package/.beads/hooks/pre-push +24 -0
  8. package/.beads/hooks/prepare-commit-msg +24 -0
  9. package/.beads/metadata.json +7 -0
  10. package/.github/workflows/ci.yml +43 -0
  11. package/.github/workflows/release.yml +115 -0
  12. package/AGENTS.md +150 -0
  13. package/README.md +210 -0
  14. package/biome.json +36 -0
  15. package/config/mcporter.json +8 -0
  16. package/dist/cli/theme.js +202 -0
  17. package/dist/cli/theme.js.map +1 -0
  18. package/dist/cli.js +1883 -0
  19. package/dist/cli.js.map +1 -0
  20. package/dist/config.js +223 -0
  21. package/dist/config.js.map +1 -0
  22. package/dist/daemon.js +745 -0
  23. package/dist/daemon.js.map +1 -0
  24. package/dist/engine/constants.js +131 -0
  25. package/dist/engine/constants.js.map +1 -0
  26. package/dist/engine/deep-research.js +167 -0
  27. package/dist/engine/deep-research.js.map +1 -0
  28. package/dist/engine/defuddle-utils.js +57 -0
  29. package/dist/engine/defuddle-utils.js.map +1 -0
  30. package/dist/engine/github-fetch.js +232 -0
  31. package/dist/engine/github-fetch.js.map +1 -0
  32. package/dist/engine/helpers.js +372 -0
  33. package/dist/engine/helpers.js.map +1 -0
  34. package/dist/engine/limiter.js +75 -0
  35. package/dist/engine/limiter.js.map +1 -0
  36. package/dist/engine/policy.js +313 -0
  37. package/dist/engine/policy.js.map +1 -0
  38. package/dist/engine/runtime-utils.js +65 -0
  39. package/dist/engine/runtime-utils.js.map +1 -0
  40. package/dist/engine/search-discovery.js +275 -0
  41. package/dist/engine/search-discovery.js.map +1 -0
  42. package/dist/engine/url-utils.js +72 -0
  43. package/dist/engine/url-utils.js.map +1 -0
  44. package/dist/engine.js +2030 -0
  45. package/dist/engine.js.map +1 -0
  46. package/dist/mcp.js +282 -0
  47. package/dist/mcp.js.map +1 -0
  48. package/dist/types/cli.js +2 -0
  49. package/dist/types/cli.js.map +1 -0
  50. package/dist/types/config.js +2 -0
  51. package/dist/types/config.js.map +1 -0
  52. package/dist/types/daemon.js +2 -0
  53. package/dist/types/daemon.js.map +1 -0
  54. package/dist/types/engine.js +2 -0
  55. package/dist/types/engine.js.map +1 -0
  56. package/package.json +66 -0
  57. package/packages/pi-yagami-search/README.md +39 -0
  58. package/packages/pi-yagami-search/extensions/yagami-search.ts +273 -0
  59. package/packages/pi-yagami-search/package.json +41 -0
  60. package/src/cli/theme.ts +260 -0
  61. package/src/cli.ts +2226 -0
  62. package/src/config.ts +250 -0
  63. package/src/daemon.ts +990 -0
  64. package/src/engine/constants.ts +147 -0
  65. package/src/engine/deep-research.ts +207 -0
  66. package/src/engine/defuddle-utils.ts +75 -0
  67. package/src/engine/github-fetch.ts +265 -0
  68. package/src/engine/helpers.ts +394 -0
  69. package/src/engine/limiter.ts +97 -0
  70. package/src/engine/policy.ts +392 -0
  71. package/src/engine/runtime-utils.ts +79 -0
  72. package/src/engine/search-discovery.ts +351 -0
  73. package/src/engine/url-utils.ts +86 -0
  74. package/src/engine.ts +2516 -0
  75. package/src/mcp.ts +337 -0
  76. package/src/shims-cli.d.ts +3 -0
  77. package/src/types/cli.ts +7 -0
  78. package/src/types/config.ts +53 -0
  79. package/src/types/daemon.ts +22 -0
  80. package/src/types/engine.ts +194 -0
  81. package/tsconfig.json +18 -0
@@ -0,0 +1,275 @@
1
+ import { categoryProfile, clampInteger, domainMatches, getHostname, isTrackingOrAdUrl, isValidPublicHostname, normalizeDomainFilter, normalizeWhitespace, parseStringList, stripHtml, unwrapDuckDuckGoHref, } from "./helpers.js";
2
+ import { normalizeUrl } from "./url-utils.js";
3
+ const SEARCH_ENGINE_PRESETS = {
4
+ duckduckgo: "https://duckduckgo.com/html/?q={query}",
5
+ bing: "https://www.bing.com/search?q={query}",
6
+ google: "https://www.google.com/search?q={query}",
7
+ brave: "https://search.brave.com/search?q={query}",
8
+ };
9
+ function normalizeSearchEngine(value, fallback = "duckduckgo") {
10
+ const normalized = String(value ?? "")
11
+ .trim()
12
+ .toLowerCase();
13
+ if (normalized === "duckduckgo")
14
+ return "duckduckgo";
15
+ if (normalized === "bing")
16
+ return "bing";
17
+ if (normalized === "google")
18
+ return "google";
19
+ if (normalized === "brave")
20
+ return "brave";
21
+ if (normalized === "custom")
22
+ return "custom";
23
+ return fallback;
24
+ }
25
+ export function resolveSearchEngineTemplate(engine, customTemplate) {
26
+ const normalizedTemplate = String(customTemplate ?? "").trim();
27
+ if (normalizedTemplate) {
28
+ return {
29
+ engine: "custom",
30
+ template: normalizedTemplate,
31
+ };
32
+ }
33
+ if (engine !== "custom") {
34
+ return {
35
+ engine,
36
+ template: SEARCH_ENGINE_PRESETS[engine],
37
+ };
38
+ }
39
+ return {
40
+ engine: "duckduckgo",
41
+ template: SEARCH_ENGINE_PRESETS.duckduckgo,
42
+ };
43
+ }
44
+ function buildSearchUrlFromTemplate(template, query) {
45
+ const encodedQuery = encodeURIComponent(query);
46
+ if (template.includes("{query}")) {
47
+ return template.replace(/\{query\}/g, encodedQuery);
48
+ }
49
+ if (template.includes("%s")) {
50
+ return template.replace(/%s/g, encodedQuery);
51
+ }
52
+ try {
53
+ const url = new URL(template);
54
+ url.searchParams.set("q", query);
55
+ return url.toString();
56
+ }
57
+ catch {
58
+ return `${template}${encodedQuery}`;
59
+ }
60
+ }
61
+ export async function parseDuckDuckGoResults(html, options = {}) {
62
+ const limit = options.limit ?? 40;
63
+ const { parseHTML } = (await import("linkedom"));
64
+ const { document } = parseHTML(html || "");
65
+ const results = [];
66
+ const seen = new Set();
67
+ const resultNodes = Array.from((document.querySelectorAll?.call(document, ".result") || []));
68
+ for (const node of resultNodes) {
69
+ const querySelector = node.querySelector;
70
+ if (!querySelector)
71
+ continue;
72
+ const link = querySelector.call(node, "a.result__a") ||
73
+ querySelector.call(node, "h2 a") ||
74
+ querySelector.call(node, "a[href]");
75
+ if (!link)
76
+ continue;
77
+ const getAttribute = link.getAttribute;
78
+ const href = unwrapDuckDuckGoHref((getAttribute?.call(link, "href") || ""));
79
+ let normalized;
80
+ try {
81
+ normalized = normalizeUrl(href);
82
+ }
83
+ catch {
84
+ continue;
85
+ }
86
+ const hostname = getHostname(normalized);
87
+ if (!isValidPublicHostname(hostname))
88
+ continue;
89
+ if (isTrackingOrAdUrl(normalized))
90
+ continue;
91
+ if (seen.has(normalized))
92
+ continue;
93
+ seen.add(normalized);
94
+ const textContent = link.textContent;
95
+ const innerHTML = link.innerHTML;
96
+ const title = normalizeWhitespace(textContent || stripHtml(innerHTML || ""));
97
+ const snippetNode = querySelector.call(node, ".result__snippet") ||
98
+ querySelector.call(node, ".result-snippet") ||
99
+ querySelector.call(node, ".result__extras");
100
+ const snippet = normalizeWhitespace(String(snippetNode?.textContent || ""));
101
+ results.push({
102
+ rank: results.length + 1,
103
+ url: normalized,
104
+ title,
105
+ snippet,
106
+ domain: hostname,
107
+ });
108
+ if (results.length >= limit)
109
+ break;
110
+ }
111
+ if (results.length > 0) {
112
+ return results;
113
+ }
114
+ const fallback = [];
115
+ const seenFallback = new Set();
116
+ const regex = /<a[^>]*href=["']([^"']+)["'][^>]*>([\s\S]*?)<\/a>/gi;
117
+ const htmlInput = html || "";
118
+ let match = regex.exec(htmlInput);
119
+ while (match !== null) {
120
+ const href = unwrapDuckDuckGoHref(match[1]);
121
+ let normalized;
122
+ try {
123
+ normalized = normalizeUrl(href);
124
+ }
125
+ catch {
126
+ match = regex.exec(htmlInput);
127
+ continue;
128
+ }
129
+ const hostname = getHostname(normalized);
130
+ if (!isValidPublicHostname(hostname)) {
131
+ match = regex.exec(htmlInput);
132
+ continue;
133
+ }
134
+ if (isTrackingOrAdUrl(normalized)) {
135
+ match = regex.exec(htmlInput);
136
+ continue;
137
+ }
138
+ if (seenFallback.has(normalized)) {
139
+ match = regex.exec(htmlInput);
140
+ continue;
141
+ }
142
+ seenFallback.add(normalized);
143
+ const title = stripHtml(match[2]);
144
+ if (!title) {
145
+ match = regex.exec(htmlInput);
146
+ continue;
147
+ }
148
+ fallback.push({
149
+ rank: fallback.length + 1,
150
+ url: normalized,
151
+ title,
152
+ snippet: "",
153
+ domain: hostname,
154
+ });
155
+ if (fallback.length >= limit)
156
+ break;
157
+ match = regex.exec(htmlInput);
158
+ }
159
+ return fallback;
160
+ }
161
+ export async function discoverSearchResults(query, options = {}) {
162
+ const startedAt = Date.now();
163
+ const numResults = clampInteger(options.numResults, 12, { min: 1, max: 100 });
164
+ const category = String(options.category || "")
165
+ .trim()
166
+ .toLowerCase();
167
+ const includeDomains = parseStringList(options.includeDomains)
168
+ .map((value) => normalizeDomainFilter(value))
169
+ .filter(Boolean);
170
+ const excludeDomains = parseStringList(options.excludeDomains)
171
+ .map((value) => normalizeDomainFilter(value))
172
+ .filter(Boolean);
173
+ const includeText = parseStringList(options.includeText).map((value) => value.toLowerCase());
174
+ const excludeText = parseStringList(options.excludeText).map((value) => value.toLowerCase());
175
+ const profile = categoryProfile(category);
176
+ const mergedIncludeDomains = Array.from(new Set([...includeDomains, ...profile.includeDomains.map((value) => normalizeDomainFilter(value))])).filter(Boolean);
177
+ const mergedIncludeText = Array.from(new Set([...includeText, ...profile.includeText.map((value) => value.toLowerCase())])).filter(Boolean);
178
+ const queryHint = String(profile.queryHint || "").trim();
179
+ const effectiveQuery = queryHint ? `${query} ${queryHint}` : query;
180
+ const requestedEngine = normalizeSearchEngine(options.searchEngine, "duckduckgo");
181
+ const requestedSearch = resolveSearchEngineTemplate(requestedEngine, options.searchEngineUrlTemplate);
182
+ const fallbackSearch = requestedSearch.engine === "duckduckgo"
183
+ ? null
184
+ : {
185
+ engine: "duckduckgo",
186
+ template: SEARCH_ENGINE_PRESETS.duckduckgo,
187
+ };
188
+ const attempts = fallbackSearch ? [requestedSearch, fallbackSearch] : [requestedSearch];
189
+ let activeSearch = requestedSearch;
190
+ let searchUrl = "";
191
+ let results = [];
192
+ let lastError = null;
193
+ for (const attempt of attempts) {
194
+ const candidateSearchUrl = buildSearchUrlFromTemplate(attempt.template, effectiveQuery);
195
+ try {
196
+ const response = await fetch(candidateSearchUrl, {
197
+ method: "GET",
198
+ headers: {
199
+ "user-agent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/124.0 Safari/537.36",
200
+ "accept-language": "en-US,en;q=0.9",
201
+ },
202
+ });
203
+ if (!response.ok) {
204
+ const text = await response.text();
205
+ throw new Error(`Search discovery failed (HTTP ${response.status}): ${text.slice(0, 300)}`);
206
+ }
207
+ const html = await response.text();
208
+ const parsedResults = await parseDuckDuckGoResults(html, { limit: Math.max(numResults * 4, 24) });
209
+ const minAcceptableResults = Math.max(1, Math.min(2, numResults));
210
+ if (parsedResults.length < minAcceptableResults && attempt.engine !== "duckduckgo" && fallbackSearch) {
211
+ continue;
212
+ }
213
+ activeSearch = attempt;
214
+ searchUrl = candidateSearchUrl;
215
+ results = parsedResults;
216
+ break;
217
+ }
218
+ catch (error) {
219
+ lastError = error;
220
+ if (attempt.engine !== "duckduckgo" && fallbackSearch) {
221
+ continue;
222
+ }
223
+ throw error;
224
+ }
225
+ }
226
+ if (!searchUrl) {
227
+ if (lastError instanceof Error)
228
+ throw lastError;
229
+ if (lastError !== null && lastError !== undefined)
230
+ throw new Error(String(lastError));
231
+ throw new Error("Search discovery failed: no search URL could be resolved.");
232
+ }
233
+ if (category === "pdf") {
234
+ results = results.filter((result) => String(result.url || "")
235
+ .toLowerCase()
236
+ .includes(".pdf"));
237
+ }
238
+ if (mergedIncludeDomains.length > 0) {
239
+ results = results.filter((result) => domainMatches(String(result.domain || ""), mergedIncludeDomains));
240
+ }
241
+ if (excludeDomains.length > 0) {
242
+ results = results.filter((result) => !domainMatches(String(result.domain || ""), excludeDomains));
243
+ }
244
+ if (mergedIncludeText.length > 0) {
245
+ results = results.filter((result) => {
246
+ const haystack = `${result.title || ""} ${result.snippet || ""} ${result.url || ""}`.toLowerCase();
247
+ return mergedIncludeText.every((term) => haystack.includes(term));
248
+ });
249
+ }
250
+ if (excludeText.length > 0) {
251
+ results = results.filter((result) => {
252
+ const haystack = `${result.title || ""} ${result.snippet || ""} ${result.url || ""}`.toLowerCase();
253
+ return excludeText.every((term) => !haystack.includes(term));
254
+ });
255
+ }
256
+ results = results.slice(0, numResults).map((result, index) => ({ ...result, rank: index + 1 }));
257
+ return {
258
+ query,
259
+ effectiveQuery,
260
+ category: category || null,
261
+ searchEngine: activeSearch.engine,
262
+ searchEngineRequested: requestedSearch.engine,
263
+ searchUrlTemplate: activeSearch.template,
264
+ searchUrl,
265
+ results,
266
+ durationMs: Date.now() - startedAt,
267
+ filters: {
268
+ includeDomains: mergedIncludeDomains,
269
+ excludeDomains,
270
+ includeText: mergedIncludeText,
271
+ excludeText,
272
+ },
273
+ };
274
+ }
275
+ //# sourceMappingURL=search-discovery.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search-discovery.js","sourceRoot":"","sources":["../../src/engine/search-discovery.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,72 @@
1
+ const URL_REGEX = /https?:\/\/[^\s)\]>]+/g;
2
+ export function sanitizeUrlCandidate(input) {
3
+ let value = String(input ?? "").trim();
4
+ if (!value)
5
+ return "";
6
+ if (value.startsWith("<") && value.endsWith(">")) {
7
+ value = value.slice(1, -1).trim();
8
+ }
9
+ value = value.replace(/^[\s("'`[]+/, "");
10
+ for (let i = 0; i < 3; i += 1) {
11
+ const next = value
12
+ .replace(/(?:\*{1,3}|_{1,3}|`{1,3})+$/g, "")
13
+ .replace(/[>\]"'`]+$/g, "")
14
+ .replace(/[.,;:!?]+$/g, "")
15
+ .trim();
16
+ if (next === value)
17
+ break;
18
+ value = next;
19
+ }
20
+ return value;
21
+ }
22
+ export function normalizeUrl(input) {
23
+ const raw = sanitizeUrlCandidate(input);
24
+ if (!raw)
25
+ throw new Error("URL is required");
26
+ const withProtocol = /^https?:\/\//i.test(raw) ? raw : `https://${raw}`;
27
+ const url = new URL(withProtocol);
28
+ if (!["http:", "https:"].includes(url.protocol)) {
29
+ throw new Error(`Unsupported URL protocol: ${url.protocol}`);
30
+ }
31
+ return url.toString();
32
+ }
33
+ export function normalizeUrlForDedupe(input) {
34
+ const normalized = normalizeUrl(input);
35
+ const url = new URL(normalized);
36
+ url.hash = "";
37
+ if ((url.protocol === "http:" && url.port === "80") || (url.protocol === "https:" && url.port === "443")) {
38
+ url.port = "";
39
+ }
40
+ if (url.pathname !== "/") {
41
+ const trimmedPath = url.pathname.replace(/\/+$/g, "");
42
+ url.pathname = trimmedPath || "/";
43
+ }
44
+ return url.toString();
45
+ }
46
+ export function normalizeUniqueUrls(values) {
47
+ const urls = [];
48
+ const seen = new Set();
49
+ for (const value of values) {
50
+ try {
51
+ const normalized = normalizeUrl(value);
52
+ const dedupeKey = normalizeUrlForDedupe(normalized);
53
+ if (seen.has(dedupeKey))
54
+ continue;
55
+ seen.add(dedupeKey);
56
+ urls.push(normalized);
57
+ }
58
+ catch {
59
+ // ignore malformed URL-like fragments
60
+ }
61
+ }
62
+ return urls;
63
+ }
64
+ export function extractSeedUrls(text) {
65
+ const matches = String(text ?? "").match(URL_REGEX) ?? [];
66
+ return normalizeUniqueUrls(matches);
67
+ }
68
+ export function extractCitationUrls(text) {
69
+ const matches = String(text ?? "").match(URL_REGEX) ?? [];
70
+ return normalizeUniqueUrls(matches);
71
+ }
72
+ //# sourceMappingURL=url-utils.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"url-utils.js","sourceRoot":"","sources":["../../src/engine/url-utils.ts"],"names":[],"mappings":"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"}