@rankcli/agent-runtime 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (178) hide show
  1. package/README.md +242 -0
  2. package/dist/analyzer-2CSWIQGD.mjs +6 -0
  3. package/dist/chunk-YNZYHEYM.mjs +774 -0
  4. package/dist/index.d.mts +4012 -0
  5. package/dist/index.d.ts +4012 -0
  6. package/dist/index.js +29672 -0
  7. package/dist/index.mjs +28602 -0
  8. package/package.json +53 -0
  9. package/scripts/build-deno.ts +134 -0
  10. package/src/audit/ai/analyzer.ts +347 -0
  11. package/src/audit/ai/index.ts +29 -0
  12. package/src/audit/ai/prompts/content-analysis.ts +271 -0
  13. package/src/audit/ai/types.ts +179 -0
  14. package/src/audit/checks/additional-checks.ts +439 -0
  15. package/src/audit/checks/ai-citation-worthiness.ts +399 -0
  16. package/src/audit/checks/ai-content-structure.ts +325 -0
  17. package/src/audit/checks/ai-readiness.ts +339 -0
  18. package/src/audit/checks/anchor-text.ts +179 -0
  19. package/src/audit/checks/answer-conciseness.ts +322 -0
  20. package/src/audit/checks/asset-minification.ts +270 -0
  21. package/src/audit/checks/bing-optimization.ts +206 -0
  22. package/src/audit/checks/brand-mention-optimization.ts +349 -0
  23. package/src/audit/checks/caching-headers.ts +305 -0
  24. package/src/audit/checks/canonical-advanced.ts +150 -0
  25. package/src/audit/checks/canonical-domain.ts +196 -0
  26. package/src/audit/checks/citation-quality.ts +358 -0
  27. package/src/audit/checks/client-rendering.ts +542 -0
  28. package/src/audit/checks/color-contrast.ts +342 -0
  29. package/src/audit/checks/content-freshness.ts +170 -0
  30. package/src/audit/checks/content-science.ts +589 -0
  31. package/src/audit/checks/conversion-elements.ts +526 -0
  32. package/src/audit/checks/crawlability.ts +220 -0
  33. package/src/audit/checks/directory-listing.ts +172 -0
  34. package/src/audit/checks/dom-analysis.ts +191 -0
  35. package/src/audit/checks/dom-size.ts +246 -0
  36. package/src/audit/checks/duplicate-content.ts +194 -0
  37. package/src/audit/checks/eeat-signals.ts +990 -0
  38. package/src/audit/checks/entity-seo.ts +396 -0
  39. package/src/audit/checks/featured-snippet.ts +473 -0
  40. package/src/audit/checks/freshness-signals.ts +443 -0
  41. package/src/audit/checks/funnel-intent.ts +463 -0
  42. package/src/audit/checks/hreflang.ts +174 -0
  43. package/src/audit/checks/html-compliance.ts +302 -0
  44. package/src/audit/checks/image-dimensions.ts +167 -0
  45. package/src/audit/checks/images.ts +160 -0
  46. package/src/audit/checks/indexnow.ts +275 -0
  47. package/src/audit/checks/interactive-tools.ts +475 -0
  48. package/src/audit/checks/internal-link-graph.ts +436 -0
  49. package/src/audit/checks/keyword-analysis.ts +239 -0
  50. package/src/audit/checks/keyword-cannibalization.ts +385 -0
  51. package/src/audit/checks/keyword-placement.ts +471 -0
  52. package/src/audit/checks/links.ts +203 -0
  53. package/src/audit/checks/llms-txt.ts +224 -0
  54. package/src/audit/checks/local-seo.ts +296 -0
  55. package/src/audit/checks/mobile.ts +167 -0
  56. package/src/audit/checks/modern-images.ts +226 -0
  57. package/src/audit/checks/navboost-signals.ts +395 -0
  58. package/src/audit/checks/on-page.ts +209 -0
  59. package/src/audit/checks/page-resources.ts +285 -0
  60. package/src/audit/checks/pagination.ts +180 -0
  61. package/src/audit/checks/performance.ts +153 -0
  62. package/src/audit/checks/platform-presence.ts +580 -0
  63. package/src/audit/checks/redirect-analysis.ts +153 -0
  64. package/src/audit/checks/redirect-chain.ts +389 -0
  65. package/src/audit/checks/resource-hints.ts +420 -0
  66. package/src/audit/checks/responsive-css.ts +247 -0
  67. package/src/audit/checks/responsive-images.ts +396 -0
  68. package/src/audit/checks/review-ecosystem.ts +415 -0
  69. package/src/audit/checks/robots-validation.ts +373 -0
  70. package/src/audit/checks/security-headers.ts +172 -0
  71. package/src/audit/checks/security.ts +144 -0
  72. package/src/audit/checks/serp-preview.ts +251 -0
  73. package/src/audit/checks/site-maturity.ts +444 -0
  74. package/src/audit/checks/social-meta.test.ts +275 -0
  75. package/src/audit/checks/social-meta.ts +134 -0
  76. package/src/audit/checks/soft-404.ts +151 -0
  77. package/src/audit/checks/structured-data.ts +238 -0
  78. package/src/audit/checks/tech-detection.ts +496 -0
  79. package/src/audit/checks/topical-clusters.ts +435 -0
  80. package/src/audit/checks/tracker-bloat.ts +462 -0
  81. package/src/audit/checks/tracking-verification.test.ts +371 -0
  82. package/src/audit/checks/tracking-verification.ts +636 -0
  83. package/src/audit/checks/url-safety.ts +682 -0
  84. package/src/audit/deno-entry.ts +66 -0
  85. package/src/audit/discovery/index.ts +15 -0
  86. package/src/audit/discovery/link-crawler.ts +232 -0
  87. package/src/audit/discovery/repo-routes.ts +347 -0
  88. package/src/audit/engine.ts +620 -0
  89. package/src/audit/fixes/index.ts +209 -0
  90. package/src/audit/fixes/social-meta-fixes.test.ts +329 -0
  91. package/src/audit/fixes/social-meta-fixes.ts +463 -0
  92. package/src/audit/index.ts +74 -0
  93. package/src/audit/runner.test.ts +299 -0
  94. package/src/audit/runner.ts +130 -0
  95. package/src/audit/types.ts +1953 -0
  96. package/src/content/featured-snippet.ts +367 -0
  97. package/src/content/generator.test.ts +534 -0
  98. package/src/content/generator.ts +501 -0
  99. package/src/content/headline.ts +317 -0
  100. package/src/content/index.ts +62 -0
  101. package/src/content/intent.ts +258 -0
  102. package/src/content/keyword-density.ts +349 -0
  103. package/src/content/readability.ts +262 -0
  104. package/src/executor.ts +336 -0
  105. package/src/fixer.ts +416 -0
  106. package/src/frameworks/detector.test.ts +248 -0
  107. package/src/frameworks/detector.ts +371 -0
  108. package/src/frameworks/index.ts +68 -0
  109. package/src/frameworks/recipes/angular.yaml +171 -0
  110. package/src/frameworks/recipes/astro.yaml +206 -0
  111. package/src/frameworks/recipes/django.yaml +180 -0
  112. package/src/frameworks/recipes/laravel.yaml +137 -0
  113. package/src/frameworks/recipes/nextjs.yaml +268 -0
  114. package/src/frameworks/recipes/nuxt.yaml +175 -0
  115. package/src/frameworks/recipes/rails.yaml +188 -0
  116. package/src/frameworks/recipes/react.yaml +202 -0
  117. package/src/frameworks/recipes/sveltekit.yaml +154 -0
  118. package/src/frameworks/recipes/vue.yaml +137 -0
  119. package/src/frameworks/recipes/wordpress.yaml +209 -0
  120. package/src/frameworks/suggestion-engine.ts +320 -0
  121. package/src/geo/geo-content.test.ts +305 -0
  122. package/src/geo/geo-content.ts +266 -0
  123. package/src/geo/geo-history.test.ts +473 -0
  124. package/src/geo/geo-history.ts +433 -0
  125. package/src/geo/geo-tracker.test.ts +359 -0
  126. package/src/geo/geo-tracker.ts +411 -0
  127. package/src/geo/index.ts +10 -0
  128. package/src/git/commit-helper.test.ts +261 -0
  129. package/src/git/commit-helper.ts +329 -0
  130. package/src/git/index.ts +12 -0
  131. package/src/git/pr-helper.test.ts +284 -0
  132. package/src/git/pr-helper.ts +307 -0
  133. package/src/index.ts +66 -0
  134. package/src/keywords/ai-keyword-engine.ts +1062 -0
  135. package/src/keywords/ai-summarizer.ts +387 -0
  136. package/src/keywords/ci-mode.ts +555 -0
  137. package/src/keywords/engine.ts +359 -0
  138. package/src/keywords/index.ts +151 -0
  139. package/src/keywords/llm-judge.ts +357 -0
  140. package/src/keywords/nlp-analysis.ts +706 -0
  141. package/src/keywords/prioritizer.ts +295 -0
  142. package/src/keywords/site-crawler.ts +342 -0
  143. package/src/keywords/sources/autocomplete.ts +139 -0
  144. package/src/keywords/sources/competitive-search.ts +450 -0
  145. package/src/keywords/sources/competitor-analysis.ts +374 -0
  146. package/src/keywords/sources/dataforseo.ts +206 -0
  147. package/src/keywords/sources/free-sources.ts +294 -0
  148. package/src/keywords/sources/gsc.ts +123 -0
  149. package/src/keywords/topic-grouping.ts +327 -0
  150. package/src/keywords/types.ts +144 -0
  151. package/src/keywords/wizard.ts +457 -0
  152. package/src/loader.ts +40 -0
  153. package/src/reports/index.ts +7 -0
  154. package/src/reports/report-generator.test.ts +293 -0
  155. package/src/reports/report-generator.ts +713 -0
  156. package/src/scheduler/alerts.test.ts +458 -0
  157. package/src/scheduler/alerts.ts +328 -0
  158. package/src/scheduler/index.ts +8 -0
  159. package/src/scheduler/scheduled-audit.test.ts +377 -0
  160. package/src/scheduler/scheduled-audit.ts +149 -0
  161. package/src/test/integration-test.ts +325 -0
  162. package/src/tools/analyzer.ts +373 -0
  163. package/src/tools/crawl.ts +293 -0
  164. package/src/tools/files.ts +301 -0
  165. package/src/tools/h1-fixer.ts +249 -0
  166. package/src/tools/index.ts +67 -0
  167. package/src/tracking/github-action.ts +326 -0
  168. package/src/tracking/google-analytics.ts +265 -0
  169. package/src/tracking/index.ts +45 -0
  170. package/src/tracking/report-generator.ts +386 -0
  171. package/src/tracking/search-console.ts +335 -0
  172. package/src/types.ts +134 -0
  173. package/src/utils/http.ts +302 -0
  174. package/src/wasm-adapter.ts +297 -0
  175. package/src/wasm-entry.ts +14 -0
  176. package/tsconfig.json +17 -0
  177. package/tsup.wasm.config.ts +26 -0
  178. package/vitest.config.ts +15 -0
@@ -0,0 +1,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
+ }