@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,475 @@
1
+ // Interactive Tools & Dwell Time Detection
2
+ // Reference: "6 Advanced SEO Tips for 2026"
3
+ // "Adding free tools on your website is a fantastic way to attract target customers"
4
+ // "Increase the dwell time on your website because people are using the free tools"
5
+ // "Generate backlinks - free tools are link magnets"
6
+
7
+ import * as cheerio from 'cheerio';
8
+ import type { AuditIssue } from '../types.js';
9
+
10
+ export interface InteractiveElement {
11
+ type: 'calculator' | 'quiz' | 'form' | 'widget' | 'embed' | 'tool' | 'interactive';
12
+ description: string;
13
+ location: string;
14
+ hasUserInput: boolean;
15
+ }
16
+
17
+ export interface InteractiveToolsData {
18
+ hasInteractiveTools: boolean;
19
+ tools: InteractiveElement[];
20
+ dwellTimeSignals: {
21
+ hasCalculator: boolean;
22
+ hasQuiz: boolean;
23
+ hasInteractiveWidget: boolean;
24
+ hasVideoEmbed: boolean;
25
+ hasCodePlayground: boolean;
26
+ hasConfigurator: boolean;
27
+ hasChatWidget: boolean;
28
+ };
29
+ toolCount: number;
30
+ dwellTimeScore: number; // 0-100
31
+ recommendations: string[];
32
+ }
33
+
34
+ // Patterns for detecting interactive tools
35
+ const CALCULATOR_PATTERNS = [
36
+ /calculator/i,
37
+ /estimator/i,
38
+ /cost-calc/i,
39
+ /price-calc/i,
40
+ /roi-calc/i,
41
+ /savings/i,
42
+ /compute/i,
43
+ ];
44
+
45
+ const QUIZ_PATTERNS = [
46
+ /quiz/i,
47
+ /assessment/i,
48
+ /survey/i,
49
+ /questionnaire/i,
50
+ /test-your/i,
51
+ /find-out/i,
52
+ ];
53
+
54
+ const WIDGET_PATTERNS = [
55
+ /widget/i,
56
+ /tool/i,
57
+ /generator/i,
58
+ /builder/i,
59
+ /creator/i,
60
+ /converter/i,
61
+ /checker/i,
62
+ /analyzer/i,
63
+ /finder/i,
64
+ ];
65
+
66
+ const CONFIGURATOR_PATTERNS = [
67
+ /configurator/i,
68
+ /customizer/i,
69
+ /designer/i,
70
+ /planner/i,
71
+ /selector/i,
72
+ /picker/i,
73
+ ];
74
+
75
+ /**
76
+ * Detect calculators on the page
77
+ */
78
+ export function detectCalculators($: cheerio.CheerioAPI): InteractiveElement[] {
79
+ const calculators: InteractiveElement[] = [];
80
+
81
+ // Check for calculator-related elements by class/id
82
+ const calcElements = $('[class*="calc"], [id*="calc"], [class*="estimat"], [id*="estimat"], [data-tool="calculator"]');
83
+ calcElements.each((_, el) => {
84
+ const $el = $(el);
85
+ const hasInputs = $el.find('input, select, [type="range"]').length > 0;
86
+ if (hasInputs) {
87
+ calculators.push({
88
+ type: 'calculator',
89
+ description: $el.attr('aria-label') || $el.find('h2, h3, h4').first().text().trim() || 'Calculator widget',
90
+ location: $el.attr('id') || $el.attr('class')?.split(' ')[0] || 'unknown',
91
+ hasUserInput: true,
92
+ });
93
+ }
94
+ });
95
+
96
+ // Check for form elements that look like calculators
97
+ $('form').each((_, form) => {
98
+ const $form = $(form);
99
+ const formText = ($form.attr('class') || '') + ($form.attr('id') || '') + $form.text();
100
+ if (CALCULATOR_PATTERNS.some(p => p.test(formText))) {
101
+ const hasInputs = $form.find('input[type="number"], input[type="range"], select').length > 0;
102
+ if (hasInputs) {
103
+ calculators.push({
104
+ type: 'calculator',
105
+ description: $form.attr('aria-label') || $form.find('h2, h3, h4').first().text().trim() || 'Calculator form',
106
+ location: $form.attr('id') || 'form',
107
+ hasUserInput: true,
108
+ });
109
+ }
110
+ }
111
+ });
112
+
113
+ return calculators;
114
+ }
115
+
116
+ /**
117
+ * Detect quizzes and assessments
118
+ */
119
+ export function detectQuizzes($: cheerio.CheerioAPI): InteractiveElement[] {
120
+ const quizzes: InteractiveElement[] = [];
121
+
122
+ // Check for quiz-related elements
123
+ const quizElements = $('[class*="quiz"], [id*="quiz"], [class*="assessment"], [data-component="quiz"]');
124
+ quizElements.each((_, el) => {
125
+ const $el = $(el);
126
+ quizzes.push({
127
+ type: 'quiz',
128
+ description: $el.attr('aria-label') || $el.find('h2, h3').first().text().trim() || 'Quiz/Assessment',
129
+ location: $el.attr('id') || $el.attr('class')?.split(' ')[0] || 'unknown',
130
+ hasUserInput: true,
131
+ });
132
+ });
133
+
134
+ // Check for question patterns
135
+ const questionContainers = $('[class*="question"], [data-question]');
136
+ if (questionContainers.length >= 3) {
137
+ quizzes.push({
138
+ type: 'quiz',
139
+ description: 'Multi-question assessment',
140
+ location: 'multiple-questions',
141
+ hasUserInput: true,
142
+ });
143
+ }
144
+
145
+ return quizzes;
146
+ }
147
+
148
+ /**
149
+ * Detect interactive widgets
150
+ */
151
+ export function detectWidgets($: cheerio.CheerioAPI, html: string): InteractiveElement[] {
152
+ const widgets: InteractiveElement[] = [];
153
+
154
+ // Check for widget patterns in classes/ids
155
+ WIDGET_PATTERNS.forEach(pattern => {
156
+ const elements = $(`[class*="${pattern.source.replace(/\\i$/, '').toLowerCase()}"], [id*="${pattern.source.replace(/\\i$/, '').toLowerCase()}"]`);
157
+ elements.each((_, el) => {
158
+ const $el = $(el);
159
+ const hasInteraction = $el.find('input, button, select, [onclick], [role="button"]').length > 0;
160
+ if (hasInteraction) {
161
+ widgets.push({
162
+ type: 'widget',
163
+ description: $el.attr('aria-label') || $el.attr('title') || pattern.source,
164
+ location: $el.attr('id') || $el.attr('class')?.split(' ')[0] || 'unknown',
165
+ hasUserInput: hasInteraction,
166
+ });
167
+ }
168
+ });
169
+ });
170
+
171
+ // Check for configurators
172
+ CONFIGURATOR_PATTERNS.forEach(pattern => {
173
+ if (pattern.test(html)) {
174
+ const elements = $(`[class*="${pattern.source.replace(/\\i$/, '').toLowerCase()}"]`);
175
+ elements.each((_, el) => {
176
+ const $el = $(el);
177
+ widgets.push({
178
+ type: 'tool',
179
+ description: 'Product configurator/customizer',
180
+ location: $el.attr('id') || 'configurator',
181
+ hasUserInput: true,
182
+ });
183
+ });
184
+ }
185
+ });
186
+
187
+ return widgets;
188
+ }
189
+
190
+ /**
191
+ * Detect code playgrounds and sandboxes
192
+ */
193
+ export function detectCodePlaygrounds($: cheerio.CheerioAPI, html: string): InteractiveElement[] {
194
+ const playgrounds: InteractiveElement[] = [];
195
+
196
+ // Check for common code playground embeds
197
+ const playgroundPatterns = [
198
+ { name: 'CodePen', pattern: /codepen\.io\/.*\/embed/i },
199
+ { name: 'CodeSandbox', pattern: /codesandbox\.io\/embed/i },
200
+ { name: 'JSFiddle', pattern: /jsfiddle\.net.*\/embed/i },
201
+ { name: 'Replit', pattern: /replit\.com.*embed/i },
202
+ { name: 'StackBlitz', pattern: /stackblitz\.com\/edit/i },
203
+ ];
204
+
205
+ for (const { name, pattern } of playgroundPatterns) {
206
+ if (pattern.test(html)) {
207
+ playgrounds.push({
208
+ type: 'interactive',
209
+ description: `${name} code playground`,
210
+ location: 'embed',
211
+ hasUserInput: true,
212
+ });
213
+ }
214
+ }
215
+
216
+ // Check for Monaco editor or CodeMirror
217
+ if (html.includes('monaco-editor') || html.includes('CodeMirror')) {
218
+ playgrounds.push({
219
+ type: 'interactive',
220
+ description: 'Code editor widget',
221
+ location: 'embedded',
222
+ hasUserInput: true,
223
+ });
224
+ }
225
+
226
+ return playgrounds;
227
+ }
228
+
229
+ /**
230
+ * Detect chat widgets
231
+ */
232
+ export function detectChatWidgets(html: string): InteractiveElement[] {
233
+ const chatWidgets: InteractiveElement[] = [];
234
+
235
+ const chatPatterns = [
236
+ { name: 'Intercom', pattern: /intercom\.com|intercomcdn/i },
237
+ { name: 'Drift', pattern: /drift\.com|js\.driftt/i },
238
+ { name: 'Zendesk', pattern: /zendesk\.com|zdassets/i },
239
+ { name: 'Crisp', pattern: /crisp\.chat/i },
240
+ { name: 'LiveChat', pattern: /livechat\.com|livechatinc/i },
241
+ { name: 'Tawk.to', pattern: /tawk\.to/i },
242
+ { name: 'HubSpot Chat', pattern: /hs-scripts\.com.*chat/i },
243
+ { name: 'Freshdesk', pattern: /freshdesk\.com/i },
244
+ ];
245
+
246
+ for (const { name, pattern } of chatPatterns) {
247
+ if (pattern.test(html)) {
248
+ chatWidgets.push({
249
+ type: 'widget',
250
+ description: `${name} chat widget`,
251
+ location: 'global',
252
+ hasUserInput: true,
253
+ });
254
+ }
255
+ }
256
+
257
+ return chatWidgets;
258
+ }
259
+
260
+ /**
261
+ * Detect video embeds (for dwell time)
262
+ */
263
+ export function detectVideoEmbeds($: cheerio.CheerioAPI, html: string): InteractiveElement[] {
264
+ const videos: InteractiveElement[] = [];
265
+
266
+ // YouTube
267
+ if (html.includes('youtube.com/embed') || html.includes('youtube-nocookie.com/embed')) {
268
+ const youtubeEmbeds = $('iframe[src*="youtube"]');
269
+ videos.push({
270
+ type: 'embed',
271
+ description: `${youtubeEmbeds.length} YouTube video(s) embedded`,
272
+ location: 'content',
273
+ hasUserInput: false,
274
+ });
275
+ }
276
+
277
+ // Vimeo
278
+ if (html.includes('player.vimeo.com')) {
279
+ videos.push({
280
+ type: 'embed',
281
+ description: 'Vimeo video embedded',
282
+ location: 'content',
283
+ hasUserInput: false,
284
+ });
285
+ }
286
+
287
+ // Wistia
288
+ if (html.includes('wistia.com') || html.includes('wistia.net')) {
289
+ videos.push({
290
+ type: 'embed',
291
+ description: 'Wistia video embedded',
292
+ location: 'content',
293
+ hasUserInput: false,
294
+ });
295
+ }
296
+
297
+ // Loom
298
+ if (html.includes('loom.com/embed')) {
299
+ videos.push({
300
+ type: 'embed',
301
+ description: 'Loom video embedded',
302
+ location: 'content',
303
+ hasUserInput: false,
304
+ });
305
+ }
306
+
307
+ // HTML5 video
308
+ const html5Videos = $('video');
309
+ if (html5Videos.length > 0) {
310
+ videos.push({
311
+ type: 'embed',
312
+ description: `${html5Videos.length} HTML5 video(s)`,
313
+ location: 'content',
314
+ hasUserInput: false,
315
+ });
316
+ }
317
+
318
+ return videos;
319
+ }
320
+
321
+ /**
322
+ * Calculate dwell time score based on interactive elements
323
+ */
324
+ function calculateDwellTimeScore(data: Omit<InteractiveToolsData, 'dwellTimeScore' | 'recommendations'>): number {
325
+ let score = 0;
326
+
327
+ // Interactive tools (high value)
328
+ if (data.dwellTimeSignals.hasCalculator) score += 25;
329
+ if (data.dwellTimeSignals.hasQuiz) score += 20;
330
+ if (data.dwellTimeSignals.hasConfigurator) score += 25;
331
+ if (data.dwellTimeSignals.hasCodePlayground) score += 15;
332
+
333
+ // Video embeds (medium value)
334
+ if (data.dwellTimeSignals.hasVideoEmbed) score += 15;
335
+
336
+ // Widgets (lower value)
337
+ if (data.dwellTimeSignals.hasInteractiveWidget) score += 10;
338
+ if (data.dwellTimeSignals.hasChatWidget) score += 5;
339
+
340
+ return Math.min(score, 100);
341
+ }
342
+
343
+ /**
344
+ * Main function: Analyze interactive tools on the page
345
+ */
346
+ export function analyzeInteractiveTools(
347
+ html: string,
348
+ url: string
349
+ ): { issues: AuditIssue[]; data: InteractiveToolsData } {
350
+ const $ = cheerio.load(html);
351
+ const issues: AuditIssue[] = [];
352
+
353
+ // Detect all interactive elements
354
+ const calculators = detectCalculators($);
355
+ const quizzes = detectQuizzes($);
356
+ const widgets = detectWidgets($, html);
357
+ const codePlaygrounds = detectCodePlaygrounds($, html);
358
+ const chatWidgets = detectChatWidgets(html);
359
+ const videoEmbeds = detectVideoEmbeds($, html);
360
+
361
+ // Combine all tools
362
+ const allTools: InteractiveElement[] = [
363
+ ...calculators,
364
+ ...quizzes,
365
+ ...widgets,
366
+ ...codePlaygrounds,
367
+ ...chatWidgets,
368
+ ...videoEmbeds,
369
+ ];
370
+
371
+ // Deduplicate by location
372
+ const uniqueTools = allTools.filter((tool, index, self) =>
373
+ index === self.findIndex(t => t.location === tool.location && t.type === tool.type)
374
+ );
375
+
376
+ // Build dwell time signals
377
+ const dwellTimeSignals = {
378
+ hasCalculator: calculators.length > 0,
379
+ hasQuiz: quizzes.length > 0,
380
+ hasInteractiveWidget: widgets.length > 0,
381
+ hasVideoEmbed: videoEmbeds.length > 0,
382
+ hasCodePlayground: codePlaygrounds.length > 0,
383
+ hasConfigurator: widgets.some(w => w.description.toLowerCase().includes('configurator')),
384
+ hasChatWidget: chatWidgets.length > 0,
385
+ };
386
+
387
+ // Calculate dwell time score
388
+ const partialData = {
389
+ hasInteractiveTools: uniqueTools.length > 0,
390
+ tools: uniqueTools,
391
+ dwellTimeSignals,
392
+ toolCount: uniqueTools.length,
393
+ };
394
+ const dwellTimeScore = calculateDwellTimeScore(partialData);
395
+
396
+ // Generate recommendations
397
+ const recommendations: string[] = [];
398
+
399
+ if (!dwellTimeSignals.hasCalculator && !dwellTimeSignals.hasQuiz) {
400
+ recommendations.push('Add an interactive calculator or quiz to boost dwell time and generate backlinks');
401
+ }
402
+
403
+ if (!dwellTimeSignals.hasVideoEmbed) {
404
+ recommendations.push('Embed a YouTube video to increase engagement and dwell time');
405
+ }
406
+
407
+ if (!dwellTimeSignals.hasInteractiveWidget && !dwellTimeSignals.hasConfigurator) {
408
+ recommendations.push('Consider adding a free tool (generator, checker, converter) as a link magnet');
409
+ }
410
+
411
+ // Generate issues
412
+
413
+ // Check if page type suggests it should have interactive tools
414
+ const isServicePage = /services?|pricing|product/i.test(url) ||
415
+ $('h1').text().toLowerCase().match(/service|pricing|product/);
416
+ const isBlogPage = /blog|article|post/i.test(url);
417
+
418
+ // No interactive elements on service page
419
+ if (isServicePage && uniqueTools.length === 0) {
420
+ issues.push({
421
+ code: 'NO_INTERACTIVE_TOOLS',
422
+ severity: 'notice',
423
+ category: 'content',
424
+ title: 'Service page lacks interactive tools',
425
+ description: 'This service/product page has no interactive elements like calculators or configurators.',
426
+ impact: 'Interactive tools increase dwell time by 2-3x and are powerful link magnets for backlink acquisition.',
427
+ howToFix: 'Add a relevant calculator (ROI, cost estimator, savings) or product configurator to the page.',
428
+ affectedUrls: [url],
429
+ details: {
430
+ pageType: 'service/product',
431
+ recommendation: 'Consider adding a pricing calculator, ROI estimator, or product configurator',
432
+ },
433
+ });
434
+ }
435
+
436
+ // Blog page without video
437
+ if (isBlogPage && !dwellTimeSignals.hasVideoEmbed) {
438
+ issues.push({
439
+ code: 'BLOG_NO_VIDEO',
440
+ severity: 'notice',
441
+ category: 'content',
442
+ title: 'Blog post lacks video content',
443
+ description: 'This blog post has no embedded video content.',
444
+ impact: 'Embedded YouTube videos increase dwell time and can appear in video search results.',
445
+ howToFix: 'Create and embed a YouTube video that summarizes or expands on the blog content.',
446
+ affectedUrls: [url],
447
+ });
448
+ }
449
+
450
+ // Low dwell time signals overall
451
+ if (dwellTimeScore < 20 && uniqueTools.length === 0) {
452
+ issues.push({
453
+ code: 'LOW_DWELL_TIME_SIGNALS',
454
+ severity: 'notice',
455
+ category: 'content',
456
+ title: 'Page lacks dwell time optimization',
457
+ description: 'No interactive tools, calculators, quizzes, or video embeds detected.',
458
+ impact: 'Interactive content significantly increases time on page, a user engagement signal.',
459
+ howToFix: 'Add at least one interactive element: calculator, quiz, embedded video, or free tool.',
460
+ affectedUrls: [url],
461
+ });
462
+ }
463
+
464
+ return {
465
+ issues,
466
+ data: {
467
+ hasInteractiveTools: uniqueTools.length > 0,
468
+ tools: uniqueTools,
469
+ dwellTimeSignals,
470
+ toolCount: uniqueTools.length,
471
+ dwellTimeScore,
472
+ recommendations,
473
+ },
474
+ };
475
+ }