@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,275 @@
|
|
|
1
|
+
// IndexNow Protocol Implementation Check
|
|
2
|
+
// IndexNow enables instant indexing by Bing, Yandex, Seznam, Naver, and other participating search engines
|
|
3
|
+
|
|
4
|
+
import { httpGet } from '../../utils/http.js';
|
|
5
|
+
import * as cheerio from 'cheerio';
|
|
6
|
+
import type { AuditIssue } from '../types.js';
|
|
7
|
+
|
|
8
|
+
export interface IndexNowData {
|
|
9
|
+
hasKey: boolean;
|
|
10
|
+
keyFileUrl: string | null;
|
|
11
|
+
keyValue: string | null;
|
|
12
|
+
keyValid: boolean;
|
|
13
|
+
hasLinkTag: boolean;
|
|
14
|
+
inRobotsTxt: boolean;
|
|
15
|
+
cmsPlugin: string | null;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
// Common IndexNow key file locations
|
|
19
|
+
const KEY_FILE_PATTERNS = [
|
|
20
|
+
// Standard pattern: /{key}.txt
|
|
21
|
+
/^[a-f0-9]{32}\.txt$/i,
|
|
22
|
+
/^[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}\.txt$/i,
|
|
23
|
+
];
|
|
24
|
+
|
|
25
|
+
// CMS plugins that implement IndexNow
|
|
26
|
+
const CMS_PLUGINS: Record<string, string[]> = {
|
|
27
|
+
WordPress: ['IndexNow', 'Yoast SEO Premium', 'Rank Math SEO', 'All in One SEO'],
|
|
28
|
+
Wix: ['IndexNow built-in'],
|
|
29
|
+
Cloudflare: ['Cloudflare IndexNow'],
|
|
30
|
+
};
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Check for IndexNow key in robots.txt
|
|
34
|
+
*/
|
|
35
|
+
export async function checkRobotsTxtForIndexNow(
|
|
36
|
+
baseUrl: string
|
|
37
|
+
): Promise<{ found: boolean; key: string | null }> {
|
|
38
|
+
try {
|
|
39
|
+
const response = await httpGet<string>(`${new URL(baseUrl).origin}/robots.txt`, {
|
|
40
|
+
|
|
41
|
+
timeout: 10000,
|
|
42
|
+
validateStatus: () => true,
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
if (response.status !== 200) {
|
|
46
|
+
return { found: false, key: null };
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
const content = response.data as string;
|
|
50
|
+
|
|
51
|
+
// Look for IndexNow comment or directive
|
|
52
|
+
const indexNowPattern = /indexnow[:\s]+([a-f0-9-]+)/i;
|
|
53
|
+
const match = content.match(indexNowPattern);
|
|
54
|
+
|
|
55
|
+
if (match) {
|
|
56
|
+
return { found: true, key: match[1] };
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
return { found: false, key: null };
|
|
60
|
+
} catch {
|
|
61
|
+
return { found: false, key: null };
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
/**
|
|
66
|
+
* Validate IndexNow key format
|
|
67
|
+
*/
|
|
68
|
+
export function isValidIndexNowKey(key: string): boolean {
|
|
69
|
+
// Key should be 8-128 characters, hexadecimal or UUID format
|
|
70
|
+
if (key.length < 8 || key.length > 128) return false;
|
|
71
|
+
|
|
72
|
+
// UUID format
|
|
73
|
+
if (/^[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}$/i.test(key)) {
|
|
74
|
+
return true;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
// Hex format (32 chars)
|
|
78
|
+
if (/^[a-f0-9]{32}$/i.test(key)) {
|
|
79
|
+
return true;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
// Other alphanumeric formats
|
|
83
|
+
if (/^[a-zA-Z0-9-]{8,128}$/.test(key)) {
|
|
84
|
+
return true;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
return false;
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
/**
|
|
91
|
+
* Try to find and validate IndexNow key file
|
|
92
|
+
*/
|
|
93
|
+
export async function findIndexNowKeyFile(
|
|
94
|
+
baseUrl: string,
|
|
95
|
+
possibleKeys: string[]
|
|
96
|
+
): Promise<{ found: boolean; url: string | null; key: string | null }> {
|
|
97
|
+
const origin = new URL(baseUrl).origin;
|
|
98
|
+
|
|
99
|
+
for (const key of possibleKeys) {
|
|
100
|
+
if (!isValidIndexNowKey(key)) continue;
|
|
101
|
+
|
|
102
|
+
const keyUrl = `${origin}/${key}.txt`;
|
|
103
|
+
|
|
104
|
+
try {
|
|
105
|
+
const response = await httpGet<string>(keyUrl, {
|
|
106
|
+
|
|
107
|
+
timeout: 5000,
|
|
108
|
+
validateStatus: () => true,
|
|
109
|
+
});
|
|
110
|
+
|
|
111
|
+
if (response.status === 200) {
|
|
112
|
+
const content = (response.data as string).trim();
|
|
113
|
+
if (content === key) {
|
|
114
|
+
return { found: true, url: keyUrl, key };
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
} catch {
|
|
118
|
+
// Key file not found
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
return { found: false, url: null, key: null };
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
/**
|
|
126
|
+
* Check for IndexNow link tag in HTML
|
|
127
|
+
*/
|
|
128
|
+
export function checkIndexNowLinkTag(html: string): { found: boolean; href: string | null } {
|
|
129
|
+
const $ = cheerio.load(html);
|
|
130
|
+
|
|
131
|
+
// Check for indexnow link tag (non-standard but sometimes used)
|
|
132
|
+
const indexNowLink =
|
|
133
|
+
$('link[rel="indexnow"]').attr('href') ||
|
|
134
|
+
$('link[rel="index-now"]').attr('href') ||
|
|
135
|
+
$('meta[name="indexnow-key"]').attr('content') ||
|
|
136
|
+
null;
|
|
137
|
+
|
|
138
|
+
return { found: !!indexNowLink, href: indexNowLink };
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
/**
|
|
142
|
+
* Detect CMS/framework and potential IndexNow plugin
|
|
143
|
+
*/
|
|
144
|
+
export function detectIndexNowCMSSupport(html: string, headers: Record<string, string>): string | null {
|
|
145
|
+
const $ = cheerio.load(html);
|
|
146
|
+
|
|
147
|
+
// Check for WordPress
|
|
148
|
+
const isWordPress =
|
|
149
|
+
html.includes('wp-content') ||
|
|
150
|
+
html.includes('wp-includes') ||
|
|
151
|
+
$('meta[name="generator"][content*="WordPress"]').length > 0;
|
|
152
|
+
|
|
153
|
+
if (isWordPress) {
|
|
154
|
+
// Check for known IndexNow plugins
|
|
155
|
+
if (html.includes('yoast-seo') || html.includes('wordpress-seo')) {
|
|
156
|
+
return 'WordPress + Yoast SEO (may have IndexNow)';
|
|
157
|
+
}
|
|
158
|
+
if (html.includes('rank-math') || html.includes('rankmath')) {
|
|
159
|
+
return 'WordPress + Rank Math (has IndexNow)';
|
|
160
|
+
}
|
|
161
|
+
return 'WordPress (IndexNow plugins available)';
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
// Check for Wix
|
|
165
|
+
if (html.includes('wix.com') || html.includes('_wix_browser_sess')) {
|
|
166
|
+
return 'Wix (IndexNow built-in)';
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
// Check for Cloudflare
|
|
170
|
+
if (headers['cf-ray'] || headers['server']?.includes('cloudflare')) {
|
|
171
|
+
return 'Cloudflare (IndexNow integration available)';
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
return null;
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
/**
|
|
178
|
+
* Main function: Analyze IndexNow implementation
|
|
179
|
+
*/
|
|
180
|
+
export async function analyzeIndexNow(
|
|
181
|
+
html: string,
|
|
182
|
+
url: string,
|
|
183
|
+
headers: Record<string, string> = {}
|
|
184
|
+
): Promise<{ issues: AuditIssue[]; data: IndexNowData }> {
|
|
185
|
+
const issues: AuditIssue[] = [];
|
|
186
|
+
const origin = new URL(url).origin;
|
|
187
|
+
|
|
188
|
+
// Check for link tag
|
|
189
|
+
const linkTag = checkIndexNowLinkTag(html);
|
|
190
|
+
|
|
191
|
+
// Check robots.txt
|
|
192
|
+
const robotsCheck = await checkRobotsTxtForIndexNow(url);
|
|
193
|
+
|
|
194
|
+
// Try to find key file with any discovered keys
|
|
195
|
+
const possibleKeys: string[] = [];
|
|
196
|
+
if (linkTag.href) possibleKeys.push(linkTag.href);
|
|
197
|
+
if (robotsCheck.key) possibleKeys.push(robotsCheck.key);
|
|
198
|
+
|
|
199
|
+
// Also try common key patterns
|
|
200
|
+
// Note: In a real implementation, you might crawl the root directory
|
|
201
|
+
// For now, we'll just check if the key file exists with discovered keys
|
|
202
|
+
|
|
203
|
+
const keyFile = await findIndexNowKeyFile(url, possibleKeys);
|
|
204
|
+
|
|
205
|
+
// Detect CMS support
|
|
206
|
+
const cmsPlugin = detectIndexNowCMSSupport(html, headers);
|
|
207
|
+
|
|
208
|
+
const data: IndexNowData = {
|
|
209
|
+
hasKey: keyFile.found,
|
|
210
|
+
keyFileUrl: keyFile.url,
|
|
211
|
+
keyValue: keyFile.key,
|
|
212
|
+
keyValid: keyFile.key ? isValidIndexNowKey(keyFile.key) : false,
|
|
213
|
+
hasLinkTag: linkTag.found,
|
|
214
|
+
inRobotsTxt: robotsCheck.found,
|
|
215
|
+
cmsPlugin,
|
|
216
|
+
};
|
|
217
|
+
|
|
218
|
+
// Generate issues
|
|
219
|
+
if (!keyFile.found && !linkTag.found && !robotsCheck.found) {
|
|
220
|
+
const suggestion = cmsPlugin
|
|
221
|
+
? `Your platform (${cmsPlugin}) may support IndexNow. Check for built-in support or plugins.`
|
|
222
|
+
: 'Create a key at indexnow.org, add the key file to your root directory, and submit URLs via the IndexNow API.';
|
|
223
|
+
|
|
224
|
+
issues.push({
|
|
225
|
+
code: 'INDEXNOW_NOT_IMPLEMENTED',
|
|
226
|
+
severity: 'notice',
|
|
227
|
+
category: 'crawlability',
|
|
228
|
+
title: 'IndexNow not implemented',
|
|
229
|
+
description: 'Site does not appear to have IndexNow instant indexing set up.',
|
|
230
|
+
impact: 'Content changes may take longer to appear in Bing, Yandex, and other IndexNow-supporting search engines.',
|
|
231
|
+
howToFix: suggestion,
|
|
232
|
+
affectedUrls: [url],
|
|
233
|
+
details: { cmsPlugin },
|
|
234
|
+
});
|
|
235
|
+
} else if (keyFile.found && !data.keyValid) {
|
|
236
|
+
issues.push({
|
|
237
|
+
code: 'INDEXNOW_INVALID_KEY',
|
|
238
|
+
severity: 'warning',
|
|
239
|
+
category: 'crawlability',
|
|
240
|
+
title: 'IndexNow key format invalid',
|
|
241
|
+
description: `The IndexNow key "${keyFile.key}" does not match expected formats.`,
|
|
242
|
+
impact: 'IndexNow submissions may fail with invalid key format.',
|
|
243
|
+
howToFix: 'Use a valid key format: 32-character hex or UUID format.',
|
|
244
|
+
affectedUrls: [keyFile.url || url],
|
|
245
|
+
});
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
// Informational notice if IndexNow is implemented
|
|
249
|
+
if (keyFile.found && data.keyValid) {
|
|
250
|
+
// This is good! No issue needed, but we track the data
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
return { issues, data };
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
/**
|
|
257
|
+
* Generate IndexNow submission URL for a page
|
|
258
|
+
*/
|
|
259
|
+
export function generateIndexNowSubmitUrl(
|
|
260
|
+
pageUrl: string,
|
|
261
|
+
key: string,
|
|
262
|
+
searchEngine: 'bing' | 'yandex' | 'seznam' | 'naver' = 'bing'
|
|
263
|
+
): string {
|
|
264
|
+
const endpoints: Record<string, string> = {
|
|
265
|
+
bing: 'https://www.bing.com/indexnow',
|
|
266
|
+
yandex: 'https://yandex.com/indexnow',
|
|
267
|
+
seznam: 'https://search.seznam.cz/indexnow',
|
|
268
|
+
naver: 'https://searchadvisor.naver.com/indexnow',
|
|
269
|
+
};
|
|
270
|
+
|
|
271
|
+
const endpoint = endpoints[searchEngine];
|
|
272
|
+
const parsedUrl = new URL(pageUrl);
|
|
273
|
+
|
|
274
|
+
return `${endpoint}?url=${encodeURIComponent(pageUrl)}&key=${key}&keyLocation=${encodeURIComponent(`${parsedUrl.origin}/${key}.txt`)}`;
|
|
275
|
+
}
|