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