@rankcli/agent-runtime 0.0.1

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 (178) hide show
  1. package/README.md +242 -0
  2. package/dist/analyzer-2CSWIQGD.mjs +6 -0
  3. package/dist/chunk-YNZYHEYM.mjs +774 -0
  4. package/dist/index.d.mts +4012 -0
  5. package/dist/index.d.ts +4012 -0
  6. package/dist/index.js +29672 -0
  7. package/dist/index.mjs +28602 -0
  8. package/package.json +53 -0
  9. package/scripts/build-deno.ts +134 -0
  10. package/src/audit/ai/analyzer.ts +347 -0
  11. package/src/audit/ai/index.ts +29 -0
  12. package/src/audit/ai/prompts/content-analysis.ts +271 -0
  13. package/src/audit/ai/types.ts +179 -0
  14. package/src/audit/checks/additional-checks.ts +439 -0
  15. package/src/audit/checks/ai-citation-worthiness.ts +399 -0
  16. package/src/audit/checks/ai-content-structure.ts +325 -0
  17. package/src/audit/checks/ai-readiness.ts +339 -0
  18. package/src/audit/checks/anchor-text.ts +179 -0
  19. package/src/audit/checks/answer-conciseness.ts +322 -0
  20. package/src/audit/checks/asset-minification.ts +270 -0
  21. package/src/audit/checks/bing-optimization.ts +206 -0
  22. package/src/audit/checks/brand-mention-optimization.ts +349 -0
  23. package/src/audit/checks/caching-headers.ts +305 -0
  24. package/src/audit/checks/canonical-advanced.ts +150 -0
  25. package/src/audit/checks/canonical-domain.ts +196 -0
  26. package/src/audit/checks/citation-quality.ts +358 -0
  27. package/src/audit/checks/client-rendering.ts +542 -0
  28. package/src/audit/checks/color-contrast.ts +342 -0
  29. package/src/audit/checks/content-freshness.ts +170 -0
  30. package/src/audit/checks/content-science.ts +589 -0
  31. package/src/audit/checks/conversion-elements.ts +526 -0
  32. package/src/audit/checks/crawlability.ts +220 -0
  33. package/src/audit/checks/directory-listing.ts +172 -0
  34. package/src/audit/checks/dom-analysis.ts +191 -0
  35. package/src/audit/checks/dom-size.ts +246 -0
  36. package/src/audit/checks/duplicate-content.ts +194 -0
  37. package/src/audit/checks/eeat-signals.ts +990 -0
  38. package/src/audit/checks/entity-seo.ts +396 -0
  39. package/src/audit/checks/featured-snippet.ts +473 -0
  40. package/src/audit/checks/freshness-signals.ts +443 -0
  41. package/src/audit/checks/funnel-intent.ts +463 -0
  42. package/src/audit/checks/hreflang.ts +174 -0
  43. package/src/audit/checks/html-compliance.ts +302 -0
  44. package/src/audit/checks/image-dimensions.ts +167 -0
  45. package/src/audit/checks/images.ts +160 -0
  46. package/src/audit/checks/indexnow.ts +275 -0
  47. package/src/audit/checks/interactive-tools.ts +475 -0
  48. package/src/audit/checks/internal-link-graph.ts +436 -0
  49. package/src/audit/checks/keyword-analysis.ts +239 -0
  50. package/src/audit/checks/keyword-cannibalization.ts +385 -0
  51. package/src/audit/checks/keyword-placement.ts +471 -0
  52. package/src/audit/checks/links.ts +203 -0
  53. package/src/audit/checks/llms-txt.ts +224 -0
  54. package/src/audit/checks/local-seo.ts +296 -0
  55. package/src/audit/checks/mobile.ts +167 -0
  56. package/src/audit/checks/modern-images.ts +226 -0
  57. package/src/audit/checks/navboost-signals.ts +395 -0
  58. package/src/audit/checks/on-page.ts +209 -0
  59. package/src/audit/checks/page-resources.ts +285 -0
  60. package/src/audit/checks/pagination.ts +180 -0
  61. package/src/audit/checks/performance.ts +153 -0
  62. package/src/audit/checks/platform-presence.ts +580 -0
  63. package/src/audit/checks/redirect-analysis.ts +153 -0
  64. package/src/audit/checks/redirect-chain.ts +389 -0
  65. package/src/audit/checks/resource-hints.ts +420 -0
  66. package/src/audit/checks/responsive-css.ts +247 -0
  67. package/src/audit/checks/responsive-images.ts +396 -0
  68. package/src/audit/checks/review-ecosystem.ts +415 -0
  69. package/src/audit/checks/robots-validation.ts +373 -0
  70. package/src/audit/checks/security-headers.ts +172 -0
  71. package/src/audit/checks/security.ts +144 -0
  72. package/src/audit/checks/serp-preview.ts +251 -0
  73. package/src/audit/checks/site-maturity.ts +444 -0
  74. package/src/audit/checks/social-meta.test.ts +275 -0
  75. package/src/audit/checks/social-meta.ts +134 -0
  76. package/src/audit/checks/soft-404.ts +151 -0
  77. package/src/audit/checks/structured-data.ts +238 -0
  78. package/src/audit/checks/tech-detection.ts +496 -0
  79. package/src/audit/checks/topical-clusters.ts +435 -0
  80. package/src/audit/checks/tracker-bloat.ts +462 -0
  81. package/src/audit/checks/tracking-verification.test.ts +371 -0
  82. package/src/audit/checks/tracking-verification.ts +636 -0
  83. package/src/audit/checks/url-safety.ts +682 -0
  84. package/src/audit/deno-entry.ts +66 -0
  85. package/src/audit/discovery/index.ts +15 -0
  86. package/src/audit/discovery/link-crawler.ts +232 -0
  87. package/src/audit/discovery/repo-routes.ts +347 -0
  88. package/src/audit/engine.ts +620 -0
  89. package/src/audit/fixes/index.ts +209 -0
  90. package/src/audit/fixes/social-meta-fixes.test.ts +329 -0
  91. package/src/audit/fixes/social-meta-fixes.ts +463 -0
  92. package/src/audit/index.ts +74 -0
  93. package/src/audit/runner.test.ts +299 -0
  94. package/src/audit/runner.ts +130 -0
  95. package/src/audit/types.ts +1953 -0
  96. package/src/content/featured-snippet.ts +367 -0
  97. package/src/content/generator.test.ts +534 -0
  98. package/src/content/generator.ts +501 -0
  99. package/src/content/headline.ts +317 -0
  100. package/src/content/index.ts +62 -0
  101. package/src/content/intent.ts +258 -0
  102. package/src/content/keyword-density.ts +349 -0
  103. package/src/content/readability.ts +262 -0
  104. package/src/executor.ts +336 -0
  105. package/src/fixer.ts +416 -0
  106. package/src/frameworks/detector.test.ts +248 -0
  107. package/src/frameworks/detector.ts +371 -0
  108. package/src/frameworks/index.ts +68 -0
  109. package/src/frameworks/recipes/angular.yaml +171 -0
  110. package/src/frameworks/recipes/astro.yaml +206 -0
  111. package/src/frameworks/recipes/django.yaml +180 -0
  112. package/src/frameworks/recipes/laravel.yaml +137 -0
  113. package/src/frameworks/recipes/nextjs.yaml +268 -0
  114. package/src/frameworks/recipes/nuxt.yaml +175 -0
  115. package/src/frameworks/recipes/rails.yaml +188 -0
  116. package/src/frameworks/recipes/react.yaml +202 -0
  117. package/src/frameworks/recipes/sveltekit.yaml +154 -0
  118. package/src/frameworks/recipes/vue.yaml +137 -0
  119. package/src/frameworks/recipes/wordpress.yaml +209 -0
  120. package/src/frameworks/suggestion-engine.ts +320 -0
  121. package/src/geo/geo-content.test.ts +305 -0
  122. package/src/geo/geo-content.ts +266 -0
  123. package/src/geo/geo-history.test.ts +473 -0
  124. package/src/geo/geo-history.ts +433 -0
  125. package/src/geo/geo-tracker.test.ts +359 -0
  126. package/src/geo/geo-tracker.ts +411 -0
  127. package/src/geo/index.ts +10 -0
  128. package/src/git/commit-helper.test.ts +261 -0
  129. package/src/git/commit-helper.ts +329 -0
  130. package/src/git/index.ts +12 -0
  131. package/src/git/pr-helper.test.ts +284 -0
  132. package/src/git/pr-helper.ts +307 -0
  133. package/src/index.ts +66 -0
  134. package/src/keywords/ai-keyword-engine.ts +1062 -0
  135. package/src/keywords/ai-summarizer.ts +387 -0
  136. package/src/keywords/ci-mode.ts +555 -0
  137. package/src/keywords/engine.ts +359 -0
  138. package/src/keywords/index.ts +151 -0
  139. package/src/keywords/llm-judge.ts +357 -0
  140. package/src/keywords/nlp-analysis.ts +706 -0
  141. package/src/keywords/prioritizer.ts +295 -0
  142. package/src/keywords/site-crawler.ts +342 -0
  143. package/src/keywords/sources/autocomplete.ts +139 -0
  144. package/src/keywords/sources/competitive-search.ts +450 -0
  145. package/src/keywords/sources/competitor-analysis.ts +374 -0
  146. package/src/keywords/sources/dataforseo.ts +206 -0
  147. package/src/keywords/sources/free-sources.ts +294 -0
  148. package/src/keywords/sources/gsc.ts +123 -0
  149. package/src/keywords/topic-grouping.ts +327 -0
  150. package/src/keywords/types.ts +144 -0
  151. package/src/keywords/wizard.ts +457 -0
  152. package/src/loader.ts +40 -0
  153. package/src/reports/index.ts +7 -0
  154. package/src/reports/report-generator.test.ts +293 -0
  155. package/src/reports/report-generator.ts +713 -0
  156. package/src/scheduler/alerts.test.ts +458 -0
  157. package/src/scheduler/alerts.ts +328 -0
  158. package/src/scheduler/index.ts +8 -0
  159. package/src/scheduler/scheduled-audit.test.ts +377 -0
  160. package/src/scheduler/scheduled-audit.ts +149 -0
  161. package/src/test/integration-test.ts +325 -0
  162. package/src/tools/analyzer.ts +373 -0
  163. package/src/tools/crawl.ts +293 -0
  164. package/src/tools/files.ts +301 -0
  165. package/src/tools/h1-fixer.ts +249 -0
  166. package/src/tools/index.ts +67 -0
  167. package/src/tracking/github-action.ts +326 -0
  168. package/src/tracking/google-analytics.ts +265 -0
  169. package/src/tracking/index.ts +45 -0
  170. package/src/tracking/report-generator.ts +386 -0
  171. package/src/tracking/search-console.ts +335 -0
  172. package/src/types.ts +134 -0
  173. package/src/utils/http.ts +302 -0
  174. package/src/wasm-adapter.ts +297 -0
  175. package/src/wasm-entry.ts +14 -0
  176. package/tsconfig.json +17 -0
  177. package/tsup.wasm.config.ts +26 -0
  178. package/vitest.config.ts +15 -0
