@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,387 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AI-Powered Site Summarizer
|
|
3
|
+
*
|
|
4
|
+
* Uses OpenAI to understand what a site does from crawled content.
|
|
5
|
+
* Extracts: product, audience, features, industry, positioning.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import OpenAI from 'openai';
|
|
9
|
+
import type { SiteCrawlResult } from './site-crawler.js';
|
|
10
|
+
|
|
11
|
+
export interface SiteSummary {
|
|
12
|
+
/** Primary product/service name */
|
|
13
|
+
productName: string;
|
|
14
|
+
/** What the product does (1-2 sentences) */
|
|
15
|
+
productDescription: string;
|
|
16
|
+
/** Target audience description */
|
|
17
|
+
targetAudience: string;
|
|
18
|
+
/** Key features/benefits (3-7 items) */
|
|
19
|
+
keyFeatures: string[];
|
|
20
|
+
/** Industry/category */
|
|
21
|
+
industry: string;
|
|
22
|
+
/** Business model */
|
|
23
|
+
businessModel: 'saas' | 'ecommerce' | 'marketplace' | 'agency' | 'media' | 'other';
|
|
24
|
+
/** Unique value proposition */
|
|
25
|
+
valueProposition: string;
|
|
26
|
+
/** Niche focus (if specific segment) */
|
|
27
|
+
nicheFocus?: string;
|
|
28
|
+
/** Pricing tier detected */
|
|
29
|
+
pricingTier?: 'free' | 'freemium' | 'paid' | 'enterprise' | 'unknown';
|
|
30
|
+
/** Geographic focus */
|
|
31
|
+
geographicFocus?: string;
|
|
32
|
+
/** Primary use cases */
|
|
33
|
+
useCases: string[];
|
|
34
|
+
/** Problems solved */
|
|
35
|
+
problemsSolved: string[];
|
|
36
|
+
/** Seed keyword suggestions */
|
|
37
|
+
suggestedSeedKeywords: string[];
|
|
38
|
+
/** Confidence score (0-1) */
|
|
39
|
+
confidence: number;
|
|
40
|
+
/** What we're uncertain about */
|
|
41
|
+
uncertainties: string[];
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
export interface SummarizerOptions {
|
|
45
|
+
openaiApiKey: string;
|
|
46
|
+
model?: string;
|
|
47
|
+
maxTokens?: number;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
const SUMMARIZER_PROMPT = `You are an expert SEO analyst. Analyze the following website content and extract key information for keyword research.
|
|
51
|
+
|
|
52
|
+
WEBSITE CONTENT:
|
|
53
|
+
{content}
|
|
54
|
+
|
|
55
|
+
HEADINGS FOUND:
|
|
56
|
+
{headings}
|
|
57
|
+
|
|
58
|
+
PAGE TYPES DETECTED:
|
|
59
|
+
- Product pages: {productPages}
|
|
60
|
+
- Pricing pages: {pricingPages}
|
|
61
|
+
- Blog posts: {blogPages}
|
|
62
|
+
- Feature pages: {featurePages}
|
|
63
|
+
|
|
64
|
+
Analyze this content and respond with a JSON object containing:
|
|
65
|
+
|
|
66
|
+
{
|
|
67
|
+
"productName": "The name of the product/service",
|
|
68
|
+
"productDescription": "What it does in 1-2 sentences",
|
|
69
|
+
"targetAudience": "Who this is for (be specific)",
|
|
70
|
+
"keyFeatures": ["feature1", "feature2", "..."],
|
|
71
|
+
"industry": "The industry/category (e.g., 'Marketing Automation', 'Project Management')",
|
|
72
|
+
"businessModel": "saas|ecommerce|marketplace|agency|media|other",
|
|
73
|
+
"valueProposition": "The unique value proposition",
|
|
74
|
+
"nicheFocus": "Specific niche if focused (e.g., 'small businesses', 'developers'), or null if broad",
|
|
75
|
+
"pricingTier": "free|freemium|paid|enterprise|unknown",
|
|
76
|
+
"geographicFocus": "Geographic focus if any, or null",
|
|
77
|
+
"useCases": ["use case 1", "use case 2", "..."],
|
|
78
|
+
"problemsSolved": ["problem 1", "problem 2", "..."],
|
|
79
|
+
"suggestedSeedKeywords": ["keyword1", "keyword2", "..."],
|
|
80
|
+
"confidence": 0.0 to 1.0,
|
|
81
|
+
"uncertainties": ["What we couldn't determine from the content"]
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
IMPORTANT:
|
|
85
|
+
- suggestedSeedKeywords should include: product category keywords, feature keywords, problem keywords, and audience keywords
|
|
86
|
+
- Be specific about the niche - don't say "businesses" if you can say "small e-commerce stores"
|
|
87
|
+
- confidence should be low if the content is thin or unclear
|
|
88
|
+
- List specific uncertainties that would need user input to resolve
|
|
89
|
+
|
|
90
|
+
Respond ONLY with the JSON object, no other text.`;
|
|
91
|
+
|
|
92
|
+
/**
|
|
93
|
+
* Summarize a site using AI
|
|
94
|
+
*/
|
|
95
|
+
export async function summarizeSite(
|
|
96
|
+
crawlResult: SiteCrawlResult,
|
|
97
|
+
options: SummarizerOptions
|
|
98
|
+
): Promise<SiteSummary> {
|
|
99
|
+
const openai = new OpenAI({ apiKey: options.openaiApiKey });
|
|
100
|
+
const model = options.model || 'gpt-4o-mini';
|
|
101
|
+
|
|
102
|
+
// Prepare content (limit for token constraints)
|
|
103
|
+
let content = crawlResult.aggregatedContent;
|
|
104
|
+
if (content.length > 15000) {
|
|
105
|
+
content = content.substring(0, 15000) + '\n...[truncated]';
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
// Prepare headings
|
|
109
|
+
const headings = crawlResult.uniqueHeadings.slice(0, 30).join('\n');
|
|
110
|
+
|
|
111
|
+
// Build prompt
|
|
112
|
+
const prompt = SUMMARIZER_PROMPT
|
|
113
|
+
.replace('{content}', content)
|
|
114
|
+
.replace('{headings}', headings)
|
|
115
|
+
.replace('{productPages}', String(crawlResult.detectedPageTypes.product))
|
|
116
|
+
.replace('{pricingPages}', String(crawlResult.detectedPageTypes.pricing))
|
|
117
|
+
.replace('{blogPages}', String(crawlResult.detectedPageTypes.blog))
|
|
118
|
+
.replace('{featurePages}', String(crawlResult.detectedPageTypes.feature));
|
|
119
|
+
|
|
120
|
+
console.log('🤖 Analyzing site content with AI...');
|
|
121
|
+
|
|
122
|
+
try {
|
|
123
|
+
const response = await openai.chat.completions.create({
|
|
124
|
+
model,
|
|
125
|
+
messages: [
|
|
126
|
+
{
|
|
127
|
+
role: 'system',
|
|
128
|
+
content: 'You are an expert SEO analyst. Always respond with valid JSON only.',
|
|
129
|
+
},
|
|
130
|
+
{
|
|
131
|
+
role: 'user',
|
|
132
|
+
content: prompt,
|
|
133
|
+
},
|
|
134
|
+
],
|
|
135
|
+
temperature: 0.3,
|
|
136
|
+
max_tokens: options.maxTokens || 2000,
|
|
137
|
+
});
|
|
138
|
+
|
|
139
|
+
const responseText = response.choices[0]?.message?.content || '';
|
|
140
|
+
|
|
141
|
+
// Parse JSON response
|
|
142
|
+
const jsonMatch = responseText.match(/\{[\s\S]*\}/);
|
|
143
|
+
if (!jsonMatch) {
|
|
144
|
+
throw new Error('No JSON found in response');
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
const summary = JSON.parse(jsonMatch[0]) as SiteSummary;
|
|
148
|
+
|
|
149
|
+
// Validate and set defaults
|
|
150
|
+
return {
|
|
151
|
+
productName: summary.productName || 'Unknown Product',
|
|
152
|
+
productDescription: summary.productDescription || 'No description available',
|
|
153
|
+
targetAudience: summary.targetAudience || 'General audience',
|
|
154
|
+
keyFeatures: summary.keyFeatures || [],
|
|
155
|
+
industry: summary.industry || 'Unknown',
|
|
156
|
+
businessModel: summary.businessModel || 'other',
|
|
157
|
+
valueProposition: summary.valueProposition || '',
|
|
158
|
+
nicheFocus: summary.nicheFocus || undefined,
|
|
159
|
+
pricingTier: summary.pricingTier || 'unknown',
|
|
160
|
+
geographicFocus: summary.geographicFocus || undefined,
|
|
161
|
+
useCases: summary.useCases || [],
|
|
162
|
+
problemsSolved: summary.problemsSolved || [],
|
|
163
|
+
suggestedSeedKeywords: summary.suggestedSeedKeywords || [],
|
|
164
|
+
confidence: Math.min(1, Math.max(0, summary.confidence || 0.5)),
|
|
165
|
+
uncertainties: summary.uncertainties || [],
|
|
166
|
+
};
|
|
167
|
+
} catch (error) {
|
|
168
|
+
console.error('AI summarization failed:', error);
|
|
169
|
+
|
|
170
|
+
// Return fallback summary
|
|
171
|
+
return createFallbackSummary(crawlResult);
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
/**
|
|
176
|
+
* Create a basic summary without AI (fallback)
|
|
177
|
+
*/
|
|
178
|
+
export function createFallbackSummary(crawlResult: SiteCrawlResult): SiteSummary {
|
|
179
|
+
// Extract from page titles
|
|
180
|
+
const titles = crawlResult.pages.map((p) => p.title).filter(Boolean);
|
|
181
|
+
const productName = titles[0]?.split(/[|\-–—:]/)[0]?.trim() || crawlResult.domain;
|
|
182
|
+
|
|
183
|
+
// Extract key phrases from headings
|
|
184
|
+
const keyFeatures = crawlResult.uniqueHeadings
|
|
185
|
+
.filter((h) => h.length > 5 && h.length < 60)
|
|
186
|
+
.slice(0, 7);
|
|
187
|
+
|
|
188
|
+
// Extract potential seed keywords from titles and headings
|
|
189
|
+
const allText = [...titles, ...crawlResult.uniqueHeadings].join(' ').toLowerCase();
|
|
190
|
+
const words = allText.split(/\s+/).filter((w) => w.length > 4);
|
|
191
|
+
const wordFreq = new Map<string, number>();
|
|
192
|
+
for (const word of words) {
|
|
193
|
+
wordFreq.set(word, (wordFreq.get(word) || 0) + 1);
|
|
194
|
+
}
|
|
195
|
+
const suggestedSeedKeywords = Array.from(wordFreq.entries())
|
|
196
|
+
.sort((a, b) => b[1] - a[1])
|
|
197
|
+
.slice(0, 10)
|
|
198
|
+
.map(([word]) => word);
|
|
199
|
+
|
|
200
|
+
// Detect business model from page types
|
|
201
|
+
let businessModel: SiteSummary['businessModel'] = 'other';
|
|
202
|
+
if (crawlResult.detectedPageTypes.pricing > 0) {
|
|
203
|
+
businessModel = 'saas';
|
|
204
|
+
} else if (crawlResult.detectedPageTypes.product > 2) {
|
|
205
|
+
businessModel = 'ecommerce';
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
return {
|
|
209
|
+
productName,
|
|
210
|
+
productDescription: `Website at ${crawlResult.domain}`,
|
|
211
|
+
targetAudience: 'Unknown - requires AI analysis or user input',
|
|
212
|
+
keyFeatures,
|
|
213
|
+
industry: 'Unknown',
|
|
214
|
+
businessModel,
|
|
215
|
+
valueProposition: '',
|
|
216
|
+
useCases: [],
|
|
217
|
+
problemsSolved: [],
|
|
218
|
+
suggestedSeedKeywords,
|
|
219
|
+
confidence: 0.2, // Low confidence for fallback
|
|
220
|
+
uncertainties: [
|
|
221
|
+
'Target audience could not be determined',
|
|
222
|
+
'Industry classification is uncertain',
|
|
223
|
+
'Value proposition is unclear',
|
|
224
|
+
'Use cases need user input',
|
|
225
|
+
],
|
|
226
|
+
};
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
/**
|
|
230
|
+
* Enhance summary with competitive context
|
|
231
|
+
*/
|
|
232
|
+
export async function enhanceSummaryWithCompetitors(
|
|
233
|
+
summary: SiteSummary,
|
|
234
|
+
competitors: string[],
|
|
235
|
+
options: SummarizerOptions
|
|
236
|
+
): Promise<SiteSummary & { competitivePositioning: string }> {
|
|
237
|
+
const openai = new OpenAI({ apiKey: options.openaiApiKey });
|
|
238
|
+
|
|
239
|
+
const prompt = `Given this product summary:
|
|
240
|
+
Product: ${summary.productName}
|
|
241
|
+
Description: ${summary.productDescription}
|
|
242
|
+
Industry: ${summary.industry}
|
|
243
|
+
Key Features: ${summary.keyFeatures.join(', ')}
|
|
244
|
+
|
|
245
|
+
And these competitors: ${competitors.join(', ')}
|
|
246
|
+
|
|
247
|
+
Analyze the competitive positioning. If this product focuses on a niche segment while competitors are broader, identify that advantage. Respond with JSON:
|
|
248
|
+
|
|
249
|
+
{
|
|
250
|
+
"competitivePositioning": "Description of how this product is positioned vs competitors",
|
|
251
|
+
"nicheAdvantages": ["advantage1", "advantage2"],
|
|
252
|
+
"competitorKeywordsToTarget": ["keyword1", "keyword2"],
|
|
253
|
+
"competitorKeywordsToAvoid": ["keyword1", "keyword2"],
|
|
254
|
+
"updatedUncertainties": ["uncertainty1", "uncertainty2"]
|
|
255
|
+
}`;
|
|
256
|
+
|
|
257
|
+
try {
|
|
258
|
+
const response = await openai.chat.completions.create({
|
|
259
|
+
model: options.model || 'gpt-4o-mini',
|
|
260
|
+
messages: [
|
|
261
|
+
{ role: 'system', content: 'Respond with valid JSON only.' },
|
|
262
|
+
{ role: 'user', content: prompt },
|
|
263
|
+
],
|
|
264
|
+
temperature: 0.3,
|
|
265
|
+
max_tokens: 1000,
|
|
266
|
+
});
|
|
267
|
+
|
|
268
|
+
const responseText = response.choices[0]?.message?.content || '';
|
|
269
|
+
const jsonMatch = responseText.match(/\{[\s\S]*\}/);
|
|
270
|
+
if (jsonMatch) {
|
|
271
|
+
const analysis = JSON.parse(jsonMatch[0]);
|
|
272
|
+
return {
|
|
273
|
+
...summary,
|
|
274
|
+
competitivePositioning: analysis.competitivePositioning || '',
|
|
275
|
+
suggestedSeedKeywords: [
|
|
276
|
+
...summary.suggestedSeedKeywords,
|
|
277
|
+
...(analysis.competitorKeywordsToTarget || []),
|
|
278
|
+
],
|
|
279
|
+
uncertainties: [
|
|
280
|
+
...summary.uncertainties.filter(
|
|
281
|
+
(u) => !analysis.updatedUncertainties?.includes(u)
|
|
282
|
+
),
|
|
283
|
+
...(analysis.updatedUncertainties || []),
|
|
284
|
+
],
|
|
285
|
+
};
|
|
286
|
+
}
|
|
287
|
+
} catch (error) {
|
|
288
|
+
console.error('Competitive analysis failed:', error);
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
return { ...summary, competitivePositioning: '' };
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
/**
|
|
295
|
+
* Generate questions to reduce uncertainty
|
|
296
|
+
*/
|
|
297
|
+
export function generateUncertaintyQuestions(
|
|
298
|
+
summary: SiteSummary
|
|
299
|
+
): Array<{
|
|
300
|
+
id: string;
|
|
301
|
+
question: string;
|
|
302
|
+
category: string;
|
|
303
|
+
impact: number;
|
|
304
|
+
options?: string[];
|
|
305
|
+
}> {
|
|
306
|
+
const questions: Array<{
|
|
307
|
+
id: string;
|
|
308
|
+
question: string;
|
|
309
|
+
category: string;
|
|
310
|
+
impact: number;
|
|
311
|
+
options?: string[];
|
|
312
|
+
}> = [];
|
|
313
|
+
|
|
314
|
+
// Always ask about target audience if uncertain
|
|
315
|
+
if (summary.confidence < 0.7 || summary.targetAudience.includes('Unknown')) {
|
|
316
|
+
questions.push({
|
|
317
|
+
id: 'target_audience',
|
|
318
|
+
question: 'Who is your primary target customer?',
|
|
319
|
+
category: 'audience',
|
|
320
|
+
impact: 0.3,
|
|
321
|
+
options: [
|
|
322
|
+
'Small businesses (1-50 employees)',
|
|
323
|
+
'Mid-market companies (50-500 employees)',
|
|
324
|
+
'Enterprise (500+ employees)',
|
|
325
|
+
'Developers/Technical users',
|
|
326
|
+
'Marketing teams',
|
|
327
|
+
'Individual consumers',
|
|
328
|
+
'Freelancers/Agencies',
|
|
329
|
+
],
|
|
330
|
+
});
|
|
331
|
+
}
|
|
332
|
+
|
|
333
|
+
// Ask about industry if uncertain
|
|
334
|
+
if (summary.industry === 'Unknown' || summary.confidence < 0.6) {
|
|
335
|
+
questions.push({
|
|
336
|
+
id: 'industry',
|
|
337
|
+
question: 'What industry or category best describes your product?',
|
|
338
|
+
category: 'product',
|
|
339
|
+
impact: 0.25,
|
|
340
|
+
});
|
|
341
|
+
}
|
|
342
|
+
|
|
343
|
+
// Ask about main problem solved
|
|
344
|
+
if (summary.problemsSolved.length === 0) {
|
|
345
|
+
questions.push({
|
|
346
|
+
id: 'main_problem',
|
|
347
|
+
question: 'What is the main problem your product solves?',
|
|
348
|
+
category: 'product',
|
|
349
|
+
impact: 0.35,
|
|
350
|
+
});
|
|
351
|
+
}
|
|
352
|
+
|
|
353
|
+
// Ask about competitors
|
|
354
|
+
questions.push({
|
|
355
|
+
id: 'competitors',
|
|
356
|
+
question: 'Who are your main competitors? (comma-separated)',
|
|
357
|
+
category: 'competition',
|
|
358
|
+
impact: 0.2,
|
|
359
|
+
});
|
|
360
|
+
|
|
361
|
+
// Ask about differentiator
|
|
362
|
+
if (!summary.nicheFocus) {
|
|
363
|
+
questions.push({
|
|
364
|
+
id: 'differentiator',
|
|
365
|
+
question: 'What makes your product different from competitors?',
|
|
366
|
+
category: 'positioning',
|
|
367
|
+
impact: 0.25,
|
|
368
|
+
});
|
|
369
|
+
}
|
|
370
|
+
|
|
371
|
+
// Ask about goals
|
|
372
|
+
questions.push({
|
|
373
|
+
id: 'seo_goals',
|
|
374
|
+
question: 'What are your SEO goals?',
|
|
375
|
+
category: 'goals',
|
|
376
|
+
impact: 0.15,
|
|
377
|
+
options: [
|
|
378
|
+
'Get more signups/trials',
|
|
379
|
+
'Generate leads',
|
|
380
|
+
'Build brand awareness',
|
|
381
|
+
'Drive organic traffic',
|
|
382
|
+
'Rank for specific keywords',
|
|
383
|
+
],
|
|
384
|
+
});
|
|
385
|
+
|
|
386
|
+
return questions.sort((a, b) => b.impact - a.impact);
|
|
387
|
+
}
|