@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,396 @@
|
|
|
1
|
+
// Entity SEO & Knowledge Graph Optimization
|
|
2
|
+
// Reference: "Entity-first SEO: How to align content with Google's Knowledge Graph"
|
|
3
|
+
// Google's Knowledge Graph understands entities better than keywords
|
|
4
|
+
// Three pillars: Precision, Coverage, Connectivity
|
|
5
|
+
|
|
6
|
+
import * as cheerio from 'cheerio';
|
|
7
|
+
import type { AuditIssue } from '../types.js';
|
|
8
|
+
|
|
9
|
+
export interface EntitySEOData {
|
|
10
|
+
entitySignals: {
|
|
11
|
+
hasOrganizationSchema: boolean;
|
|
12
|
+
hasPersonSchema: boolean;
|
|
13
|
+
hasProductSchema: boolean;
|
|
14
|
+
hasSameAsReferences: boolean;
|
|
15
|
+
sameAsUrls: string[];
|
|
16
|
+
hasMainEntityOfPage: boolean;
|
|
17
|
+
hasAboutReference: boolean;
|
|
18
|
+
};
|
|
19
|
+
entityClarity: {
|
|
20
|
+
primaryEntityType: string | null;
|
|
21
|
+
entityNameConsistency: boolean;
|
|
22
|
+
hasEntityDefinition: boolean;
|
|
23
|
+
hasWikipediaStyleIntro: boolean;
|
|
24
|
+
entityMentionCount: number;
|
|
25
|
+
};
|
|
26
|
+
topicalCoverage: {
|
|
27
|
+
hasRelatedEntities: boolean;
|
|
28
|
+
relatedEntityCount: number;
|
|
29
|
+
hasEntityRelationships: boolean;
|
|
30
|
+
topicalDepthScore: number; // 0-100
|
|
31
|
+
};
|
|
32
|
+
knowledgePanelSignals: {
|
|
33
|
+
hasCompanyInfo: boolean;
|
|
34
|
+
hasFounderInfo: boolean;
|
|
35
|
+
hasHeadquarters: boolean;
|
|
36
|
+
hasIndustryMention: boolean;
|
|
37
|
+
hasDateFounded: boolean;
|
|
38
|
+
};
|
|
39
|
+
entitySEOScore: number; // 0-100
|
|
40
|
+
recommendations: string[];
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* Extract entity signals from structured data
|
|
45
|
+
*/
|
|
46
|
+
function extractEntitySignals($: cheerio.CheerioAPI, html: string): EntitySEOData['entitySignals'] {
|
|
47
|
+
let hasOrganizationSchema = false;
|
|
48
|
+
let hasPersonSchema = false;
|
|
49
|
+
let hasProductSchema = false;
|
|
50
|
+
let hasSameAsReferences = false;
|
|
51
|
+
let hasMainEntityOfPage = false;
|
|
52
|
+
let hasAboutReference = false;
|
|
53
|
+
const sameAsUrls: string[] = [];
|
|
54
|
+
|
|
55
|
+
// Parse JSON-LD
|
|
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
|
+
const type = schema['@type'];
|
|
63
|
+
|
|
64
|
+
if (type === 'Organization' || type === 'Corporation' || type === 'LocalBusiness') {
|
|
65
|
+
hasOrganizationSchema = true;
|
|
66
|
+
}
|
|
67
|
+
if (type === 'Person') {
|
|
68
|
+
hasPersonSchema = true;
|
|
69
|
+
}
|
|
70
|
+
if (type === 'Product' || type === 'SoftwareApplication') {
|
|
71
|
+
hasProductSchema = true;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
if (schema.sameAs) {
|
|
75
|
+
hasSameAsReferences = true;
|
|
76
|
+
if (Array.isArray(schema.sameAs)) {
|
|
77
|
+
sameAsUrls.push(...(schema.sameAs as string[]));
|
|
78
|
+
} else if (typeof schema.sameAs === 'string') {
|
|
79
|
+
sameAsUrls.push(schema.sameAs);
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
if (schema.mainEntityOfPage) {
|
|
84
|
+
hasMainEntityOfPage = true;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
if (schema.about) {
|
|
88
|
+
hasAboutReference = true;
|
|
89
|
+
}
|
|
90
|
+
};
|
|
91
|
+
|
|
92
|
+
if (Array.isArray(data)) {
|
|
93
|
+
data.forEach(item => processSchema(item as Record<string, unknown>));
|
|
94
|
+
} else if (data['@graph']) {
|
|
95
|
+
(data['@graph'] as Record<string, unknown>[]).forEach(processSchema);
|
|
96
|
+
} else {
|
|
97
|
+
processSchema(data);
|
|
98
|
+
}
|
|
99
|
+
} catch {
|
|
100
|
+
// Invalid JSON
|
|
101
|
+
}
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
return {
|
|
105
|
+
hasOrganizationSchema,
|
|
106
|
+
hasPersonSchema,
|
|
107
|
+
hasProductSchema,
|
|
108
|
+
hasSameAsReferences,
|
|
109
|
+
sameAsUrls,
|
|
110
|
+
hasMainEntityOfPage,
|
|
111
|
+
hasAboutReference,
|
|
112
|
+
};
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* Analyze entity clarity on the page
|
|
117
|
+
*/
|
|
118
|
+
function analyzeEntityClarity($: cheerio.CheerioAPI, html: string): EntitySEOData['entityClarity'] {
|
|
119
|
+
const title = $('title').text().trim();
|
|
120
|
+
const h1 = $('h1').first().text().trim();
|
|
121
|
+
const metaDesc = $('meta[name="description"]').attr('content') || '';
|
|
122
|
+
const firstParagraph = $('p').first().text().trim();
|
|
123
|
+
|
|
124
|
+
// Detect primary entity type
|
|
125
|
+
let primaryEntityType: string | null = null;
|
|
126
|
+
const bodyText = $('body').text().toLowerCase();
|
|
127
|
+
|
|
128
|
+
if (/company|corporation|business|agency|firm/i.test(bodyText)) {
|
|
129
|
+
primaryEntityType = 'Organization';
|
|
130
|
+
} else if (/product|software|tool|app|service/i.test(h1 + ' ' + title)) {
|
|
131
|
+
primaryEntityType = 'Product/Software';
|
|
132
|
+
} else if (/person|author|founder|ceo|expert/i.test(bodyText)) {
|
|
133
|
+
primaryEntityType = 'Person';
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
// Check for entity name consistency
|
|
137
|
+
// The entity name should appear consistently in title, H1, and first paragraph
|
|
138
|
+
const titleWords = title.split(/[\s|—-]+/).filter(w => w.length > 3);
|
|
139
|
+
const h1Words = h1.split(/[\s|—-]+/).filter(w => w.length > 3);
|
|
140
|
+
|
|
141
|
+
// Find common proper nouns (capitalized words)
|
|
142
|
+
const properNouns = [...title.matchAll(/[A-Z][a-z]+(?:\s+[A-Z][a-z]+)*/g)].map(m => m[0]);
|
|
143
|
+
const entityNameConsistency = properNouns.length > 0 &&
|
|
144
|
+
properNouns.some(name => h1.includes(name) || firstParagraph.includes(name));
|
|
145
|
+
|
|
146
|
+
// Wikipedia-style intro (entity + "is a" definition)
|
|
147
|
+
const hasWikipediaStyleIntro =
|
|
148
|
+
/is (a|an|the)\s+\w+/i.test(firstParagraph) ||
|
|
149
|
+
/\w+ is (a|an) (software|company|tool|platform|service|agency)/i.test(firstParagraph);
|
|
150
|
+
|
|
151
|
+
// Entity definition present
|
|
152
|
+
const hasEntityDefinition =
|
|
153
|
+
hasWikipediaStyleIntro ||
|
|
154
|
+
firstParagraph.length > 100;
|
|
155
|
+
|
|
156
|
+
// Count entity mentions
|
|
157
|
+
const brandName = properNouns[0] || title.split(/[|—-]/)[0].trim();
|
|
158
|
+
const entityMentionCount = brandName ?
|
|
159
|
+
(bodyText.match(new RegExp(brandName.toLowerCase(), 'g')) || []).length : 0;
|
|
160
|
+
|
|
161
|
+
return {
|
|
162
|
+
primaryEntityType,
|
|
163
|
+
entityNameConsistency,
|
|
164
|
+
hasEntityDefinition,
|
|
165
|
+
hasWikipediaStyleIntro,
|
|
166
|
+
entityMentionCount,
|
|
167
|
+
};
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
/**
|
|
171
|
+
* Analyze topical coverage and entity relationships
|
|
172
|
+
*/
|
|
173
|
+
function analyzeTopicalCoverage($: cheerio.CheerioAPI, html: string): EntitySEOData['topicalCoverage'] {
|
|
174
|
+
// Look for related entity mentions (competitors, partners, integrations)
|
|
175
|
+
const bodyText = $('body').text();
|
|
176
|
+
|
|
177
|
+
// Common relationship patterns
|
|
178
|
+
const relationshipPatterns = [
|
|
179
|
+
/integrates? with/i,
|
|
180
|
+
/works? with/i,
|
|
181
|
+
/similar to/i,
|
|
182
|
+
/alternative to/i,
|
|
183
|
+
/compared to/i,
|
|
184
|
+
/partner(s|ship)? with/i,
|
|
185
|
+
/competitor/i,
|
|
186
|
+
/used by/i,
|
|
187
|
+
/trusted by/i,
|
|
188
|
+
];
|
|
189
|
+
|
|
190
|
+
const hasEntityRelationships = relationshipPatterns.some(p => p.test(bodyText));
|
|
191
|
+
|
|
192
|
+
// Count related entities (look for proper nouns that aren't the main entity)
|
|
193
|
+
const allProperNouns = bodyText.match(/[A-Z][a-z]+(?:\s+[A-Z][a-z]+)*/g) || [];
|
|
194
|
+
const uniqueEntities = [...new Set(allProperNouns)].filter(e => e.length > 3);
|
|
195
|
+
const relatedEntityCount = Math.min(uniqueEntities.length, 50); // Cap at 50
|
|
196
|
+
|
|
197
|
+
const hasRelatedEntities = relatedEntityCount > 5;
|
|
198
|
+
|
|
199
|
+
// Topical depth score
|
|
200
|
+
let topicalDepthScore = 0;
|
|
201
|
+
if (hasEntityRelationships) topicalDepthScore += 30;
|
|
202
|
+
if (relatedEntityCount >= 10) topicalDepthScore += 30;
|
|
203
|
+
else if (relatedEntityCount >= 5) topicalDepthScore += 15;
|
|
204
|
+
if ($('h2, h3').length >= 5) topicalDepthScore += 20;
|
|
205
|
+
if (bodyText.length > 5000) topicalDepthScore += 20;
|
|
206
|
+
|
|
207
|
+
return {
|
|
208
|
+
hasRelatedEntities,
|
|
209
|
+
relatedEntityCount,
|
|
210
|
+
hasEntityRelationships,
|
|
211
|
+
topicalDepthScore: Math.min(topicalDepthScore, 100),
|
|
212
|
+
};
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
/**
|
|
216
|
+
* Analyze Knowledge Panel signals
|
|
217
|
+
*/
|
|
218
|
+
function analyzeKnowledgePanelSignals($: cheerio.CheerioAPI, html: string): EntitySEOData['knowledgePanelSignals'] {
|
|
219
|
+
const bodyText = $('body').text().toLowerCase();
|
|
220
|
+
const schemaText = $('script[type="application/ld+json"]').text();
|
|
221
|
+
|
|
222
|
+
return {
|
|
223
|
+
hasCompanyInfo: /about us|about the company|our story|who we are/i.test(bodyText) ||
|
|
224
|
+
/Organization|Corporation/i.test(schemaText),
|
|
225
|
+
hasFounderInfo: /founder|founded by|ceo|chief executive/i.test(bodyText),
|
|
226
|
+
hasHeadquarters: /headquarters|based in|located in|office/i.test(bodyText) ||
|
|
227
|
+
/address/i.test(schemaText),
|
|
228
|
+
hasIndustryMention: /industry|sector|market|specializ/i.test(bodyText),
|
|
229
|
+
hasDateFounded: /founded in|established in|since \d{4}|founded \d{4}/i.test(bodyText) ||
|
|
230
|
+
/foundingDate/i.test(schemaText),
|
|
231
|
+
};
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
/**
|
|
235
|
+
* Calculate overall Entity SEO score
|
|
236
|
+
*/
|
|
237
|
+
function calculateEntitySEOScore(
|
|
238
|
+
signals: EntitySEOData['entitySignals'],
|
|
239
|
+
clarity: EntitySEOData['entityClarity'],
|
|
240
|
+
coverage: EntitySEOData['topicalCoverage'],
|
|
241
|
+
kpSignals: EntitySEOData['knowledgePanelSignals']
|
|
242
|
+
): number {
|
|
243
|
+
let score = 0;
|
|
244
|
+
|
|
245
|
+
// Entity signals (max 30)
|
|
246
|
+
if (signals.hasOrganizationSchema || signals.hasPersonSchema) score += 10;
|
|
247
|
+
if (signals.hasSameAsReferences) score += 10;
|
|
248
|
+
if (signals.hasMainEntityOfPage) score += 5;
|
|
249
|
+
if (signals.hasAboutReference) score += 5;
|
|
250
|
+
|
|
251
|
+
// Entity clarity (max 30)
|
|
252
|
+
if (clarity.primaryEntityType) score += 10;
|
|
253
|
+
if (clarity.entityNameConsistency) score += 10;
|
|
254
|
+
if (clarity.hasWikipediaStyleIntro) score += 10;
|
|
255
|
+
|
|
256
|
+
// Topical coverage (max 20)
|
|
257
|
+
score += Math.round(coverage.topicalDepthScore * 0.2);
|
|
258
|
+
|
|
259
|
+
// Knowledge panel signals (max 20)
|
|
260
|
+
const kpCount = Object.values(kpSignals).filter(Boolean).length;
|
|
261
|
+
score += kpCount * 4;
|
|
262
|
+
|
|
263
|
+
return Math.min(score, 100);
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
/**
|
|
267
|
+
* Main function: Analyze Entity SEO optimization
|
|
268
|
+
*/
|
|
269
|
+
export function analyzeEntitySEO(
|
|
270
|
+
html: string,
|
|
271
|
+
url: string
|
|
272
|
+
): { issues: AuditIssue[]; data: EntitySEOData } {
|
|
273
|
+
const $ = cheerio.load(html);
|
|
274
|
+
const issues: AuditIssue[] = [];
|
|
275
|
+
|
|
276
|
+
// Run all analyses
|
|
277
|
+
const entitySignals = extractEntitySignals($, html);
|
|
278
|
+
const entityClarity = analyzeEntityClarity($, html);
|
|
279
|
+
const topicalCoverage = analyzeTopicalCoverage($, html);
|
|
280
|
+
const knowledgePanelSignals = analyzeKnowledgePanelSignals($, html);
|
|
281
|
+
|
|
282
|
+
// Calculate score
|
|
283
|
+
const entitySEOScore = calculateEntitySEOScore(
|
|
284
|
+
entitySignals,
|
|
285
|
+
entityClarity,
|
|
286
|
+
topicalCoverage,
|
|
287
|
+
knowledgePanelSignals
|
|
288
|
+
);
|
|
289
|
+
|
|
290
|
+
// Generate recommendations
|
|
291
|
+
const recommendations: string[] = [];
|
|
292
|
+
|
|
293
|
+
if (!entitySignals.hasOrganizationSchema && !entitySignals.hasPersonSchema) {
|
|
294
|
+
recommendations.push('Add Organization or Person schema to establish entity identity');
|
|
295
|
+
}
|
|
296
|
+
if (!entitySignals.hasSameAsReferences) {
|
|
297
|
+
recommendations.push('Add sameAs references to official social profiles and Wikipedia/Wikidata');
|
|
298
|
+
}
|
|
299
|
+
if (!entityClarity.hasWikipediaStyleIntro) {
|
|
300
|
+
recommendations.push('Add a Wikipedia-style definition in the first paragraph (Entity is a...)');
|
|
301
|
+
}
|
|
302
|
+
if (!entitySignals.hasMainEntityOfPage) {
|
|
303
|
+
recommendations.push('Specify mainEntityOfPage in schema to clarify what the page is about');
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
// Generate issues
|
|
307
|
+
|
|
308
|
+
// No entity schema
|
|
309
|
+
if (!entitySignals.hasOrganizationSchema && !entitySignals.hasPersonSchema && !entitySignals.hasProductSchema) {
|
|
310
|
+
issues.push({
|
|
311
|
+
code: 'NO_ENTITY_SCHEMA',
|
|
312
|
+
severity: 'warning',
|
|
313
|
+
category: 'structured-data',
|
|
314
|
+
title: 'Missing entity schema (Organization/Person/Product)',
|
|
315
|
+
description: 'No schema markup defining the primary entity on this page.',
|
|
316
|
+
impact: 'Entity schema helps Google understand what/who this page is about for Knowledge Graph.',
|
|
317
|
+
howToFix: 'Add Organization, Person, or Product schema with complete properties.',
|
|
318
|
+
affectedUrls: [url],
|
|
319
|
+
});
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
// No sameAs references
|
|
323
|
+
if (!entitySignals.hasSameAsReferences && (entitySignals.hasOrganizationSchema || entitySignals.hasPersonSchema)) {
|
|
324
|
+
issues.push({
|
|
325
|
+
code: 'NO_SAMEAS_REFERENCES',
|
|
326
|
+
severity: 'notice',
|
|
327
|
+
category: 'structured-data',
|
|
328
|
+
title: 'Entity schema missing sameAs references',
|
|
329
|
+
description: 'Entity schema found but no sameAs links to authoritative sources.',
|
|
330
|
+
impact: 'sameAs helps Google connect your entity to Knowledge Graph entries (Wikipedia, social profiles).',
|
|
331
|
+
howToFix: 'Add sameAs array with links to Wikipedia, Wikidata, LinkedIn, Twitter, Crunchbase, etc.',
|
|
332
|
+
affectedUrls: [url],
|
|
333
|
+
details: {
|
|
334
|
+
hasOrganization: entitySignals.hasOrganizationSchema,
|
|
335
|
+
hasPerson: entitySignals.hasPersonSchema,
|
|
336
|
+
},
|
|
337
|
+
});
|
|
338
|
+
}
|
|
339
|
+
|
|
340
|
+
// No Wikipedia-style intro
|
|
341
|
+
if (!entityClarity.hasWikipediaStyleIntro) {
|
|
342
|
+
issues.push({
|
|
343
|
+
code: 'NO_ENTITY_DEFINITION',
|
|
344
|
+
severity: 'notice',
|
|
345
|
+
category: 'content',
|
|
346
|
+
title: 'Missing entity definition in content',
|
|
347
|
+
description: 'First paragraph lacks a clear definition of the primary entity.',
|
|
348
|
+
impact: 'Clear entity definitions help AI systems extract and cite your content correctly.',
|
|
349
|
+
howToFix: 'Start content with "[Entity Name] is a [type] that [does what]" pattern.',
|
|
350
|
+
affectedUrls: [url],
|
|
351
|
+
});
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
// Low entity consistency
|
|
355
|
+
if (!entityClarity.entityNameConsistency && entityClarity.entityMentionCount < 3) {
|
|
356
|
+
issues.push({
|
|
357
|
+
code: 'LOW_ENTITY_CONSISTENCY',
|
|
358
|
+
severity: 'notice',
|
|
359
|
+
category: 'content',
|
|
360
|
+
title: 'Entity name not consistently used',
|
|
361
|
+
description: 'The primary entity name doesn\'t appear consistently across title, H1, and content.',
|
|
362
|
+
impact: 'Consistent entity naming strengthens topical signals for Knowledge Graph.',
|
|
363
|
+
howToFix: 'Ensure the exact entity name appears in title, H1, and first paragraph.',
|
|
364
|
+
affectedUrls: [url],
|
|
365
|
+
details: {
|
|
366
|
+
entityMentionCount: entityClarity.entityMentionCount,
|
|
367
|
+
},
|
|
368
|
+
});
|
|
369
|
+
}
|
|
370
|
+
|
|
371
|
+
// No mainEntityOfPage
|
|
372
|
+
if (!entitySignals.hasMainEntityOfPage && entitySignals.hasOrganizationSchema) {
|
|
373
|
+
issues.push({
|
|
374
|
+
code: 'NO_MAIN_ENTITY_OF_PAGE',
|
|
375
|
+
severity: 'notice',
|
|
376
|
+
category: 'structured-data',
|
|
377
|
+
title: 'Schema missing mainEntityOfPage property',
|
|
378
|
+
description: 'Entity schema doesn\'t specify what the page is primarily about.',
|
|
379
|
+
impact: 'mainEntityOfPage helps Google understand the primary topic of each page.',
|
|
380
|
+
howToFix: 'Add mainEntityOfPage property pointing to the current URL or a @id reference.',
|
|
381
|
+
affectedUrls: [url],
|
|
382
|
+
});
|
|
383
|
+
}
|
|
384
|
+
|
|
385
|
+
return {
|
|
386
|
+
issues,
|
|
387
|
+
data: {
|
|
388
|
+
entitySignals,
|
|
389
|
+
entityClarity,
|
|
390
|
+
topicalCoverage,
|
|
391
|
+
knowledgePanelSignals,
|
|
392
|
+
entitySEOScore,
|
|
393
|
+
recommendations,
|
|
394
|
+
},
|
|
395
|
+
};
|
|
396
|
+
}
|