@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,443 @@
1
+ // Freshness Signals & QDF Optimization
2
+ // Reference: "Query Deserves Freshness" Google algorithm
3
+ // AI assistants cite content 25.7% fresher than organic results
4
+ // dateModified in schema boosts AI citation odds by ~18%
5
+
6
+ import * as cheerio from 'cheerio';
7
+ import type { AuditIssue } from '../types.js';
8
+
9
+ export interface FreshnessSignalsData {
10
+ dateSignals: {
11
+ hasDatePublished: boolean;
12
+ hasDateModified: boolean;
13
+ hasSchemaDatePublished: boolean;
14
+ hasSchemaDateModified: boolean;
15
+ visibleDateOnPage: string | null;
16
+ schemaDatePublished: string | null;
17
+ schemaDateModified: string | null;
18
+ daysSincePublished: number | null;
19
+ daysSinceModified: number | null;
20
+ };
21
+ freshnessIndicators: {
22
+ hasCurrentYearMention: boolean;
23
+ hasRecentDateReferences: boolean;
24
+ hasUpdateNotice: boolean;
25
+ hasChangeLog: boolean;
26
+ hasVersionNumber: boolean;
27
+ contentFreshnessSignal: 'fresh' | 'dated' | 'evergreen' | 'unknown';
28
+ };
29
+ qdfRelevance: {
30
+ isLikelyQDFTopic: boolean;
31
+ qdfIndicators: string[];
32
+ updateFrequencyRecommendation: string;
33
+ };
34
+ aiCitationOptimization: {
35
+ hasLastUpdatedMeta: boolean;
36
+ hasClearDateFormat: boolean;
37
+ freshnessSchemaComplete: boolean;
38
+ aiCitationReadiness: 'high' | 'medium' | 'low';
39
+ };
40
+ freshnessScore: number; // 0-100
41
+ recommendations: string[];
42
+ }
43
+
44
+ /**
45
+ * Extract date signals from page
46
+ */
47
+ function extractDateSignals($: cheerio.CheerioAPI, html: string): FreshnessSignalsData['dateSignals'] {
48
+ const now = new Date();
49
+
50
+ // Look for dates in schema
51
+ let schemaDatePublished: string | null = null;
52
+ let schemaDateModified: string | null = null;
53
+ let hasSchemaDatePublished = false;
54
+ let hasSchemaDateModified = false;
55
+
56
+ $('script[type="application/ld+json"]').each((_, el) => {
57
+ try {
58
+ const content = $(el).html() || '';
59
+ const data = JSON.parse(content);
60
+
61
+ const processSchema = (schema: Record<string, unknown>) => {
62
+ if (schema.datePublished) {
63
+ hasSchemaDatePublished = true;
64
+ schemaDatePublished = schema.datePublished as string;
65
+ }
66
+ if (schema.dateModified) {
67
+ hasSchemaDateModified = true;
68
+ schemaDateModified = schema.dateModified as string;
69
+ }
70
+ };
71
+
72
+ if (Array.isArray(data)) {
73
+ data.forEach(item => processSchema(item as Record<string, unknown>));
74
+ } else if (data['@graph']) {
75
+ (data['@graph'] as Record<string, unknown>[]).forEach(processSchema);
76
+ } else {
77
+ processSchema(data);
78
+ }
79
+ } catch {
80
+ // Invalid JSON
81
+ }
82
+ });
83
+
84
+ // Look for meta tags with dates
85
+ const metaDatePublished = $('meta[property="article:published_time"], meta[name="date"], meta[name="pubdate"]').attr('content');
86
+ const metaDateModified = $('meta[property="article:modified_time"], meta[name="lastmod"]').attr('content');
87
+
88
+ // Look for visible date on page
89
+ const datePatterns = [
90
+ /(?:published|posted|written|created)(?:\s+on)?:?\s*(\w+\s+\d{1,2},?\s+\d{4}|\d{4}-\d{2}-\d{2})/i,
91
+ /(?:updated|modified|last\s+updated|revised)(?:\s+on)?:?\s*(\w+\s+\d{1,2},?\s+\d{4}|\d{4}-\d{2}-\d{2})/i,
92
+ /(\w+\s+\d{1,2},?\s+\d{4})/,
93
+ /(\d{4}-\d{2}-\d{2})/,
94
+ ];
95
+
96
+ const bodyText = $('body').text();
97
+ let visibleDateOnPage: string | null = null;
98
+
99
+ for (const pattern of datePatterns) {
100
+ const match = bodyText.match(pattern);
101
+ if (match) {
102
+ visibleDateOnPage = match[1] || match[0];
103
+ break;
104
+ }
105
+ }
106
+
107
+ // Calculate days since dates
108
+ let daysSincePublished: number | null = null;
109
+ let daysSinceModified: number | null = null;
110
+
111
+ const publishedDate = schemaDatePublished || metaDatePublished;
112
+ if (publishedDate) {
113
+ try {
114
+ const date = new Date(publishedDate);
115
+ daysSincePublished = Math.floor((now.getTime() - date.getTime()) / (1000 * 60 * 60 * 24));
116
+ } catch {
117
+ // Invalid date
118
+ }
119
+ }
120
+
121
+ const modifiedDate = schemaDateModified || metaDateModified;
122
+ if (modifiedDate) {
123
+ try {
124
+ const date = new Date(modifiedDate);
125
+ daysSinceModified = Math.floor((now.getTime() - date.getTime()) / (1000 * 60 * 60 * 24));
126
+ } catch {
127
+ // Invalid date
128
+ }
129
+ }
130
+
131
+ return {
132
+ hasDatePublished: !!(metaDatePublished || schemaDatePublished),
133
+ hasDateModified: !!(metaDateModified || schemaDateModified),
134
+ hasSchemaDatePublished,
135
+ hasSchemaDateModified,
136
+ visibleDateOnPage,
137
+ schemaDatePublished,
138
+ schemaDateModified,
139
+ daysSincePublished,
140
+ daysSinceModified,
141
+ };
142
+ }
143
+
144
+ /**
145
+ * Analyze freshness indicators in content
146
+ */
147
+ function analyzeFreshnessIndicators($: cheerio.CheerioAPI, html: string): FreshnessSignalsData['freshnessIndicators'] {
148
+ const bodyText = $('body').text();
149
+ const title = $('title').text();
150
+ const currentYear = new Date().getFullYear();
151
+
152
+ // Check for current year mention
153
+ const hasCurrentYearMention =
154
+ bodyText.includes(currentYear.toString()) ||
155
+ title.includes(currentYear.toString());
156
+
157
+ // Check for recent date references (within last 6 months)
158
+ const recentDatePatterns = [
159
+ new RegExp(`(${currentYear}|${currentYear - 1})`),
160
+ /this year|this month|recently|latest|newest|updated/i,
161
+ ];
162
+ const hasRecentDateReferences = recentDatePatterns.some(p => p.test(bodyText));
163
+
164
+ // Check for update notice
165
+ const hasUpdateNotice =
166
+ /updated?:?\s*(on|in)?\s*\w+\s+\d{1,2},?\s+\d{4}/i.test(bodyText) ||
167
+ /last\s+(?:updated|modified|revised)/i.test(bodyText) ||
168
+ $('[class*="update"], [class*="modified"]').length > 0;
169
+
170
+ // Check for changelog
171
+ const hasChangeLog =
172
+ /changelog|change\s+log|revision\s+history|update\s+history/i.test(bodyText) ||
173
+ $('[id*="changelog"], [class*="changelog"]').length > 0;
174
+
175
+ // Check for version number
176
+ const hasVersionNumber = /v?\d+\.\d+(\.\d+)?|version\s+\d+/i.test(bodyText);
177
+
178
+ // Determine content freshness signal
179
+ let contentFreshnessSignal: 'fresh' | 'dated' | 'evergreen' | 'unknown' = 'unknown';
180
+
181
+ if (hasCurrentYearMention && hasRecentDateReferences) {
182
+ contentFreshnessSignal = 'fresh';
183
+ } else if (/\d{4}/.test(title) && !new RegExp(`(${currentYear}|${currentYear - 1})`).test(title)) {
184
+ contentFreshnessSignal = 'dated';
185
+ } else if (/guide|tutorial|how\s+to|tips|basics|fundamentals/i.test(title)) {
186
+ contentFreshnessSignal = 'evergreen';
187
+ }
188
+
189
+ return {
190
+ hasCurrentYearMention,
191
+ hasRecentDateReferences,
192
+ hasUpdateNotice,
193
+ hasChangeLog,
194
+ hasVersionNumber,
195
+ contentFreshnessSignal,
196
+ };
197
+ }
198
+
199
+ /**
200
+ * Analyze QDF relevance
201
+ */
202
+ function analyzeQDFRelevance($: cheerio.CheerioAPI, html: string): FreshnessSignalsData['qdfRelevance'] {
203
+ const title = $('title').text().toLowerCase();
204
+ const bodyText = $('body').text().toLowerCase();
205
+
206
+ const qdfIndicators: string[] = [];
207
+ let updateFrequency = 'monthly';
208
+
209
+ // QDF topic patterns
210
+ if (/news|breaking|announcement|release|launch/i.test(title)) {
211
+ qdfIndicators.push('News/announcement topic');
212
+ updateFrequency = 'daily';
213
+ }
214
+ if (/\d{4}|trends?|statistics?|data|report/i.test(title)) {
215
+ qdfIndicators.push('Year-specific or data content');
216
+ updateFrequency = 'quarterly';
217
+ }
218
+ if (/review|comparison|vs\.?|alternative/i.test(title)) {
219
+ qdfIndicators.push('Review/comparison content');
220
+ updateFrequency = 'monthly';
221
+ }
222
+ if (/price|pricing|cost/i.test(title)) {
223
+ qdfIndicators.push('Pricing information');
224
+ updateFrequency = 'monthly';
225
+ }
226
+ if (/best|top\s+\d+/i.test(title)) {
227
+ qdfIndicators.push('Best-of list content');
228
+ updateFrequency = 'quarterly';
229
+ }
230
+ if (/election|score|result|update/i.test(bodyText)) {
231
+ qdfIndicators.push('Time-sensitive topic signals');
232
+ updateFrequency = 'weekly';
233
+ }
234
+
235
+ return {
236
+ isLikelyQDFTopic: qdfIndicators.length > 0,
237
+ qdfIndicators,
238
+ updateFrequencyRecommendation: updateFrequency,
239
+ };
240
+ }
241
+
242
+ /**
243
+ * Analyze AI citation optimization
244
+ */
245
+ function analyzeAICitationOptimization(
246
+ dateSignals: FreshnessSignalsData['dateSignals'],
247
+ freshnessIndicators: FreshnessSignalsData['freshnessIndicators']
248
+ ): FreshnessSignalsData['aiCitationOptimization'] {
249
+ const hasLastUpdatedMeta = dateSignals.hasSchemaDateModified || dateSignals.hasDateModified;
250
+ const hasClearDateFormat = !!(dateSignals.schemaDatePublished || dateSignals.schemaDateModified);
251
+ const freshnessSchemaComplete = dateSignals.hasSchemaDatePublished && dateSignals.hasSchemaDateModified;
252
+
253
+ let readinessScore = 0;
254
+ if (hasLastUpdatedMeta) readinessScore += 30;
255
+ if (hasClearDateFormat) readinessScore += 20;
256
+ if (freshnessSchemaComplete) readinessScore += 30;
257
+ if (freshnessIndicators.hasCurrentYearMention) readinessScore += 10;
258
+ if (freshnessIndicators.hasUpdateNotice) readinessScore += 10;
259
+
260
+ const aiCitationReadiness = readinessScore >= 70 ? 'high' : readinessScore >= 40 ? 'medium' : 'low';
261
+
262
+ return {
263
+ hasLastUpdatedMeta,
264
+ hasClearDateFormat,
265
+ freshnessSchemaComplete,
266
+ aiCitationReadiness,
267
+ };
268
+ }
269
+
270
+ /**
271
+ * Calculate overall freshness score
272
+ */
273
+ function calculateFreshnessScore(
274
+ dateSignals: FreshnessSignalsData['dateSignals'],
275
+ freshnessIndicators: FreshnessSignalsData['freshnessIndicators'],
276
+ aiOptimization: FreshnessSignalsData['aiCitationOptimization']
277
+ ): number {
278
+ let score = 50; // Baseline
279
+
280
+ // Date signals (max +30)
281
+ if (dateSignals.hasSchemaDatePublished) score += 10;
282
+ if (dateSignals.hasSchemaDateModified) score += 15;
283
+ if (dateSignals.visibleDateOnPage) score += 5;
284
+
285
+ // Freshness indicators (max +30)
286
+ if (freshnessIndicators.hasCurrentYearMention) score += 10;
287
+ if (freshnessIndicators.hasUpdateNotice) score += 10;
288
+ if (freshnessIndicators.contentFreshnessSignal === 'fresh') score += 10;
289
+ else if (freshnessIndicators.contentFreshnessSignal === 'dated') score -= 20;
290
+
291
+ // Recency penalty/boost
292
+ if (dateSignals.daysSinceModified !== null) {
293
+ if (dateSignals.daysSinceModified < 30) score += 10;
294
+ else if (dateSignals.daysSinceModified < 90) score += 5;
295
+ else if (dateSignals.daysSinceModified > 365) score -= 10;
296
+ }
297
+
298
+ return Math.max(0, Math.min(100, score));
299
+ }
300
+
301
+ /**
302
+ * Main function: Analyze freshness signals
303
+ */
304
+ export function analyzeFreshnessSignals(
305
+ html: string,
306
+ url: string
307
+ ): { issues: AuditIssue[]; data: FreshnessSignalsData } {
308
+ const $ = cheerio.load(html);
309
+ const issues: AuditIssue[] = [];
310
+
311
+ // Run all analyses
312
+ const dateSignals = extractDateSignals($, html);
313
+ const freshnessIndicators = analyzeFreshnessIndicators($, html);
314
+ const qdfRelevance = analyzeQDFRelevance($, html);
315
+ const aiCitationOptimization = analyzeAICitationOptimization(dateSignals, freshnessIndicators);
316
+
317
+ // Calculate score
318
+ const freshnessScore = calculateFreshnessScore(
319
+ dateSignals,
320
+ freshnessIndicators,
321
+ aiCitationOptimization
322
+ );
323
+
324
+ // Generate recommendations
325
+ const recommendations: string[] = [];
326
+
327
+ if (!dateSignals.hasSchemaDateModified) {
328
+ recommendations.push('Add dateModified to schema - boosts AI citation odds by ~18%');
329
+ }
330
+ if (qdfRelevance.isLikelyQDFTopic && !freshnessIndicators.hasUpdateNotice) {
331
+ recommendations.push(`This QDF topic should be updated ${qdfRelevance.updateFrequencyRecommendation}`);
332
+ }
333
+ if (!freshnessIndicators.hasCurrentYearMention && /\d{4}/.test($('title').text())) {
334
+ recommendations.push('Update year references in title and content for freshness signals');
335
+ }
336
+
337
+ // Generate issues
338
+
339
+ // No dateModified schema
340
+ if (!dateSignals.hasSchemaDateModified) {
341
+ issues.push({
342
+ code: 'NO_DATE_MODIFIED_SCHEMA',
343
+ severity: 'warning',
344
+ category: 'structured-data',
345
+ title: 'Missing dateModified in schema',
346
+ description: 'Schema markup lacks dateModified property.',
347
+ impact: 'dateModified signals content freshness to search engines and AI assistants, boosting citation odds by ~18%.',
348
+ howToFix: 'Add dateModified property to your Article/WebPage schema with ISO 8601 format.',
349
+ affectedUrls: [url],
350
+ details: {
351
+ hasDatePublished: dateSignals.hasSchemaDatePublished,
352
+ },
353
+ });
354
+ }
355
+
356
+ // Dated content on QDF topic
357
+ if (qdfRelevance.isLikelyQDFTopic && dateSignals.daysSinceModified && dateSignals.daysSinceModified > 180) {
358
+ issues.push({
359
+ code: 'STALE_QDF_CONTENT',
360
+ severity: 'warning',
361
+ category: 'content',
362
+ title: 'Time-sensitive content may be stale',
363
+ description: `Content was last modified ${dateSignals.daysSinceModified} days ago on a topic that benefits from freshness.`,
364
+ impact: 'Query Deserves Freshness (QDF) prioritizes fresh content for this topic type.',
365
+ howToFix: `Update this content ${qdfRelevance.updateFrequencyRecommendation}. Add new data, stats, or insights.`,
366
+ affectedUrls: [url],
367
+ details: {
368
+ daysSinceModified: dateSignals.daysSinceModified,
369
+ qdfIndicators: qdfRelevance.qdfIndicators,
370
+ updateFrequency: qdfRelevance.updateFrequencyRecommendation,
371
+ },
372
+ });
373
+ }
374
+
375
+ // Outdated year in title
376
+ const title = $('title').text();
377
+ const currentYear = new Date().getFullYear();
378
+ const yearMatch = title.match(/20\d{2}/);
379
+ if (yearMatch && parseInt(yearMatch[0]) < currentYear - 1) {
380
+ issues.push({
381
+ code: 'OUTDATED_YEAR_IN_TITLE',
382
+ severity: 'warning',
383
+ category: 'on-page',
384
+ title: 'Title contains outdated year',
385
+ description: `Title mentions ${yearMatch[0]} but current year is ${currentYear}.`,
386
+ impact: 'Outdated years in titles signal stale content and reduce CTR.',
387
+ howToFix: `Update title to ${currentYear} and refresh the content to match.`,
388
+ affectedUrls: [url],
389
+ details: {
390
+ yearInTitle: yearMatch[0],
391
+ currentYear,
392
+ },
393
+ });
394
+ }
395
+
396
+ // No visible date on content page
397
+ if (!dateSignals.visibleDateOnPage && /blog|article|post|news/i.test(url)) {
398
+ issues.push({
399
+ code: 'NO_VISIBLE_DATE',
400
+ severity: 'notice',
401
+ category: 'content',
402
+ title: 'No visible publish/update date',
403
+ description: 'Content page lacks a visible date for users.',
404
+ impact: 'Users prefer seeing when content was published/updated for credibility.',
405
+ howToFix: 'Display publish date and last updated date on the page.',
406
+ affectedUrls: [url],
407
+ });
408
+ }
409
+
410
+ // Low AI citation readiness
411
+ if (aiCitationOptimization.aiCitationReadiness === 'low' && qdfRelevance.isLikelyQDFTopic) {
412
+ issues.push({
413
+ code: 'LOW_AI_CITATION_READINESS',
414
+ severity: 'notice',
415
+ category: 'content',
416
+ title: 'Low AI citation readiness for fresh content',
417
+ description: 'This page lacks freshness signals that AI assistants prioritize.',
418
+ impact: 'AI assistants cite content 25.7% fresher than organic results. Missing signals reduce citation odds.',
419
+ howToFix: 'Add dateModified schema, visible update date, and current year mentions.',
420
+ affectedUrls: [url],
421
+ details: {
422
+ aiCitationReadiness: aiCitationOptimization.aiCitationReadiness,
423
+ missingSignals: [
424
+ !dateSignals.hasSchemaDateModified && 'dateModified schema',
425
+ !freshnessIndicators.hasUpdateNotice && 'visible update notice',
426
+ !freshnessIndicators.hasCurrentYearMention && 'current year mention',
427
+ ].filter(Boolean),
428
+ },
429
+ });
430
+ }
431
+
432
+ return {
433
+ issues,
434
+ data: {
435
+ dateSignals,
436
+ freshnessIndicators,
437
+ qdfRelevance,
438
+ aiCitationOptimization,
439
+ freshnessScore,
440
+ recommendations,
441
+ },
442
+ };
443
+ }