@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.
- package/README.md +242 -0
- package/dist/analyzer-2CSWIQGD.mjs +6 -0
- package/dist/chunk-YNZYHEYM.mjs +774 -0
- package/dist/index.d.mts +4012 -0
- package/dist/index.d.ts +4012 -0
- package/dist/index.js +29672 -0
- package/dist/index.mjs +28602 -0
- package/package.json +53 -0
- package/scripts/build-deno.ts +134 -0
- package/src/audit/ai/analyzer.ts +347 -0
- package/src/audit/ai/index.ts +29 -0
- package/src/audit/ai/prompts/content-analysis.ts +271 -0
- package/src/audit/ai/types.ts +179 -0
- package/src/audit/checks/additional-checks.ts +439 -0
- package/src/audit/checks/ai-citation-worthiness.ts +399 -0
- package/src/audit/checks/ai-content-structure.ts +325 -0
- package/src/audit/checks/ai-readiness.ts +339 -0
- package/src/audit/checks/anchor-text.ts +179 -0
- package/src/audit/checks/answer-conciseness.ts +322 -0
- package/src/audit/checks/asset-minification.ts +270 -0
- package/src/audit/checks/bing-optimization.ts +206 -0
- package/src/audit/checks/brand-mention-optimization.ts +349 -0
- package/src/audit/checks/caching-headers.ts +305 -0
- package/src/audit/checks/canonical-advanced.ts +150 -0
- package/src/audit/checks/canonical-domain.ts +196 -0
- package/src/audit/checks/citation-quality.ts +358 -0
- package/src/audit/checks/client-rendering.ts +542 -0
- package/src/audit/checks/color-contrast.ts +342 -0
- package/src/audit/checks/content-freshness.ts +170 -0
- package/src/audit/checks/content-science.ts +589 -0
- package/src/audit/checks/conversion-elements.ts +526 -0
- package/src/audit/checks/crawlability.ts +220 -0
- package/src/audit/checks/directory-listing.ts +172 -0
- package/src/audit/checks/dom-analysis.ts +191 -0
- package/src/audit/checks/dom-size.ts +246 -0
- package/src/audit/checks/duplicate-content.ts +194 -0
- package/src/audit/checks/eeat-signals.ts +990 -0
- package/src/audit/checks/entity-seo.ts +396 -0
- package/src/audit/checks/featured-snippet.ts +473 -0
- package/src/audit/checks/freshness-signals.ts +443 -0
- package/src/audit/checks/funnel-intent.ts +463 -0
- package/src/audit/checks/hreflang.ts +174 -0
- package/src/audit/checks/html-compliance.ts +302 -0
- package/src/audit/checks/image-dimensions.ts +167 -0
- package/src/audit/checks/images.ts +160 -0
- package/src/audit/checks/indexnow.ts +275 -0
- package/src/audit/checks/interactive-tools.ts +475 -0
- package/src/audit/checks/internal-link-graph.ts +436 -0
- package/src/audit/checks/keyword-analysis.ts +239 -0
- package/src/audit/checks/keyword-cannibalization.ts +385 -0
- package/src/audit/checks/keyword-placement.ts +471 -0
- package/src/audit/checks/links.ts +203 -0
- package/src/audit/checks/llms-txt.ts +224 -0
- package/src/audit/checks/local-seo.ts +296 -0
- package/src/audit/checks/mobile.ts +167 -0
- package/src/audit/checks/modern-images.ts +226 -0
- package/src/audit/checks/navboost-signals.ts +395 -0
- package/src/audit/checks/on-page.ts +209 -0
- package/src/audit/checks/page-resources.ts +285 -0
- package/src/audit/checks/pagination.ts +180 -0
- package/src/audit/checks/performance.ts +153 -0
- package/src/audit/checks/platform-presence.ts +580 -0
- package/src/audit/checks/redirect-analysis.ts +153 -0
- package/src/audit/checks/redirect-chain.ts +389 -0
- package/src/audit/checks/resource-hints.ts +420 -0
- package/src/audit/checks/responsive-css.ts +247 -0
- package/src/audit/checks/responsive-images.ts +396 -0
- package/src/audit/checks/review-ecosystem.ts +415 -0
- package/src/audit/checks/robots-validation.ts +373 -0
- package/src/audit/checks/security-headers.ts +172 -0
- package/src/audit/checks/security.ts +144 -0
- package/src/audit/checks/serp-preview.ts +251 -0
- package/src/audit/checks/site-maturity.ts +444 -0
- package/src/audit/checks/social-meta.test.ts +275 -0
- package/src/audit/checks/social-meta.ts +134 -0
- package/src/audit/checks/soft-404.ts +151 -0
- package/src/audit/checks/structured-data.ts +238 -0
- package/src/audit/checks/tech-detection.ts +496 -0
- package/src/audit/checks/topical-clusters.ts +435 -0
- package/src/audit/checks/tracker-bloat.ts +462 -0
- package/src/audit/checks/tracking-verification.test.ts +371 -0
- package/src/audit/checks/tracking-verification.ts +636 -0
- package/src/audit/checks/url-safety.ts +682 -0
- package/src/audit/deno-entry.ts +66 -0
- package/src/audit/discovery/index.ts +15 -0
- package/src/audit/discovery/link-crawler.ts +232 -0
- package/src/audit/discovery/repo-routes.ts +347 -0
- package/src/audit/engine.ts +620 -0
- package/src/audit/fixes/index.ts +209 -0
- package/src/audit/fixes/social-meta-fixes.test.ts +329 -0
- package/src/audit/fixes/social-meta-fixes.ts +463 -0
- package/src/audit/index.ts +74 -0
- package/src/audit/runner.test.ts +299 -0
- package/src/audit/runner.ts +130 -0
- package/src/audit/types.ts +1953 -0
- package/src/content/featured-snippet.ts +367 -0
- package/src/content/generator.test.ts +534 -0
- package/src/content/generator.ts +501 -0
- package/src/content/headline.ts +317 -0
- package/src/content/index.ts +62 -0
- package/src/content/intent.ts +258 -0
- package/src/content/keyword-density.ts +349 -0
- package/src/content/readability.ts +262 -0
- package/src/executor.ts +336 -0
- package/src/fixer.ts +416 -0
- package/src/frameworks/detector.test.ts +248 -0
- package/src/frameworks/detector.ts +371 -0
- package/src/frameworks/index.ts +68 -0
- package/src/frameworks/recipes/angular.yaml +171 -0
- package/src/frameworks/recipes/astro.yaml +206 -0
- package/src/frameworks/recipes/django.yaml +180 -0
- package/src/frameworks/recipes/laravel.yaml +137 -0
- package/src/frameworks/recipes/nextjs.yaml +268 -0
- package/src/frameworks/recipes/nuxt.yaml +175 -0
- package/src/frameworks/recipes/rails.yaml +188 -0
- package/src/frameworks/recipes/react.yaml +202 -0
- package/src/frameworks/recipes/sveltekit.yaml +154 -0
- package/src/frameworks/recipes/vue.yaml +137 -0
- package/src/frameworks/recipes/wordpress.yaml +209 -0
- package/src/frameworks/suggestion-engine.ts +320 -0
- package/src/geo/geo-content.test.ts +305 -0
- package/src/geo/geo-content.ts +266 -0
- package/src/geo/geo-history.test.ts +473 -0
- package/src/geo/geo-history.ts +433 -0
- package/src/geo/geo-tracker.test.ts +359 -0
- package/src/geo/geo-tracker.ts +411 -0
- package/src/geo/index.ts +10 -0
- package/src/git/commit-helper.test.ts +261 -0
- package/src/git/commit-helper.ts +329 -0
- package/src/git/index.ts +12 -0
- package/src/git/pr-helper.test.ts +284 -0
- package/src/git/pr-helper.ts +307 -0
- package/src/index.ts +66 -0
- package/src/keywords/ai-keyword-engine.ts +1062 -0
- package/src/keywords/ai-summarizer.ts +387 -0
- package/src/keywords/ci-mode.ts +555 -0
- package/src/keywords/engine.ts +359 -0
- package/src/keywords/index.ts +151 -0
- package/src/keywords/llm-judge.ts +357 -0
- package/src/keywords/nlp-analysis.ts +706 -0
- package/src/keywords/prioritizer.ts +295 -0
- package/src/keywords/site-crawler.ts +342 -0
- package/src/keywords/sources/autocomplete.ts +139 -0
- package/src/keywords/sources/competitive-search.ts +450 -0
- package/src/keywords/sources/competitor-analysis.ts +374 -0
- package/src/keywords/sources/dataforseo.ts +206 -0
- package/src/keywords/sources/free-sources.ts +294 -0
- package/src/keywords/sources/gsc.ts +123 -0
- package/src/keywords/topic-grouping.ts +327 -0
- package/src/keywords/types.ts +144 -0
- package/src/keywords/wizard.ts +457 -0
- package/src/loader.ts +40 -0
- package/src/reports/index.ts +7 -0
- package/src/reports/report-generator.test.ts +293 -0
- package/src/reports/report-generator.ts +713 -0
- package/src/scheduler/alerts.test.ts +458 -0
- package/src/scheduler/alerts.ts +328 -0
- package/src/scheduler/index.ts +8 -0
- package/src/scheduler/scheduled-audit.test.ts +377 -0
- package/src/scheduler/scheduled-audit.ts +149 -0
- package/src/test/integration-test.ts +325 -0
- package/src/tools/analyzer.ts +373 -0
- package/src/tools/crawl.ts +293 -0
- package/src/tools/files.ts +301 -0
- package/src/tools/h1-fixer.ts +249 -0
- package/src/tools/index.ts +67 -0
- package/src/tracking/github-action.ts +326 -0
- package/src/tracking/google-analytics.ts +265 -0
- package/src/tracking/index.ts +45 -0
- package/src/tracking/report-generator.ts +386 -0
- package/src/tracking/search-console.ts +335 -0
- package/src/types.ts +134 -0
- package/src/utils/http.ts +302 -0
- package/src/wasm-adapter.ts +297 -0
- package/src/wasm-entry.ts +14 -0
- package/tsconfig.json +17 -0
- package/tsup.wasm.config.ts +26 -0
- 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
|
+
}
|