@@ -0,0 +1,293 @@
1
+ import { httpGet } from '../utils/http.js';
2
+ import * as cheerio from 'cheerio';
3
+ import type { CrawlResult, MetaData, HeadingStructure, ImageInfo, LinkInfo, SchemaData, ToolResult } from '../types.js';
4
+
5
+ export async function crawlUrl(params: { url: string }): Promise<ToolResult> {
6
+ const { url } = params;
7
+
8
+ try {
9
+ const start = Date.now();
10
+ const response = await httpGet<string>(url, {
11
+
12
+ timeout: 30000,
13
+ validateStatus: () => true,
14
+ });
15
+ const loadTime = Date.now() - start;
16
+
17
+ const result: CrawlResult = {
18
+ url,
19
+ html: response.data,
20
+ statusCode: response.status,
21
+ headers: response.headers as Record<string, string>,
22
+ loadTime,
23
+ };
24
+
25
+ return { success: true, data: result };
26
+ } catch (error) {
27
+ return {
28
+ success: false,
29
+ error: error instanceof Error ? error.message : 'Unknown error crawling URL'
30
+ };
31
+ }
32
+ }
33
+
34
+ export async function extractMeta(params: { html: string; url: string }): Promise<ToolResult> {
35
+ const { html, url } = params;
36
+
37
+ try {
38
+ const $ = cheerio.load(html);
39
+
40
+ const meta: MetaData = {
41
+ title: $('title').text().trim() || undefined,
42
+ description: $('meta[name="description"]').attr('content')?.trim(),
43
+ canonical: $('link[rel="canonical"]').attr('href'),
44
+ robots: $('meta[name="robots"]').attr('content'),
45
+ viewport: $('meta[name="viewport"]').attr('content'),
46
+ charset: $('meta[charset]').attr('charset') || $('meta[http-equiv="Content-Type"]').attr('content'),
47
+ openGraph: {
48
+ title: $('meta[property="og:title"]').attr('content'),
49
+ description: $('meta[property="og:description"]').attr('content'),
50
+ image: $('meta[property="og:image"]').attr('content'),
51
+ url: $('meta[property="og:url"]').attr('content'),
52
+ type: $('meta[property="og:type"]').attr('content'),
53
+ siteName: $('meta[property="og:site_name"]').attr('content'),
54
+ },
55
+ twitter: {
56
+ card: $('meta[name="twitter:card"]').attr('content'),
57
+ title: $('meta[name="twitter:title"]').attr('content'),
58
+ description: $('meta[name="twitter:description"]').attr('content'),
59
+ image: $('meta[name="twitter:image"]').attr('content'),
60
+ site: $('meta[name="twitter:site"]').attr('content'),
61
+ },
62
+ other: {},
63
+ };
64
+
65
+ // Extract other meta tags
66
+ $('meta').each((_, el) => {
67
+ const name = $(el).attr('name') || $(el).attr('property');
68
+ const content = $(el).attr('content');
69
+ if (name && content && !name.startsWith('og:') && !name.startsWith('twitter:')) {
70
+ if (!['description', 'robots', 'viewport'].includes(name)) {
71
+ meta.other[name] = content;
72
+ }
73
+ }
74
+ });
75
+
76
+ return { success: true, data: meta };
77
+ } catch (error) {
78
+ return {
79
+ success: false,
80
+ error: error instanceof Error ? error.message : 'Error extracting meta'
81
+ };
82
+ }
83
+ }
84
+
85
+ export async function analyzeHeadings(params: { html: string }): Promise<ToolResult> {
86
+ const { html } = params;
87
+
88
+ try {
89
+ const $ = cheerio.load(html);
90
+ const headings: HeadingStructure[] = [];
91
+
92
+ $('h1, h2, h3, h4, h5, h6').each((_, el) => {
93
+ const tag = el.tagName.toLowerCase();
94
+ headings.push({
95
+ tag,
96
+ text: $(el).text().trim(),
97
+ level: parseInt(tag.charAt(1), 10),
98
+ });
99
+ });
100
+
101
+ return { success: true, data: headings };
102
+ } catch (error) {
103
+ return {
104
+ success: false,
105
+ error: error instanceof Error ? error.message : 'Error analyzing headings'
106
+ };
107
+ }
108
+ }
109
+
110
+ export async function extractImages(params: { html: string }): Promise<ToolResult> {
111
+ const { html } = params;
112
+
113
+ try {
114
+ const $ = cheerio.load(html);
115
+ const images: ImageInfo[] = [];
116
+
117
+ $('img').each((_, el) => {
118
+ images.push({
119
+ src: $(el).attr('src') || '',
120
+ alt: $(el).attr('alt') || null,
121
+ width: $(el).attr('width'),
122
+ height: $(el).attr('height'),
123
+ loading: $(el).attr('loading'),
124
+ });
125
+ });
126
+
127
+ return { success: true, data: images };
128
+ } catch (error) {
129
+ return {
130
+ success: false,
131
+ error: error instanceof Error ? error.message : 'Error extracting images'
132
+ };
133
+ }
134
+ }
135
+
136
+ export async function extractLinks(params: { html: string; baseUrl: string }): Promise<ToolResult> {
137
+ const { html, baseUrl } = params;
138
+
139
+ try {
140
+ const $ = cheerio.load(html);
141
+ const links: LinkInfo[] = [];
142
+ const baseHostname = new URL(baseUrl).hostname;
143
+
144
+ $('a[href]').each((_, el) => {
145
+ const href = $(el).attr('href') || '';
146
+ const rel = $(el).attr('rel') || '';
147
+
148
+ let isInternal = false;
149
+ try {
150
+ if (href.startsWith('/') || href.startsWith('#')) {
151
+ isInternal = true;
152
+ } else {
153
+ const linkHostname = new URL(href).hostname;
154
+ isInternal = linkHostname === baseHostname;
155
+ }
156
+ } catch {
157
+ isInternal = true; // Relative URLs are internal
158
+ }
159
+
160
+ links.push({
161
+ href,
162
+ text: $(el).text().trim(),
163
+ isInternal,
164
+ isNofollow: rel.includes('nofollow'),
165
+ });
166
+ });
167
+
168
+ return { success: true, data: links };
169
+ } catch (error) {
170
+ return {
171
+ success: false,
172
+ error: error instanceof Error ? error.message : 'Error extracting links'
173
+ };
174
+ }
175
+ }
176
+
177
+ export async function extractSchema(params: { html: string }): Promise<ToolResult> {
178
+ const { html } = params;
179
+
180
+ try {
181
+ const $ = cheerio.load(html);
182
+ const schemas: SchemaData[] = [];
183
+
184
+ $('script[type="application/ld+json"]').each((_, el) => {
185
+ try {
186
+ const content = $(el).html();
187
+ if (content) {
188
+ const data = JSON.parse(content);
189
+ schemas.push({
190
+ type: data['@type'] || 'Unknown',
191
+ data,
192
+ });
193
+ }
194
+ } catch {
195
+ // Skip invalid JSON-LD
196
+ }
197
+ });
198
+
199
+ return { success: true, data: schemas };
200
+ } catch (error) {
201
+ return {
202
+ success: false,
203
+ error: error instanceof Error ? error.message : 'Error extracting schema'
204
+ };
205
+ }
206
+ }
207
+
208
+ export async function checkRobots(params: { url: string }): Promise<ToolResult> {
209
+ try {
210
+ const baseUrl = new URL(params.url);
211
+ const robotsUrl = `${baseUrl.protocol}//${baseUrl.host}/robots.txt`;
212
+
213
+ const response = await httpGet<string>(robotsUrl, {
214
+
215
+ timeout: 10000,
216
+ validateStatus: () => true,
217
+ });
218
+
219
+ if (response.status === 200) {
220
+ return {
221
+ success: true,
222
+ data: {
223
+ exists: true,
224
+ content: response.data,
225
+ url: robotsUrl,
226
+ }
227
+ };
228
+ }
229
+
230
+ return {
231
+ success: true,
232
+ data: {
233
+ exists: false,
234
+ url: robotsUrl,
235
+ }
236
+ };
237
+ } catch (error) {
238
+ return {
239
+ success: false,
240
+ error: error instanceof Error ? error.message : 'Error checking robots.txt'
241
+ };
242
+ }
243
+ }
244
+
245
+ export async function checkSitemap(params: { url: string }): Promise<ToolResult> {
246
+ try {
247
+ const baseUrl = new URL(params.url);
248
+ const sitemapUrls = [
249
+ `${baseUrl.protocol}//${baseUrl.host}/sitemap.xml`,
250
+ `${baseUrl.protocol}//${baseUrl.host}/sitemap_index.xml`,
251
+ `${baseUrl.protocol}//${baseUrl.host}/sitemap/sitemap.xml`,
252
+ ];
253
+
254
+ for (const sitemapUrl of sitemapUrls) {
255
+ try {
256
+ const response = await httpGet<string>(sitemapUrl, {
257
+
258
+ timeout: 10000,
259
+ });
260
+
261
+ if (response.status === 200 && response.data.includes('<?xml')) {
262
+ // Parse sitemap to get URL count
263
+ const $ = cheerio.load(response.data, { xmlMode: true });
264
+ const urlCount = $('url').length || $('sitemap').length;
265
+
266
+ return {
267
+ success: true,
268
+ data: {
269
+ exists: true,
270
+ url: sitemapUrl,
271
+ urlCount,
272
+ content: response.data.substring(0, 2000), // First 2KB
273
+ }
274
+ };
275
+ }
276
+ } catch {
277
+ // Try next URL
278
+ }
279
+ }
280
+
281
+ return {
282
+ success: true,
283
+ data: {
284
+ exists: false,
285
+ }
286
+ };
287
+ } catch (error) {
288
+ return {
289
+ success: false,
290
+ error: error instanceof Error ? error.message : 'Error checking sitemap'
291
+ };
292
+ }
293
+ }
@@ -0,0 +1,301 @@
1
+ import { readFileSync, existsSync, readdirSync, statSync, writeFileSync, mkdirSync } from 'fs';
2
+ import { join, dirname, extname, basename } from 'path';
3
+ import type { ToolResult, FrameworkInfo } from '../types.js';
4
+
5
+ const FRAMEWORK_SIGNATURES = [
6
+ { pattern: 'next.config', name: 'Next.js', metaPattern: 'metadata-export' as const },
7
+ { pattern: 'astro.config', name: 'Astro', metaPattern: 'frontmatter' as const },
8
+ { pattern: 'nuxt.config', name: 'Nuxt', metaPattern: 'meta-function' as const },
9
+ { pattern: 'svelte.config', name: 'SvelteKit', metaPattern: 'head-component' as const },
10
+ { pattern: 'gatsby-config', name: 'Gatsby', metaPattern: 'head-component' as const },
11
+ { pattern: 'vite.config', name: 'Vite', metaPattern: 'html-head' as const },
12
+ { pattern: 'angular.json', name: 'Angular', metaPattern: 'html-head' as const },
13
+ { pattern: 'vue.config', name: 'Vue CLI', metaPattern: 'html-head' as const },
14
+ ];
15
+
16
+ export async function readFile(params: { path: string; cwd?: string }): Promise<ToolResult> {
17
+ const { path, cwd = process.cwd() } = params;
18
+ const fullPath = path.startsWith('/') ? path : join(cwd, path);
19
+
20
+ try {
21
+ if (!existsSync(fullPath)) {
22
+ return { success: false, error: `File not found: ${path}` };
23
+ }
24
+
25
+ const content = readFileSync(fullPath, 'utf-8');
26
+ return { success: true, data: { path: fullPath, content } };
27
+ } catch (error) {
28
+ return {
29
+ success: false,
30
+ error: error instanceof Error ? error.message : 'Error reading file'
31
+ };
32
+ }
33
+ }
34
+
35
+ export async function writeFile(params: { path: string; content: string; cwd?: string }): Promise<ToolResult> {
36
+ const { path, content, cwd = process.cwd() } = params;
37
+ const fullPath = path.startsWith('/') ? path : join(cwd, path);
38
+
39
+ try {
40
+ const dir = dirname(fullPath);
41
+ if (!existsSync(dir)) {
42
+ mkdirSync(dir, { recursive: true });
43
+ }
44
+
45
+ writeFileSync(fullPath, content, 'utf-8');
46
+ return { success: true, data: { path: fullPath, written: true } };
47
+ } catch (error) {
48
+ return {
49
+ success: false,
50
+ error: error instanceof Error ? error.message : 'Error writing file'
51
+ };
52
+ }
53
+ }
54
+
55
+ export async function listFiles(params: {
56
+ path: string;
57
+ cwd?: string;
58
+ pattern?: string;
59
+ recursive?: boolean;
60
+ }): Promise<ToolResult> {
61
+ const { path, cwd = process.cwd(), pattern, recursive = false } = params;
62
+ const fullPath = path.startsWith('/') ? path : join(cwd, path);
63
+
64
+ try {
65
+ if (!existsSync(fullPath)) {
66
+ return { success: false, error: `Directory not found: ${path}` };
67
+ }
68
+
69
+ const files: string[] = [];
70
+
71
+ function scan(dir: string, relativeTo: string) {
72
+ const entries = readdirSync(dir);
73
+ for (const entry of entries) {
74
+ if (entry.startsWith('.') || entry === 'node_modules') continue;
75
+
76
+ const entryPath = join(dir, entry);
77
+ const relativePath = entryPath.replace(relativeTo + '/', '');
78
+ const stat = statSync(entryPath);
79
+
80
+ if (stat.isDirectory() && recursive) {
81
+ scan(entryPath, relativeTo);
82
+ } else if (stat.isFile()) {
83
+ if (!pattern || entry.match(new RegExp(pattern))) {
84
+ files.push(relativePath);
85
+ }
86
+ }
87
+ }
88
+ }
89
+
90
+ scan(fullPath, fullPath);
91
+
92
+ return { success: true, data: files };
93
+ } catch (error) {
94
+ return {
95
+ success: false,
96
+ error: error instanceof Error ? error.message : 'Error listing files'
97
+ };
98
+ }
99
+ }
100
+
101
+ export async function detectFramework(params: { cwd?: string }): Promise<ToolResult> {
102
+ const { cwd = process.cwd() } = params;
103
+
104
+ try {
105
+ // Check for config files
106
+ for (const fw of FRAMEWORK_SIGNATURES) {
107
+ const patterns = [
108
+ `${fw.pattern}.js`,
109
+ `${fw.pattern}.ts`,
110
+ `${fw.pattern}.mjs`,
111
+ `${fw.pattern}.cjs`,
112
+ fw.pattern,
113
+ ];
114
+
115
+ for (const p of patterns) {
116
+ if (existsSync(join(cwd, p))) {
117
+ const info: FrameworkInfo = {
118
+ name: fw.name,
119
+ metaPattern: fw.metaPattern,
120
+ };
121
+
122
+ // Check for Next.js app router vs pages router
123
+ if (fw.name === 'Next.js') {
124
+ if (existsSync(join(cwd, 'app'))) {
125
+ info.router = 'app';
126
+ info.metaPattern = 'metadata-export';
127
+ } else if (existsSync(join(cwd, 'pages')) || existsSync(join(cwd, 'src/pages'))) {
128
+ info.router = 'pages';
129
+ info.metaPattern = 'head-component';
130
+ }
131
+ }
132
+
133
+ return { success: true, data: info };
134
+ }
135
+ }
136
+ }
137
+
138
+ // Check package.json for dependencies
139
+ const pkgPath = join(cwd, 'package.json');
140
+ if (existsSync(pkgPath)) {
141
+ const pkg = JSON.parse(readFileSync(pkgPath, 'utf-8'));
142
+ const deps = { ...pkg.dependencies, ...pkg.devDependencies };
143
+
144
+ if (deps['next']) {
145
+ const hasAppDir = existsSync(join(cwd, 'app')) || existsSync(join(cwd, 'src/app'));
146
+ return {
147
+ success: true,
148
+ data: {
149
+ name: 'Next.js',
150
+ router: hasAppDir ? 'app' : 'pages',
151
+ metaPattern: hasAppDir ? 'metadata-export' : 'head-component',
152
+ } as FrameworkInfo
153
+ };
154
+ }
155
+ if (deps['astro']) {
156
+ return { success: true, data: { name: 'Astro', metaPattern: 'frontmatter' } as FrameworkInfo };
157
+ }
158
+ if (deps['nuxt']) {
159
+ return { success: true, data: { name: 'Nuxt', metaPattern: 'meta-function' } as FrameworkInfo };
160
+ }
161
+ if (deps['@sveltejs/kit']) {
162
+ return { success: true, data: { name: 'SvelteKit', metaPattern: 'head-component' } as FrameworkInfo };
163
+ }
164
+ if (deps['gatsby']) {
165
+ return { success: true, data: { name: 'Gatsby', metaPattern: 'head-component' } as FrameworkInfo };
166
+ }
167
+
168
+ // Check for Vite with React
169
+ if (deps['vite'] && deps['react']) {
170
+ return {
171
+ success: true,
172
+ data: {
173
+ name: 'Vite',
174
+ metaPattern: 'html-head',
175
+ } as FrameworkInfo
176
+ };
177
+ }
178
+
179
+ // Just Vite
180
+ if (deps['vite']) {
181
+ return { success: true, data: { name: 'Vite', metaPattern: 'html-head' } as FrameworkInfo };
182
+ }
183
+
184
+ // Plain React
185
+ if (deps['react']) {
186
+ return { success: true, data: { name: 'React', metaPattern: 'html-head' } as FrameworkInfo };
187
+ }
188
+
189
+ // Vue
190
+ if (deps['vue']) {
191
+ return { success: true, data: { name: 'Vue', metaPattern: 'html-head' } as FrameworkInfo };
192
+ }
193
+ }
194
+
195
+ return {
196
+ success: true,
197
+ data: { name: 'Unknown', metaPattern: 'html-head' } as FrameworkInfo
198
+ };
199
+ } catch (error) {
200
+ return {
201
+ success: false,
202
+ error: error instanceof Error ? error.message : 'Error detecting framework'
203
+ };
204
+ }
205
+ }
206
+
207
+ export async function findHtmlEntry(params: { cwd?: string }): Promise<ToolResult> {
208
+ const { cwd = process.cwd() } = params;
209
+
210
+ const possiblePaths = [
211
+ 'index.html',
212
+ 'public/index.html',
213
+ 'src/index.html',
214
+ ];
215
+
216
+ for (const p of possiblePaths) {
217
+ const fullPath = join(cwd, p);
218
+ if (existsSync(fullPath)) {
219
+ const content = readFileSync(fullPath, 'utf-8');
220
+ return { success: true, data: { path: p, fullPath, content } };
221
+ }
222
+ }
223
+
224
+ return { success: false, error: 'No HTML entry point found' };
225
+ }
226
+
227
+ export async function findPageFiles(params: { cwd?: string; framework?: string }): Promise<ToolResult> {
228
+ const { cwd = process.cwd(), framework } = params;
229
+
230
+ const pageExtensions = ['.tsx', '.jsx', '.ts', '.js', '.astro', '.svelte', '.vue'];
231
+ const pages: { path: string; route: string }[] = [];
232
+
233
+ // Determine where to look based on framework
234
+ const searchDirs: string[] = [];
235
+
236
+ if (framework === 'Next.js') {
237
+ if (existsSync(join(cwd, 'app'))) {
238
+ searchDirs.push('app');
239
+ }
240
+ if (existsSync(join(cwd, 'src/app'))) {
241
+ searchDirs.push('src/app');
242
+ }
243
+ if (existsSync(join(cwd, 'pages'))) {
244
+ searchDirs.push('pages');
245
+ }
246
+ if (existsSync(join(cwd, 'src/pages'))) {
247
+ searchDirs.push('src/pages');
248
+ }
249
+ } else if (framework === 'Astro') {
250
+ searchDirs.push('src/pages');
251
+ } else if (framework === 'SvelteKit') {
252
+ searchDirs.push('src/routes');
253
+ } else if (framework === 'Nuxt') {
254
+ searchDirs.push('pages');
255
+ } else {
256
+ // Generic - check common locations
257
+ searchDirs.push('src', 'pages', 'app');
258
+ }
259
+
260
+ function isPageFile(filename: string): boolean {
261
+ const base = basename(filename, extname(filename));
262
+ // Common page file patterns
263
+ return pageExtensions.includes(extname(filename)) &&
264
+ (base === 'page' || base === 'index' || base.startsWith('+page') ||
265
+ !base.startsWith('_') && !base.startsWith('['));
266
+ }
267
+
268
+ function scanDir(dir: string, basePath: string) {
269
+ if (!existsSync(dir)) return;
270
+
271
+ const entries = readdirSync(dir);
272
+ for (const entry of entries) {
273
+ if (entry.startsWith('.') || entry === 'node_modules' || entry === 'components') continue;
274
+
275
+ const entryPath = join(dir, entry);
276
+ const stat = statSync(entryPath);
277
+
278
+ if (stat.isDirectory()) {
279
+ scanDir(entryPath, join(basePath, entry));
280
+ } else if (isPageFile(entry)) {
281
+ const relativePath = join(basePath, entry);
282
+ let route = '/' + basePath.replace(/\\/g, '/');
283
+ if (route === '/') route = '/';
284
+
285
+ pages.push({
286
+ path: relativePath,
287
+ route: route === '/.' ? '/' : route,
288
+ });
289
+ }
290
+ }
291
+ }
292
+
293
+ for (const searchDir of searchDirs) {
294
+ const fullSearchDir = join(cwd, searchDir);
295
+ if (existsSync(fullSearchDir)) {
296
+ scanDir(fullSearchDir, '');
297
+ }
298
+ }
299
+
300
+ return { success: true, data: pages };
301
+ }