@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,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
+ }