@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,501 @@
1
+ /**
2
+ * Content Generator Module
3
+ *
4
+ * Generates SEO-optimized content including blog posts, changelogs,
5
+ * and README files. Supports GEO (Generative Engine Optimization)
6
+ * for AI-friendly content.
7
+ */
8
+
9
+ export interface BlogPostConfig {
10
+ topic: string;
11
+ keywords?: string[];
12
+ tone?: 'technical' | 'educational' | 'casual' | 'professional';
13
+ wordCount?: { min: number; max: number };
14
+ includeFAQ?: boolean;
15
+ geoOptimize?: boolean;
16
+ }
17
+
18
+ export interface ChangelogConfig {
19
+ fromRef: string;
20
+ toRef: string;
21
+ groupByType?: boolean;
22
+ seoKeywords?: string[];
23
+ includeBreaking?: boolean;
24
+ }
25
+
26
+ export interface ReadmeConfig {
27
+ currentContent: string;
28
+ projectName: string;
29
+ keywords?: string[];
30
+ preserveCustomSections?: boolean;
31
+ }
32
+
33
+ export interface GeneratedContent {
34
+ title: string;
35
+ content: string;
36
+ metaDescription: string;
37
+ keywords: string[];
38
+ readingTime?: string;
39
+ wordCount?: number;
40
+ hasFAQ?: boolean;
41
+ error?: string;
42
+ }
43
+
44
+ export interface ChangelogResult {
45
+ content: string;
46
+ version?: string;
47
+ date?: string;
48
+ sections: {
49
+ features?: string[];
50
+ fixes?: string[];
51
+ breaking?: string[];
52
+ docs?: string[];
53
+ other?: string[];
54
+ };
55
+ metaDescription?: string;
56
+ }
57
+
58
+ export interface ReadmeResult {
59
+ content: string;
60
+ metaDescription?: string;
61
+ suggestions: string[];
62
+ }
63
+
64
+ export interface SEOScore {
65
+ overall: number;
66
+ headingScore: number;
67
+ keywordScore: number;
68
+ lengthScore: number;
69
+ linkScore: number;
70
+ mediaScore: number;
71
+ }
72
+
73
+ export interface ContentGeneratorOptions {
74
+ llm?: LLMFunction;
75
+ exec?: ExecFunction;
76
+ }
77
+
78
+ export type LLMFunction = (prompt: string) => Promise<any>;
79
+ export type ExecFunction = (cmd: string) => Promise<{ stdout: string; stderr?: string }>;
80
+
81
+ /**
82
+ * Generate an SEO-optimized blog post
83
+ */
84
+ export async function generateBlogPost(
85
+ config: BlogPostConfig,
86
+ options: ContentGeneratorOptions = {}
87
+ ): Promise<GeneratedContent> {
88
+ const llm = options.llm;
89
+
90
+ if (!llm) {
91
+ return {
92
+ title: '',
93
+ content: '',
94
+ metaDescription: '',
95
+ keywords: [],
96
+ error: 'LLM function not provided',
97
+ };
98
+ }
99
+
100
+ try {
101
+ const prompt = buildBlogPrompt(config);
102
+ const result = await llm(prompt);
103
+
104
+ return {
105
+ title: result.title || '',
106
+ content: result.content || '',
107
+ metaDescription: (result.metaDescription || '').substring(0, 160),
108
+ keywords: result.keywords || [],
109
+ readingTime: result.readingTime,
110
+ wordCount: result.wordCount,
111
+ hasFAQ: result.hasFAQ,
112
+ };
113
+ } catch (error) {
114
+ return {
115
+ title: '',
116
+ content: '',
117
+ metaDescription: '',
118
+ keywords: [],
119
+ error: error instanceof Error ? error.message : 'Unknown error',
120
+ };
121
+ }
122
+ }
123
+
124
+ /**
125
+ * Build the prompt for blog post generation
126
+ */
127
+ function buildBlogPrompt(config: BlogPostConfig): string {
128
+ let prompt = `Generate an SEO-optimized blog post about: ${config.topic}
129
+
130
+ Requirements:
131
+ - Create an engaging title
132
+ - Write comprehensive content`;
133
+
134
+ if (config.keywords && config.keywords.length > 0) {
135
+ prompt += `\n- Include these keywords naturally: ${config.keywords.join(', ')}`;
136
+ }
137
+
138
+ if (config.tone) {
139
+ prompt += `\n- Use a ${config.tone} tone`;
140
+ }
141
+
142
+ if (config.wordCount) {
143
+ prompt += `\n- Target ${config.wordCount.min}-${config.wordCount.max} words`;
144
+ }
145
+
146
+ if (config.includeFAQ) {
147
+ prompt += `\n- Include an FAQ section`;
148
+ }
149
+
150
+ if (config.geoOptimize) {
151
+ prompt += `\n- Structure content for AI citation with bold key facts
152
+ - Use clear, citable statements
153
+ - Include structured data-friendly formatting`;
154
+ }
155
+
156
+ prompt += `\n- Generate a meta description under 160 characters
157
+ - Return as JSON with: title, content, metaDescription, keywords`;
158
+
159
+ return prompt;
160
+ }
161
+
162
+ /**
163
+ * Generate a changelog from git commits
164
+ */
165
+ export async function generateChangelog(
166
+ config: ChangelogConfig,
167
+ options: ContentGeneratorOptions = {}
168
+ ): Promise<ChangelogResult> {
169
+ const exec = options.exec;
170
+
171
+ if (!exec) {
172
+ return {
173
+ content: '',
174
+ sections: {},
175
+ };
176
+ }
177
+
178
+ try {
179
+ const result = await exec(
180
+ `git log ${config.fromRef}..${config.toRef} --pretty=format:"%s"`
181
+ );
182
+
183
+ const commits = result.stdout.trim().split('\n').filter(Boolean);
184
+
185
+ if (commits.length === 0) {
186
+ return {
187
+ content: '',
188
+ sections: {},
189
+ };
190
+ }
191
+
192
+ const sections = categorizeCommits(commits);
193
+ const content = formatChangelog(sections, config);
194
+ const version = extractVersion(config.toRef);
195
+
196
+ return {
197
+ content,
198
+ version,
199
+ date: new Date().toISOString().split('T')[0],
200
+ sections,
201
+ metaDescription: config.seoKeywords
202
+ ? `Release notes including ${config.seoKeywords.join(', ')}`
203
+ : undefined,
204
+ };
205
+ } catch (error) {
206
+ return {
207
+ content: '',
208
+ sections: {},
209
+ };
210
+ }
211
+ }
212
+
213
+ /**
214
+ * Categorize commits by conventional commit type
215
+ */
216
+ function categorizeCommits(commits: string[]): ChangelogResult['sections'] {
217
+ const sections: ChangelogResult['sections'] = {
218
+ features: [],
219
+ fixes: [],
220
+ breaking: [],
221
+ docs: [],
222
+ other: [],
223
+ };
224
+
225
+ for (const commit of commits) {
226
+ const lowerCommit = commit.toLowerCase();
227
+
228
+ if (lowerCommit.startsWith('feat')) {
229
+ sections.features!.push(commit);
230
+ } else if (lowerCommit.startsWith('fix')) {
231
+ sections.fixes!.push(commit);
232
+ } else if (lowerCommit.includes('breaking') || lowerCommit.startsWith('!')) {
233
+ sections.breaking!.push(commit);
234
+ } else if (lowerCommit.startsWith('docs')) {
235
+ sections.docs!.push(commit);
236
+ } else {
237
+ sections.other!.push(commit);
238
+ }
239
+ }
240
+
241
+ // Remove empty sections
242
+ for (const key of Object.keys(sections) as Array<keyof typeof sections>) {
243
+ if (sections[key]?.length === 0) {
244
+ delete sections[key];
245
+ }
246
+ }
247
+
248
+ return sections;
249
+ }
250
+
251
+ /**
252
+ * Format changelog sections into markdown
253
+ */
254
+ function formatChangelog(
255
+ sections: ChangelogResult['sections'],
256
+ config: ChangelogConfig
257
+ ): string {
258
+ const lines: string[] = [];
259
+
260
+ if (sections.breaking && sections.breaking.length > 0) {
261
+ lines.push('## ⚠️ Breaking Changes\n');
262
+ for (const commit of sections.breaking) {
263
+ lines.push(`- ${commit}`);
264
+ }
265
+ lines.push('');
266
+ }
267
+
268
+ if (sections.features && sections.features.length > 0) {
269
+ lines.push('## ✨ Features\n');
270
+ for (const commit of sections.features) {
271
+ lines.push(`- ${commit}`);
272
+ }
273
+ lines.push('');
274
+ }
275
+
276
+ if (sections.fixes && sections.fixes.length > 0) {
277
+ lines.push('## 🐛 Bug Fixes\n');
278
+ for (const commit of sections.fixes) {
279
+ lines.push(`- ${commit}`);
280
+ }
281
+ lines.push('');
282
+ }
283
+
284
+ if (sections.docs && sections.docs.length > 0) {
285
+ lines.push('## 📚 Documentation\n');
286
+ for (const commit of sections.docs) {
287
+ lines.push(`- ${commit}`);
288
+ }
289
+ lines.push('');
290
+ }
291
+
292
+ return lines.join('\n').trim();
293
+ }
294
+
295
+ /**
296
+ * Extract version from git ref
297
+ */
298
+ function extractVersion(ref: string): string | undefined {
299
+ const match = ref.match(/v?(\d+\.\d+\.\d+)/);
300
+ return match ? match[1] : undefined;
301
+ }
302
+
303
+ /**
304
+ * Optimize a README file for SEO
305
+ */
306
+ export async function optimizeReadme(
307
+ config: ReadmeConfig,
308
+ options: ContentGeneratorOptions = {}
309
+ ): Promise<ReadmeResult> {
310
+ const llm = options.llm;
311
+
312
+ if (!llm) {
313
+ return {
314
+ content: config.currentContent,
315
+ suggestions: ['LLM function not provided'],
316
+ };
317
+ }
318
+
319
+ try {
320
+ const prompt = buildReadmePrompt(config);
321
+ const result = await llm(prompt);
322
+
323
+ return {
324
+ content: result.content || config.currentContent,
325
+ metaDescription: result.metaDescription,
326
+ suggestions: result.suggestions || [],
327
+ };
328
+ } catch (error) {
329
+ return {
330
+ content: config.currentContent,
331
+ suggestions: ['Error optimizing README'],
332
+ };
333
+ }
334
+ }
335
+
336
+ /**
337
+ * Build the prompt for README optimization
338
+ */
339
+ function buildReadmePrompt(config: ReadmeConfig): string {
340
+ let prompt = `Optimize this README for SEO and developer experience.
341
+
342
+ Project Name: ${config.projectName}
343
+
344
+ Current README:
345
+ ${config.currentContent || '(empty)'}
346
+
347
+ Requirements:
348
+ - Add standard sections if missing (Installation, Usage, License)
349
+ - Make description SEO-friendly`;
350
+
351
+ if (config.keywords && config.keywords.length > 0) {
352
+ prompt += `\n- Include keywords: ${config.keywords.join(', ')}`;
353
+ }
354
+
355
+ if (config.preserveCustomSections) {
356
+ prompt += `\n- Preserve existing custom sections`;
357
+ }
358
+
359
+ prompt += `\n- Suggest improvements
360
+ - Return as JSON with: content, metaDescription, suggestions`;
361
+
362
+ return prompt;
363
+ }
364
+
365
+ /**
366
+ * Score content for SEO quality
367
+ */
368
+ export function scoreContentSEO(
369
+ content: string,
370
+ options: { targetKeywords?: string[] } = {}
371
+ ): SEOScore {
372
+ const headingScore = scoreHeadings(content);
373
+ const keywordScore = scoreKeywords(content, options.targetKeywords || []);
374
+ const lengthScore = scoreLength(content);
375
+ const linkScore = scoreLinks(content);
376
+ const mediaScore = scoreMedia(content);
377
+
378
+ const overall = Math.round(
379
+ headingScore * 0.2 +
380
+ keywordScore * 0.25 +
381
+ lengthScore * 0.25 +
382
+ linkScore * 0.15 +
383
+ mediaScore * 0.15
384
+ );
385
+
386
+ return {
387
+ overall,
388
+ headingScore,
389
+ keywordScore,
390
+ lengthScore,
391
+ linkScore,
392
+ mediaScore,
393
+ };
394
+ }
395
+
396
+ /**
397
+ * Score heading structure
398
+ */
399
+ function scoreHeadings(content: string): number {
400
+ let score = 0;
401
+
402
+ // Check for H1
403
+ if (content.match(/^#\s+/m)) {
404
+ score += 40;
405
+ }
406
+
407
+ // Check for H2s
408
+ const h2Count = (content.match(/^##\s+/gm) || []).length;
409
+ if (h2Count >= 1) score += 20;
410
+ if (h2Count >= 2) score += 20;
411
+
412
+ // Check for proper hierarchy (H1 before H2)
413
+ const h1Index = content.search(/^#\s+/m);
414
+ const h2Index = content.search(/^##\s+/m);
415
+ if (h1Index !== -1 && (h2Index === -1 || h1Index < h2Index)) {
416
+ score += 20;
417
+ }
418
+
419
+ return Math.min(100, score);
420
+ }
421
+
422
+ /**
423
+ * Score keyword usage
424
+ */
425
+ function scoreKeywords(content: string, keywords: string[]): number {
426
+ if (keywords.length === 0) {
427
+ return 50; // Neutral score if no keywords specified
428
+ }
429
+
430
+ const lowerContent = content.toLowerCase();
431
+ const wordCount = content.split(/\s+/).length;
432
+ let score = 0;
433
+
434
+ for (const keyword of keywords) {
435
+ const regex = new RegExp(keyword.toLowerCase(), 'gi');
436
+ const matches = lowerContent.match(regex) || [];
437
+ const density = (matches.length / wordCount) * 100;
438
+
439
+ // Ideal keyword density is 1-3%
440
+ if (density >= 0.5 && density <= 3) {
441
+ score += 100 / keywords.length;
442
+ } else if (density > 0 && density < 0.5) {
443
+ score += 50 / keywords.length;
444
+ } else if (density > 3) {
445
+ score += 30 / keywords.length; // Keyword stuffing penalty
446
+ }
447
+ }
448
+
449
+ return Math.min(100, Math.round(score));
450
+ }
451
+
452
+ /**
453
+ * Score content length
454
+ */
455
+ function scoreLength(content: string): number {
456
+ const wordCount = content.split(/\s+/).length;
457
+
458
+ // Ideal blog post length: 1500-2500 words
459
+ // Minimum effective: 300 words
460
+ if (wordCount < 100) return 10;
461
+ if (wordCount < 300) return 30;
462
+ if (wordCount < 500) return 50;
463
+ if (wordCount < 1000) return 70;
464
+ if (wordCount < 1500) return 85;
465
+ if (wordCount <= 2500) return 100;
466
+ return 90; // Slightly penalize very long content
467
+ }
468
+
469
+ /**
470
+ * Score internal links
471
+ */
472
+ function scoreLinks(content: string): number {
473
+ // Match markdown links
474
+ const links = content.match(/\[([^\]]+)\]\(([^)]+)\)/g) || [];
475
+ const internalLinks = links.filter(link => {
476
+ const url = link.match(/\]\(([^)]+)\)/)?.[1] || '';
477
+ return url.startsWith('/') || url.startsWith('#') || url.startsWith('./');
478
+ });
479
+
480
+ if (internalLinks.length === 0) return 20;
481
+ if (internalLinks.length === 1) return 50;
482
+ if (internalLinks.length === 2) return 75;
483
+ return 100;
484
+ }
485
+
486
+ /**
487
+ * Score media usage (images with alt text)
488
+ */
489
+ function scoreMedia(content: string): number {
490
+ // Match markdown images with alt text
491
+ const images = content.match(/!\[([^\]]+)\]\(([^)]+)\)/g) || [];
492
+ const imagesWithAlt = images.filter(img => {
493
+ const alt = img.match(/!\[([^\]]+)\]/)?.[1] || '';
494
+ return alt.length > 0;
495
+ });
496
+
497
+ if (images.length === 0) return 30; // No images
498
+ if (imagesWithAlt.length === 0) return 10; // Images without alt
499
+ if (imagesWithAlt.length < images.length) return 50; // Some missing alt
500
+ return 100; // All images have alt text
501
+ }