@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,139 @@
|
|
|
1
|
+
// Google Autocomplete Keyword Scraper (Free)
|
|
2
|
+
|
|
3
|
+
import { httpGet } from '../../utils/http.js';
|
|
4
|
+
import type { KeywordData } from '../types.js';
|
|
5
|
+
|
|
6
|
+
const AUTOCOMPLETE_URL = 'https://suggestqueries.google.com/complete/search';
|
|
7
|
+
|
|
8
|
+
export interface AutocompleteOptions {
|
|
9
|
+
language?: string;
|
|
10
|
+
country?: string;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export async function getAutocompleteSuggestions(
|
|
14
|
+
seedKeyword: string,
|
|
15
|
+
options: AutocompleteOptions = {}
|
|
16
|
+
): Promise<KeywordData[]> {
|
|
17
|
+
const { language = 'en', country = 'us' } = options;
|
|
18
|
+
|
|
19
|
+
try {
|
|
20
|
+
const response = await httpGet<string>(AUTOCOMPLETE_URL, {
|
|
21
|
+
params: {
|
|
22
|
+
client: 'firefox', // Returns JSON
|
|
23
|
+
q: seedKeyword,
|
|
24
|
+
hl: language,
|
|
25
|
+
gl: country,
|
|
26
|
+
},
|
|
27
|
+
timeout: 5000,
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
// Response format: [query, [suggestions], ...]
|
|
31
|
+
// Parse JSON if string (fetch returns text by default for this endpoint)
|
|
32
|
+
const data = typeof response.data === 'string' ? JSON.parse(response.data) : response.data;
|
|
33
|
+
const suggestions: string[] = (data as unknown[])[1] as string[] || [];
|
|
34
|
+
|
|
35
|
+
return suggestions.map((suggestion) => ({
|
|
36
|
+
keyword: suggestion,
|
|
37
|
+
searchVolume: 0, // Autocomplete doesn't provide volume
|
|
38
|
+
keywordDifficulty: 0, // Need to estimate or fetch separately
|
|
39
|
+
source: 'autocomplete' as const,
|
|
40
|
+
}));
|
|
41
|
+
} catch (error) {
|
|
42
|
+
console.error('Autocomplete fetch failed:', error);
|
|
43
|
+
return [];
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
// Get expanded suggestions using alphabet soup technique
|
|
48
|
+
export async function getExpandedSuggestions(
|
|
49
|
+
seedKeyword: string,
|
|
50
|
+
options: AutocompleteOptions = {}
|
|
51
|
+
): Promise<KeywordData[]> {
|
|
52
|
+
const allSuggestions: KeywordData[] = [];
|
|
53
|
+
const seen = new Set<string>();
|
|
54
|
+
|
|
55
|
+
// Base query
|
|
56
|
+
const baseSuggestions = await getAutocompleteSuggestions(seedKeyword, options);
|
|
57
|
+
for (const kw of baseSuggestions) {
|
|
58
|
+
if (!seen.has(kw.keyword.toLowerCase())) {
|
|
59
|
+
seen.add(kw.keyword.toLowerCase());
|
|
60
|
+
allSuggestions.push(kw);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
// Alphabet soup: append each letter
|
|
65
|
+
const letters = 'abcdefghijklmnopqrstuvwxyz'.split('');
|
|
66
|
+
|
|
67
|
+
// Limit concurrent requests
|
|
68
|
+
const batchSize = 5;
|
|
69
|
+
for (let i = 0; i < letters.length; i += batchSize) {
|
|
70
|
+
const batch = letters.slice(i, i + batchSize);
|
|
71
|
+
const promises = batch.map((letter) =>
|
|
72
|
+
getAutocompleteSuggestions(`${seedKeyword} ${letter}`, options)
|
|
73
|
+
);
|
|
74
|
+
|
|
75
|
+
const results = await Promise.all(promises);
|
|
76
|
+
for (const suggestions of results) {
|
|
77
|
+
for (const kw of suggestions) {
|
|
78
|
+
if (!seen.has(kw.keyword.toLowerCase())) {
|
|
79
|
+
seen.add(kw.keyword.toLowerCase());
|
|
80
|
+
allSuggestions.push(kw);
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// Small delay to avoid rate limiting
|
|
86
|
+
await new Promise((resolve) => setTimeout(resolve, 100));
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// Question modifiers
|
|
90
|
+
const questionPrefixes = ['how to', 'what is', 'why', 'when', 'where', 'best'];
|
|
91
|
+
for (const prefix of questionPrefixes) {
|
|
92
|
+
const suggestions = await getAutocompleteSuggestions(`${prefix} ${seedKeyword}`, options);
|
|
93
|
+
for (const kw of suggestions) {
|
|
94
|
+
if (!seen.has(kw.keyword.toLowerCase())) {
|
|
95
|
+
seen.add(kw.keyword.toLowerCase());
|
|
96
|
+
allSuggestions.push(kw);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
return allSuggestions;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
// Estimate keyword difficulty based on SERP analysis
|
|
105
|
+
export function estimateKeywordDifficulty(keyword: string): number {
|
|
106
|
+
// Basic heuristics (can be improved with actual SERP data)
|
|
107
|
+
const wordCount = keyword.split(' ').length;
|
|
108
|
+
|
|
109
|
+
// Longer keywords tend to be less competitive
|
|
110
|
+
if (wordCount >= 5) return 5;
|
|
111
|
+
if (wordCount >= 4) return 10;
|
|
112
|
+
if (wordCount >= 3) return 20;
|
|
113
|
+
if (wordCount >= 2) return 30;
|
|
114
|
+
|
|
115
|
+
return 50; // Single word = high competition
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
// Estimate search volume based on keyword characteristics
|
|
119
|
+
export function estimateSearchVolume(keyword: string): number {
|
|
120
|
+
// This is a rough estimate - real data should come from APIs
|
|
121
|
+
const wordCount = keyword.split(' ').length;
|
|
122
|
+
|
|
123
|
+
// Longer tail = lower volume typically
|
|
124
|
+
if (wordCount >= 5) return 50;
|
|
125
|
+
if (wordCount >= 4) return 100;
|
|
126
|
+
if (wordCount >= 3) return 300;
|
|
127
|
+
if (wordCount >= 2) return 500;
|
|
128
|
+
|
|
129
|
+
return 1000; // Single word = high volume
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
// Add estimates to keywords without data
|
|
133
|
+
export function enrichKeywordsWithEstimates(keywords: KeywordData[]): KeywordData[] {
|
|
134
|
+
return keywords.map((kw) => ({
|
|
135
|
+
...kw,
|
|
136
|
+
keywordDifficulty: kw.keywordDifficulty || estimateKeywordDifficulty(kw.keyword),
|
|
137
|
+
searchVolume: kw.searchVolume || estimateSearchVolume(kw.keyword),
|
|
138
|
+
}));
|
|
139
|
+
}
|
|
@@ -0,0 +1,450 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Competitive Search Module
|
|
3
|
+
*
|
|
4
|
+
* Uses free APIs to find existing tools, competitors, and market intelligence.
|
|
5
|
+
* Priority order: Brave Search > Serper.dev > GitHub > npm/PyPI
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
export interface CompetitorTool {
|
|
9
|
+
name: string;
|
|
10
|
+
url: string;
|
|
11
|
+
description: string;
|
|
12
|
+
isPaid: boolean;
|
|
13
|
+
pricing?: string;
|
|
14
|
+
source: 'search' | 'github' | 'npm' | 'pypi' | 'hackernews';
|
|
15
|
+
stars?: number;
|
|
16
|
+
weeklyDownloads?: number;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export interface CompetitiveSearchResult {
|
|
20
|
+
query: string;
|
|
21
|
+
existingTools: CompetitorTool[];
|
|
22
|
+
openSourceOptions: CompetitorTool[];
|
|
23
|
+
marketInsights: {
|
|
24
|
+
hasFreeSolutions: boolean;
|
|
25
|
+
hasPaidSolutions: boolean;
|
|
26
|
+
dominantPlayer?: string;
|
|
27
|
+
marketGap?: string;
|
|
28
|
+
};
|
|
29
|
+
searchSource: string;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
export interface CompetitiveSearchOptions {
|
|
33
|
+
braveApiKey?: string;
|
|
34
|
+
serperApiKey?: string;
|
|
35
|
+
githubToken?: string;
|
|
36
|
+
maxResults?: number;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Search for existing tools and competitors
|
|
41
|
+
*/
|
|
42
|
+
export async function searchCompetitors(
|
|
43
|
+
toolConcept: string,
|
|
44
|
+
options: CompetitiveSearchOptions = {}
|
|
45
|
+
): Promise<CompetitiveSearchResult> {
|
|
46
|
+
const { maxResults = 10 } = options;
|
|
47
|
+
const existingTools: CompetitorTool[] = [];
|
|
48
|
+
const openSourceOptions: CompetitorTool[] = [];
|
|
49
|
+
let searchSource = 'none';
|
|
50
|
+
|
|
51
|
+
// Try Brave Search first (2,000 free/month)
|
|
52
|
+
if (options.braveApiKey) {
|
|
53
|
+
try {
|
|
54
|
+
const braveResults = await searchBrave(toolConcept, options.braveApiKey, maxResults);
|
|
55
|
+
existingTools.push(...braveResults);
|
|
56
|
+
searchSource = 'brave';
|
|
57
|
+
} catch (e) {
|
|
58
|
+
console.warn('Brave search failed:', e);
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
// Fall back to Serper.dev (2,500 free/month)
|
|
63
|
+
if (existingTools.length === 0 && options.serperApiKey) {
|
|
64
|
+
try {
|
|
65
|
+
const serperResults = await searchSerper(toolConcept, options.serperApiKey, maxResults);
|
|
66
|
+
existingTools.push(...serperResults);
|
|
67
|
+
searchSource = 'serper';
|
|
68
|
+
} catch (e) {
|
|
69
|
+
console.warn('Serper search failed:', e);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
// Always search GitHub for open source (unlimited free)
|
|
74
|
+
try {
|
|
75
|
+
const githubResults = await searchGitHub(toolConcept, options.githubToken, maxResults);
|
|
76
|
+
openSourceOptions.push(...githubResults);
|
|
77
|
+
if (!searchSource || searchSource === 'none') searchSource = 'github';
|
|
78
|
+
} catch (e) {
|
|
79
|
+
console.warn('GitHub search failed:', e);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
// Search npm for JavaScript tools (unlimited free)
|
|
83
|
+
try {
|
|
84
|
+
const npmResults = await searchNpm(toolConcept, maxResults);
|
|
85
|
+
openSourceOptions.push(...npmResults);
|
|
86
|
+
} catch (e) {
|
|
87
|
+
console.warn('npm search failed:', e);
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
// Analyze market
|
|
91
|
+
const marketInsights = analyzeMarket(existingTools, openSourceOptions);
|
|
92
|
+
|
|
93
|
+
return {
|
|
94
|
+
query: toolConcept,
|
|
95
|
+
existingTools,
|
|
96
|
+
openSourceOptions,
|
|
97
|
+
marketInsights,
|
|
98
|
+
searchSource,
|
|
99
|
+
};
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
/**
|
|
103
|
+
* Brave Search API (2,000 free searches/month)
|
|
104
|
+
*/
|
|
105
|
+
async function searchBrave(
|
|
106
|
+
query: string,
|
|
107
|
+
apiKey: string,
|
|
108
|
+
maxResults: number
|
|
109
|
+
): Promise<CompetitorTool[]> {
|
|
110
|
+
const searchQuery = `${query} tool OR ${query} free online`;
|
|
111
|
+
|
|
112
|
+
const response = await fetch(
|
|
113
|
+
`https://api.search.brave.com/res/v1/web/search?q=${encodeURIComponent(searchQuery)}&count=${maxResults}`,
|
|
114
|
+
{
|
|
115
|
+
headers: {
|
|
116
|
+
'Accept': 'application/json',
|
|
117
|
+
'X-Subscription-Token': apiKey,
|
|
118
|
+
},
|
|
119
|
+
}
|
|
120
|
+
);
|
|
121
|
+
|
|
122
|
+
if (!response.ok) {
|
|
123
|
+
throw new Error(`Brave API error: ${response.status}`);
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
const data = await response.json();
|
|
127
|
+
const results: CompetitorTool[] = [];
|
|
128
|
+
|
|
129
|
+
for (const result of data.web?.results || []) {
|
|
130
|
+
const isPaid = detectPaidTool(result.title, result.description, result.url);
|
|
131
|
+
|
|
132
|
+
results.push({
|
|
133
|
+
name: extractToolName(result.title),
|
|
134
|
+
url: result.url,
|
|
135
|
+
description: result.description || '',
|
|
136
|
+
isPaid,
|
|
137
|
+
pricing: isPaid ? extractPricing(result.description) : 'Free',
|
|
138
|
+
source: 'search',
|
|
139
|
+
});
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
return results;
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
/**
|
|
146
|
+
* Serper.dev API (2,500 free searches/month)
|
|
147
|
+
*/
|
|
148
|
+
async function searchSerper(
|
|
149
|
+
query: string,
|
|
150
|
+
apiKey: string,
|
|
151
|
+
maxResults: number
|
|
152
|
+
): Promise<CompetitorTool[]> {
|
|
153
|
+
const searchQuery = `${query} tool OR ${query} free online`;
|
|
154
|
+
|
|
155
|
+
const response = await fetch('https://google.serper.dev/search', {
|
|
156
|
+
method: 'POST',
|
|
157
|
+
headers: {
|
|
158
|
+
'X-API-KEY': apiKey,
|
|
159
|
+
'Content-Type': 'application/json',
|
|
160
|
+
},
|
|
161
|
+
body: JSON.stringify({
|
|
162
|
+
q: searchQuery,
|
|
163
|
+
num: maxResults,
|
|
164
|
+
}),
|
|
165
|
+
});
|
|
166
|
+
|
|
167
|
+
if (!response.ok) {
|
|
168
|
+
throw new Error(`Serper API error: ${response.status}`);
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
const data = await response.json();
|
|
172
|
+
const results: CompetitorTool[] = [];
|
|
173
|
+
|
|
174
|
+
for (const result of data.organic || []) {
|
|
175
|
+
const isPaid = detectPaidTool(result.title, result.snippet, result.link);
|
|
176
|
+
|
|
177
|
+
results.push({
|
|
178
|
+
name: extractToolName(result.title),
|
|
179
|
+
url: result.link,
|
|
180
|
+
description: result.snippet || '',
|
|
181
|
+
isPaid,
|
|
182
|
+
pricing: isPaid ? extractPricing(result.snippet) : 'Free',
|
|
183
|
+
source: 'search',
|
|
184
|
+
});
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
return results;
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
/**
|
|
191
|
+
* GitHub Search API (60 req/hour unauthenticated, more with token)
|
|
192
|
+
*/
|
|
193
|
+
async function searchGitHub(
|
|
194
|
+
query: string,
|
|
195
|
+
token?: string,
|
|
196
|
+
maxResults: number = 10
|
|
197
|
+
): Promise<CompetitorTool[]> {
|
|
198
|
+
const searchQuery = `${query} in:name,description,readme`;
|
|
199
|
+
|
|
200
|
+
const headers: Record<string, string> = {
|
|
201
|
+
'Accept': 'application/vnd.github.v3+json',
|
|
202
|
+
'User-Agent': 'RankCLI-Keyword-Research',
|
|
203
|
+
};
|
|
204
|
+
|
|
205
|
+
if (token) {
|
|
206
|
+
headers['Authorization'] = `token ${token}`;
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
const response = await fetch(
|
|
210
|
+
`https://api.github.com/search/repositories?q=${encodeURIComponent(searchQuery)}&sort=stars&order=desc&per_page=${maxResults}`,
|
|
211
|
+
{ headers }
|
|
212
|
+
);
|
|
213
|
+
|
|
214
|
+
if (!response.ok) {
|
|
215
|
+
throw new Error(`GitHub API error: ${response.status}`);
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
const data = await response.json();
|
|
219
|
+
const results: CompetitorTool[] = [];
|
|
220
|
+
|
|
221
|
+
for (const repo of data.items || []) {
|
|
222
|
+
results.push({
|
|
223
|
+
name: repo.name,
|
|
224
|
+
url: repo.html_url,
|
|
225
|
+
description: repo.description || '',
|
|
226
|
+
isPaid: false,
|
|
227
|
+
pricing: 'Open Source',
|
|
228
|
+
source: 'github',
|
|
229
|
+
stars: repo.stargazers_count,
|
|
230
|
+
});
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
return results;
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
/**
|
|
237
|
+
* npm Registry Search (unlimited, no auth required)
|
|
238
|
+
*/
|
|
239
|
+
async function searchNpm(
|
|
240
|
+
query: string,
|
|
241
|
+
maxResults: number = 10
|
|
242
|
+
): Promise<CompetitorTool[]> {
|
|
243
|
+
const response = await fetch(
|
|
244
|
+
`https://registry.npmjs.org/-/v1/search?text=${encodeURIComponent(query)}&size=${maxResults}`,
|
|
245
|
+
{
|
|
246
|
+
headers: {
|
|
247
|
+
'User-Agent': 'RankCLI-Keyword-Research',
|
|
248
|
+
},
|
|
249
|
+
}
|
|
250
|
+
);
|
|
251
|
+
|
|
252
|
+
if (!response.ok) {
|
|
253
|
+
throw new Error(`npm API error: ${response.status}`);
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
const data = await response.json();
|
|
257
|
+
const results: CompetitorTool[] = [];
|
|
258
|
+
|
|
259
|
+
for (const obj of data.objects || []) {
|
|
260
|
+
const pkg = obj.package;
|
|
261
|
+
results.push({
|
|
262
|
+
name: pkg.name,
|
|
263
|
+
url: `https://www.npmjs.com/package/${pkg.name}`,
|
|
264
|
+
description: pkg.description || '',
|
|
265
|
+
isPaid: false,
|
|
266
|
+
pricing: 'Open Source (npm)',
|
|
267
|
+
source: 'npm',
|
|
268
|
+
weeklyDownloads: obj.downloads?.weekly,
|
|
269
|
+
});
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
return results;
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
/**
|
|
276
|
+
* Detect if a tool is paid based on text signals
|
|
277
|
+
*/
|
|
278
|
+
function detectPaidTool(title: string, description: string, url: string): boolean {
|
|
279
|
+
const text = `${title} ${description} ${url}`.toLowerCase();
|
|
280
|
+
|
|
281
|
+
// Strong paid signals
|
|
282
|
+
const paidSignals = [
|
|
283
|
+
'pricing', 'subscribe', 'premium', 'pro plan', 'enterprise',
|
|
284
|
+
'per month', '/mo', 'free trial', 'buy now', 'purchase',
|
|
285
|
+
'starting at', 'plans start', 'paid', 'upgrade',
|
|
286
|
+
];
|
|
287
|
+
|
|
288
|
+
// Strong free signals
|
|
289
|
+
const freeSignals = [
|
|
290
|
+
'free online', '100% free', 'completely free', 'no signup',
|
|
291
|
+
'open source', 'github.com', 'github.io', 'codepen',
|
|
292
|
+
];
|
|
293
|
+
|
|
294
|
+
const hasPaidSignal = paidSignals.some(signal => text.includes(signal));
|
|
295
|
+
const hasFreeSignal = freeSignals.some(signal => text.includes(signal));
|
|
296
|
+
|
|
297
|
+
// If has free signal and no paid signal, it's free
|
|
298
|
+
if (hasFreeSignal && !hasPaidSignal) return false;
|
|
299
|
+
|
|
300
|
+
// If has paid signal, it's paid
|
|
301
|
+
if (hasPaidSignal) return true;
|
|
302
|
+
|
|
303
|
+
// Default: assume it might be paid
|
|
304
|
+
return false;
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
/**
|
|
308
|
+
* Extract tool name from search result title
|
|
309
|
+
*/
|
|
310
|
+
function extractToolName(title: string): string {
|
|
311
|
+
// Remove common suffixes like "| Website", "- Free Online Tool", etc.
|
|
312
|
+
return title
|
|
313
|
+
.split(/[|\-–—]/)[0]
|
|
314
|
+
.replace(/free|online|tool|generator|maker/gi, '')
|
|
315
|
+
.trim() || title.trim();
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
/**
|
|
319
|
+
* Extract pricing info from description
|
|
320
|
+
*/
|
|
321
|
+
function extractPricing(text: string): string | undefined {
|
|
322
|
+
const priceMatch = text.match(/\$[\d,.]+(?:\s*\/\s*(?:mo|month|year|yr))?/i);
|
|
323
|
+
if (priceMatch) return priceMatch[0];
|
|
324
|
+
|
|
325
|
+
if (text.toLowerCase().includes('free trial')) return 'Free trial available';
|
|
326
|
+
if (text.toLowerCase().includes('freemium')) return 'Freemium';
|
|
327
|
+
|
|
328
|
+
return undefined;
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
/**
|
|
332
|
+
* Analyze market based on search results
|
|
333
|
+
*/
|
|
334
|
+
function analyzeMarket(
|
|
335
|
+
existingTools: CompetitorTool[],
|
|
336
|
+
openSourceOptions: CompetitorTool[]
|
|
337
|
+
): CompetitiveSearchResult['marketInsights'] {
|
|
338
|
+
const hasFreeSolutions = existingTools.some(t => !t.isPaid) || openSourceOptions.length > 0;
|
|
339
|
+
const hasPaidSolutions = existingTools.some(t => t.isPaid);
|
|
340
|
+
|
|
341
|
+
// Find dominant player (most stars on GitHub or appears in multiple results)
|
|
342
|
+
let dominantPlayer: string | undefined;
|
|
343
|
+
if (openSourceOptions.length > 0) {
|
|
344
|
+
const topByStars = openSourceOptions
|
|
345
|
+
.filter(t => t.stars)
|
|
346
|
+
.sort((a, b) => (b.stars || 0) - (a.stars || 0))[0];
|
|
347
|
+
if (topByStars && (topByStars.stars || 0) > 1000) {
|
|
348
|
+
dominantPlayer = topByStars.name;
|
|
349
|
+
}
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
// Identify market gap
|
|
353
|
+
let marketGap: string | undefined;
|
|
354
|
+
if (!hasFreeSolutions && hasPaidSolutions) {
|
|
355
|
+
marketGap = 'No free alternatives exist - opportunity to capture market with free tool';
|
|
356
|
+
} else if (openSourceOptions.length > 0 && !existingTools.some(t => !t.isPaid)) {
|
|
357
|
+
marketGap = 'Open source libraries exist but no easy-to-use web tool - wrap library in simple UI';
|
|
358
|
+
}
|
|
359
|
+
|
|
360
|
+
return {
|
|
361
|
+
hasFreeSolutions,
|
|
362
|
+
hasPaidSolutions,
|
|
363
|
+
dominantPlayer,
|
|
364
|
+
marketGap,
|
|
365
|
+
};
|
|
366
|
+
}
|
|
367
|
+
|
|
368
|
+
/**
|
|
369
|
+
* Search for specific format converters (e.g., "X to Mermaid converter")
|
|
370
|
+
*/
|
|
371
|
+
export async function searchFormatConverters(
|
|
372
|
+
targetFormat: string,
|
|
373
|
+
options: CompetitiveSearchOptions = {}
|
|
374
|
+
): Promise<{
|
|
375
|
+
sourceFormats: string[];
|
|
376
|
+
existingConverters: CompetitorTool[];
|
|
377
|
+
opportunities: string[];
|
|
378
|
+
}> {
|
|
379
|
+
// Common diagram/document formats to check
|
|
380
|
+
const commonFormats = [
|
|
381
|
+
'flowchart', 'uml', 'plantuml', 'draw.io', 'lucidchart',
|
|
382
|
+
'visio', 'graphviz', 'dot', 'json', 'yaml', 'csv',
|
|
383
|
+
'excel', 'markdown', 'text', 'ascii', 'png', 'svg',
|
|
384
|
+
];
|
|
385
|
+
|
|
386
|
+
const existingConverters: CompetitorTool[] = [];
|
|
387
|
+
const opportunities: string[] = [];
|
|
388
|
+
|
|
389
|
+
for (const format of commonFormats) {
|
|
390
|
+
const query = `${format} to ${targetFormat} converter`;
|
|
391
|
+
|
|
392
|
+
try {
|
|
393
|
+
const result = await searchCompetitors(query, options);
|
|
394
|
+
|
|
395
|
+
if (result.existingTools.length > 0 || result.openSourceOptions.length > 0) {
|
|
396
|
+
existingConverters.push(...result.existingTools.slice(0, 2));
|
|
397
|
+
existingConverters.push(...result.openSourceOptions.slice(0, 2));
|
|
398
|
+
} else {
|
|
399
|
+
// No converter exists - opportunity!
|
|
400
|
+
opportunities.push(`${format} to ${targetFormat}`);
|
|
401
|
+
}
|
|
402
|
+
} catch {
|
|
403
|
+
// Skip on error
|
|
404
|
+
}
|
|
405
|
+
|
|
406
|
+
// Rate limit: wait 100ms between searches
|
|
407
|
+
await new Promise(resolve => setTimeout(resolve, 100));
|
|
408
|
+
}
|
|
409
|
+
|
|
410
|
+
return {
|
|
411
|
+
sourceFormats: commonFormats,
|
|
412
|
+
existingConverters,
|
|
413
|
+
opportunities,
|
|
414
|
+
};
|
|
415
|
+
}
|
|
416
|
+
|
|
417
|
+
/**
|
|
418
|
+
* Search Hacker News for tool discussions
|
|
419
|
+
*/
|
|
420
|
+
export async function searchHackerNews(
|
|
421
|
+
query: string,
|
|
422
|
+
maxResults: number = 10
|
|
423
|
+
): Promise<Array<{
|
|
424
|
+
title: string;
|
|
425
|
+
url: string;
|
|
426
|
+
points: number;
|
|
427
|
+
comments: number;
|
|
428
|
+
}>> {
|
|
429
|
+
const response = await fetch(
|
|
430
|
+
`https://hn.algolia.com/api/v1/search?query=${encodeURIComponent(query)}&tags=story&hitsPerPage=${maxResults}`,
|
|
431
|
+
{
|
|
432
|
+
headers: {
|
|
433
|
+
'User-Agent': 'RankCLI-Keyword-Research',
|
|
434
|
+
},
|
|
435
|
+
}
|
|
436
|
+
);
|
|
437
|
+
|
|
438
|
+
if (!response.ok) {
|
|
439
|
+
throw new Error(`HN API error: ${response.status}`);
|
|
440
|
+
}
|
|
441
|
+
|
|
442
|
+
const data = await response.json();
|
|
443
|
+
|
|
444
|
+
return (data.hits || []).map((hit: any) => ({
|
|
445
|
+
title: hit.title,
|
|
446
|
+
url: hit.url || `https://news.ycombinator.com/item?id=${hit.objectID}`,
|
|
447
|
+
points: hit.points || 0,
|
|
448
|
+
comments: hit.num_comments || 0,
|
|
449
|
+
}));
|
|
450
|
+
}
|