@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,534 @@
|
|
|
1
|
+
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
2
|
+
import {
|
|
3
|
+
generateBlogPost,
|
|
4
|
+
generateChangelog,
|
|
5
|
+
optimizeReadme,
|
|
6
|
+
scoreContentSEO,
|
|
7
|
+
ContentGeneratorOptions,
|
|
8
|
+
BlogPostConfig,
|
|
9
|
+
ChangelogConfig,
|
|
10
|
+
ReadmeConfig,
|
|
11
|
+
GeneratedContent,
|
|
12
|
+
SEOScore,
|
|
13
|
+
} from './generator.js';
|
|
14
|
+
|
|
15
|
+
describe('content-generator', () => {
|
|
16
|
+
beforeEach(() => {
|
|
17
|
+
vi.clearAllMocks();
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
describe('generateBlogPost', () => {
|
|
21
|
+
it('generates blog post with title', async () => {
|
|
22
|
+
const mockLLM = vi.fn().mockResolvedValue({
|
|
23
|
+
title: 'How to Use Our API: A Complete Guide',
|
|
24
|
+
content: '# How to Use Our API\n\nThis guide covers...',
|
|
25
|
+
metaDescription: 'Learn how to use our API with this step-by-step guide.',
|
|
26
|
+
keywords: ['api', 'guide', 'tutorial'],
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
const config: BlogPostConfig = {
|
|
30
|
+
topic: 'how to use our API',
|
|
31
|
+
keywords: ['api', 'tutorial'],
|
|
32
|
+
tone: 'technical',
|
|
33
|
+
};
|
|
34
|
+
|
|
35
|
+
const result = await generateBlogPost(config, { llm: mockLLM });
|
|
36
|
+
|
|
37
|
+
expect(result.title).toBeDefined();
|
|
38
|
+
expect(result.content).toBeDefined();
|
|
39
|
+
expect(result.metaDescription).toBeDefined();
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
it('includes target keywords in content', async () => {
|
|
43
|
+
const mockLLM = vi.fn().mockResolvedValue({
|
|
44
|
+
title: 'REST API Tutorial',
|
|
45
|
+
content: '# REST API Tutorial\n\nLearn how to build a REST API...',
|
|
46
|
+
metaDescription: 'A complete REST API tutorial for beginners.',
|
|
47
|
+
keywords: ['rest', 'api', 'tutorial'],
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
const config: BlogPostConfig = {
|
|
51
|
+
topic: 'REST API tutorial',
|
|
52
|
+
keywords: ['rest api', 'tutorial'],
|
|
53
|
+
tone: 'educational',
|
|
54
|
+
};
|
|
55
|
+
|
|
56
|
+
const result = await generateBlogPost(config, { llm: mockLLM });
|
|
57
|
+
|
|
58
|
+
expect(result.keywords).toContain('rest');
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
it('generates SEO-optimized meta description', async () => {
|
|
62
|
+
const mockLLM = vi.fn().mockResolvedValue({
|
|
63
|
+
title: 'Test Post',
|
|
64
|
+
content: '# Test\n\nContent here...',
|
|
65
|
+
metaDescription: 'A great meta description under 160 chars.',
|
|
66
|
+
keywords: [],
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
const config: BlogPostConfig = {
|
|
70
|
+
topic: 'test topic',
|
|
71
|
+
};
|
|
72
|
+
|
|
73
|
+
const result = await generateBlogPost(config, { llm: mockLLM });
|
|
74
|
+
|
|
75
|
+
expect(result.metaDescription.length).toBeLessThanOrEqual(160);
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
it('respects word count limits', async () => {
|
|
79
|
+
const mockLLM = vi.fn().mockResolvedValue({
|
|
80
|
+
title: 'Short Post',
|
|
81
|
+
content: '# Short\n\nThis is a short post with limited words.',
|
|
82
|
+
metaDescription: 'A short post.',
|
|
83
|
+
keywords: [],
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
const config: BlogPostConfig = {
|
|
87
|
+
topic: 'short topic',
|
|
88
|
+
wordCount: { min: 100, max: 500 },
|
|
89
|
+
};
|
|
90
|
+
|
|
91
|
+
const result = await generateBlogPost(config, { llm: mockLLM });
|
|
92
|
+
|
|
93
|
+
expect(result.content).toBeDefined();
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
it('handles LLM errors gracefully', async () => {
|
|
97
|
+
const mockLLM = vi.fn().mockRejectedValue(new Error('LLM unavailable'));
|
|
98
|
+
|
|
99
|
+
const config: BlogPostConfig = {
|
|
100
|
+
topic: 'test topic',
|
|
101
|
+
};
|
|
102
|
+
|
|
103
|
+
const result = await generateBlogPost(config, { llm: mockLLM });
|
|
104
|
+
|
|
105
|
+
expect(result.error).toBeDefined();
|
|
106
|
+
expect(result.content).toBe('');
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
describe('generateChangelog', () => {
|
|
111
|
+
it('generates changelog from git commits', async () => {
|
|
112
|
+
const mockExec = vi.fn().mockResolvedValue({
|
|
113
|
+
stdout: `feat: add user authentication
|
|
114
|
+
fix: resolve login bug
|
|
115
|
+
chore: update dependencies`,
|
|
116
|
+
});
|
|
117
|
+
|
|
118
|
+
const config: ChangelogConfig = {
|
|
119
|
+
fromRef: 'v1.0.0',
|
|
120
|
+
toRef: 'HEAD',
|
|
121
|
+
};
|
|
122
|
+
|
|
123
|
+
const result = await generateChangelog(config, { exec: mockExec });
|
|
124
|
+
|
|
125
|
+
expect(result.content).toContain('feat');
|
|
126
|
+
expect(result.sections).toBeDefined();
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
it('groups changes by type', async () => {
|
|
130
|
+
const mockExec = vi.fn().mockResolvedValue({
|
|
131
|
+
stdout: `feat: feature 1
|
|
132
|
+
feat: feature 2
|
|
133
|
+
fix: bug fix
|
|
134
|
+
docs: update readme`,
|
|
135
|
+
});
|
|
136
|
+
|
|
137
|
+
const config: ChangelogConfig = {
|
|
138
|
+
fromRef: 'v1.0.0',
|
|
139
|
+
toRef: 'HEAD',
|
|
140
|
+
groupByType: true,
|
|
141
|
+
};
|
|
142
|
+
|
|
143
|
+
const result = await generateChangelog(config, { exec: mockExec });
|
|
144
|
+
|
|
145
|
+
expect(result.sections).toHaveProperty('features');
|
|
146
|
+
expect(result.sections).toHaveProperty('fixes');
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
it('includes date range', async () => {
|
|
150
|
+
const mockExec = vi.fn().mockResolvedValue({
|
|
151
|
+
stdout: `feat: new feature`,
|
|
152
|
+
});
|
|
153
|
+
|
|
154
|
+
const config: ChangelogConfig = {
|
|
155
|
+
fromRef: 'v1.0.0',
|
|
156
|
+
toRef: 'v1.1.0',
|
|
157
|
+
};
|
|
158
|
+
|
|
159
|
+
const result = await generateChangelog(config, { exec: mockExec });
|
|
160
|
+
|
|
161
|
+
expect(result.version).toBeDefined();
|
|
162
|
+
});
|
|
163
|
+
|
|
164
|
+
it('handles empty commit range', async () => {
|
|
165
|
+
const mockExec = vi.fn().mockResolvedValue({
|
|
166
|
+
stdout: '',
|
|
167
|
+
});
|
|
168
|
+
|
|
169
|
+
const config: ChangelogConfig = {
|
|
170
|
+
fromRef: 'v1.0.0',
|
|
171
|
+
toRef: 'v1.0.0',
|
|
172
|
+
};
|
|
173
|
+
|
|
174
|
+
const result = await generateChangelog(config, { exec: mockExec });
|
|
175
|
+
|
|
176
|
+
expect(result.content).toBe('');
|
|
177
|
+
expect(result.sections).toEqual({});
|
|
178
|
+
});
|
|
179
|
+
|
|
180
|
+
it('formats for SEO with keywords', async () => {
|
|
181
|
+
const mockExec = vi.fn().mockResolvedValue({
|
|
182
|
+
stdout: `feat: add new API endpoint`,
|
|
183
|
+
});
|
|
184
|
+
|
|
185
|
+
const config: ChangelogConfig = {
|
|
186
|
+
fromRef: 'v1.0.0',
|
|
187
|
+
toRef: 'HEAD',
|
|
188
|
+
seoKeywords: ['api', 'update'],
|
|
189
|
+
};
|
|
190
|
+
|
|
191
|
+
const result = await generateChangelog(config, { exec: mockExec });
|
|
192
|
+
|
|
193
|
+
expect(result.metaDescription).toBeDefined();
|
|
194
|
+
});
|
|
195
|
+
});
|
|
196
|
+
|
|
197
|
+
describe('optimizeReadme', () => {
|
|
198
|
+
it('adds missing sections', async () => {
|
|
199
|
+
const mockLLM = vi.fn().mockResolvedValue({
|
|
200
|
+
content: '# Project\n\n## Installation\n\n## Usage\n\n## License',
|
|
201
|
+
suggestions: ['Add badges', 'Add examples'],
|
|
202
|
+
});
|
|
203
|
+
|
|
204
|
+
const config: ReadmeConfig = {
|
|
205
|
+
currentContent: '# Project\n\nBasic readme content.',
|
|
206
|
+
projectName: 'my-project',
|
|
207
|
+
};
|
|
208
|
+
|
|
209
|
+
const result = await optimizeReadme(config, { llm: mockLLM });
|
|
210
|
+
|
|
211
|
+
expect(result.content).toContain('Installation');
|
|
212
|
+
});
|
|
213
|
+
|
|
214
|
+
it('adds SEO-friendly description', async () => {
|
|
215
|
+
const mockLLM = vi.fn().mockResolvedValue({
|
|
216
|
+
content: '# Project\n\nA powerful tool for developers.',
|
|
217
|
+
suggestions: [],
|
|
218
|
+
metaDescription: 'A powerful tool for developers.',
|
|
219
|
+
});
|
|
220
|
+
|
|
221
|
+
const config: ReadmeConfig = {
|
|
222
|
+
currentContent: '# Project',
|
|
223
|
+
projectName: 'my-project',
|
|
224
|
+
keywords: ['developer tool'],
|
|
225
|
+
};
|
|
226
|
+
|
|
227
|
+
const result = await optimizeReadme(config, { llm: mockLLM });
|
|
228
|
+
|
|
229
|
+
expect(result.metaDescription).toBeDefined();
|
|
230
|
+
});
|
|
231
|
+
|
|
232
|
+
it('suggests badge additions', async () => {
|
|
233
|
+
const mockLLM = vi.fn().mockResolvedValue({
|
|
234
|
+
content: '# Project',
|
|
235
|
+
suggestions: ['Add npm version badge', 'Add license badge'],
|
|
236
|
+
});
|
|
237
|
+
|
|
238
|
+
const config: ReadmeConfig = {
|
|
239
|
+
currentContent: '# Project',
|
|
240
|
+
projectName: 'my-project',
|
|
241
|
+
};
|
|
242
|
+
|
|
243
|
+
const result = await optimizeReadme(config, { llm: mockLLM });
|
|
244
|
+
|
|
245
|
+
expect(result.suggestions).toContain('Add npm version badge');
|
|
246
|
+
});
|
|
247
|
+
|
|
248
|
+
it('preserves existing content structure', async () => {
|
|
249
|
+
const existingContent = `# My Project
|
|
250
|
+
|
|
251
|
+
## Custom Section
|
|
252
|
+
|
|
253
|
+
My custom content here.`;
|
|
254
|
+
|
|
255
|
+
const mockLLM = vi.fn().mockResolvedValue({
|
|
256
|
+
content: `# My Project
|
|
257
|
+
|
|
258
|
+
## Custom Section
|
|
259
|
+
|
|
260
|
+
My custom content here.
|
|
261
|
+
|
|
262
|
+
## Installation
|
|
263
|
+
|
|
264
|
+
npm install my-project`,
|
|
265
|
+
suggestions: [],
|
|
266
|
+
});
|
|
267
|
+
|
|
268
|
+
const config: ReadmeConfig = {
|
|
269
|
+
currentContent: existingContent,
|
|
270
|
+
projectName: 'my-project',
|
|
271
|
+
preserveCustomSections: true,
|
|
272
|
+
};
|
|
273
|
+
|
|
274
|
+
const result = await optimizeReadme(config, { llm: mockLLM });
|
|
275
|
+
|
|
276
|
+
expect(result.content).toContain('Custom Section');
|
|
277
|
+
});
|
|
278
|
+
|
|
279
|
+
it('handles empty readme', async () => {
|
|
280
|
+
const mockLLM = vi.fn().mockResolvedValue({
|
|
281
|
+
content: '# My Project\n\nA great project.',
|
|
282
|
+
suggestions: ['Add all sections'],
|
|
283
|
+
});
|
|
284
|
+
|
|
285
|
+
const config: ReadmeConfig = {
|
|
286
|
+
currentContent: '',
|
|
287
|
+
projectName: 'my-project',
|
|
288
|
+
};
|
|
289
|
+
|
|
290
|
+
const result = await optimizeReadme(config, { llm: mockLLM });
|
|
291
|
+
|
|
292
|
+
expect(result.content.length).toBeGreaterThan(0);
|
|
293
|
+
});
|
|
294
|
+
});
|
|
295
|
+
|
|
296
|
+
describe('scoreContentSEO', () => {
|
|
297
|
+
it('scores content based on SEO factors', () => {
|
|
298
|
+
const content = `# Great Title
|
|
299
|
+
|
|
300
|
+
This is a great article about SEO optimization.
|
|
301
|
+
|
|
302
|
+
## Introduction
|
|
303
|
+
|
|
304
|
+
Learn about SEO best practices.
|
|
305
|
+
|
|
306
|
+
## Conclusion
|
|
307
|
+
|
|
308
|
+
In summary, SEO is important.`;
|
|
309
|
+
|
|
310
|
+
const score = scoreContentSEO(content, {
|
|
311
|
+
targetKeywords: ['seo', 'optimization'],
|
|
312
|
+
});
|
|
313
|
+
|
|
314
|
+
expect(score.overall).toBeGreaterThan(0);
|
|
315
|
+
expect(score.overall).toBeLessThanOrEqual(100);
|
|
316
|
+
});
|
|
317
|
+
|
|
318
|
+
it('checks for heading structure', () => {
|
|
319
|
+
const goodContent = `# Main Title
|
|
320
|
+
|
|
321
|
+
## Section 1
|
|
322
|
+
|
|
323
|
+
Content here.
|
|
324
|
+
|
|
325
|
+
## Section 2
|
|
326
|
+
|
|
327
|
+
More content.`;
|
|
328
|
+
|
|
329
|
+
const badContent = `Main Title
|
|
330
|
+
|
|
331
|
+
No headings at all.`;
|
|
332
|
+
|
|
333
|
+
const goodScore = scoreContentSEO(goodContent);
|
|
334
|
+
const badScore = scoreContentSEO(badContent);
|
|
335
|
+
|
|
336
|
+
expect(goodScore.headingScore).toBeGreaterThan(badScore.headingScore);
|
|
337
|
+
});
|
|
338
|
+
|
|
339
|
+
it('checks keyword density', () => {
|
|
340
|
+
const contentWithKeywords = `# SEO Guide
|
|
341
|
+
|
|
342
|
+
SEO is important for developers. This SEO guide covers SEO basics.`;
|
|
343
|
+
|
|
344
|
+
const contentWithoutKeywords = `# Guide
|
|
345
|
+
|
|
346
|
+
This guide covers various topics about web development.`;
|
|
347
|
+
|
|
348
|
+
const withScore = scoreContentSEO(contentWithKeywords, {
|
|
349
|
+
targetKeywords: ['seo'],
|
|
350
|
+
});
|
|
351
|
+
const withoutScore = scoreContentSEO(contentWithoutKeywords, {
|
|
352
|
+
targetKeywords: ['seo'],
|
|
353
|
+
});
|
|
354
|
+
|
|
355
|
+
expect(withScore.keywordScore).toBeGreaterThan(withoutScore.keywordScore);
|
|
356
|
+
});
|
|
357
|
+
|
|
358
|
+
it('checks content length', () => {
|
|
359
|
+
const shortContent = '# Title\n\nShort.';
|
|
360
|
+
const longContent = `# Title
|
|
361
|
+
|
|
362
|
+
${'This is a paragraph of content. '.repeat(50)}`;
|
|
363
|
+
|
|
364
|
+
const shortScore = scoreContentSEO(shortContent);
|
|
365
|
+
const longScore = scoreContentSEO(longContent);
|
|
366
|
+
|
|
367
|
+
expect(longScore.lengthScore).toBeGreaterThan(shortScore.lengthScore);
|
|
368
|
+
});
|
|
369
|
+
|
|
370
|
+
it('checks for internal links', () => {
|
|
371
|
+
const contentWithLinks = `# Guide
|
|
372
|
+
|
|
373
|
+
Check out [our API docs](/docs/api) and [tutorials](/tutorials).`;
|
|
374
|
+
|
|
375
|
+
const contentWithoutLinks = `# Guide
|
|
376
|
+
|
|
377
|
+
This is content without any links.`;
|
|
378
|
+
|
|
379
|
+
const withLinksScore = scoreContentSEO(contentWithLinks);
|
|
380
|
+
const withoutLinksScore = scoreContentSEO(contentWithoutLinks);
|
|
381
|
+
|
|
382
|
+
expect(withLinksScore.linkScore).toBeGreaterThan(withoutLinksScore.linkScore);
|
|
383
|
+
});
|
|
384
|
+
|
|
385
|
+
it('checks for images with alt text', () => {
|
|
386
|
+
const contentWithImages = `# Guide
|
|
387
|
+
|
|
388
|
+
`;
|
|
389
|
+
|
|
390
|
+
const contentWithoutImages = `# Guide
|
|
391
|
+
|
|
392
|
+
Just text content.`;
|
|
393
|
+
|
|
394
|
+
const withImagesScore = scoreContentSEO(contentWithImages);
|
|
395
|
+
const withoutImagesScore = scoreContentSEO(contentWithoutImages);
|
|
396
|
+
|
|
397
|
+
expect(withImagesScore.mediaScore).toBeGreaterThan(withoutImagesScore.mediaScore);
|
|
398
|
+
});
|
|
399
|
+
|
|
400
|
+
it('returns breakdown of score factors', () => {
|
|
401
|
+
const content = '# Test\n\nContent here.';
|
|
402
|
+
|
|
403
|
+
const score = scoreContentSEO(content);
|
|
404
|
+
|
|
405
|
+
expect(score).toHaveProperty('overall');
|
|
406
|
+
expect(score).toHaveProperty('headingScore');
|
|
407
|
+
expect(score).toHaveProperty('keywordScore');
|
|
408
|
+
expect(score).toHaveProperty('lengthScore');
|
|
409
|
+
expect(score).toHaveProperty('linkScore');
|
|
410
|
+
expect(score).toHaveProperty('mediaScore');
|
|
411
|
+
});
|
|
412
|
+
});
|
|
413
|
+
|
|
414
|
+
describe('GEO optimization', () => {
|
|
415
|
+
it('structures content for AI citation', async () => {
|
|
416
|
+
const mockLLM = vi.fn().mockResolvedValue({
|
|
417
|
+
title: 'SEO Autopilot: Best CLI Tool for Developers',
|
|
418
|
+
content: `# SEO Autopilot: Best CLI Tool for Developers
|
|
419
|
+
|
|
420
|
+
**Key Fact:** SEO Autopilot runs 200+ automated SEO checks directly from your terminal.
|
|
421
|
+
|
|
422
|
+
## Features
|
|
423
|
+
|
|
424
|
+
- Command-line interface
|
|
425
|
+
- GitHub integration
|
|
426
|
+
- Auto-fix PRs`,
|
|
427
|
+
metaDescription: 'SEO Autopilot is the best CLI tool for developers.',
|
|
428
|
+
keywords: ['seo', 'cli', 'developers'],
|
|
429
|
+
});
|
|
430
|
+
|
|
431
|
+
const config: BlogPostConfig = {
|
|
432
|
+
topic: 'SEO Autopilot CLI',
|
|
433
|
+
geoOptimize: true,
|
|
434
|
+
};
|
|
435
|
+
|
|
436
|
+
const result = await generateBlogPost(config, { llm: mockLLM });
|
|
437
|
+
|
|
438
|
+
// GEO-optimized content should have citable facts
|
|
439
|
+
expect(result.content).toMatch(/\*\*.*\*\*/); // Bold facts
|
|
440
|
+
});
|
|
441
|
+
|
|
442
|
+
it('includes FAQ section for AI parsing', async () => {
|
|
443
|
+
const mockLLM = vi.fn().mockResolvedValue({
|
|
444
|
+
title: 'FAQ Post',
|
|
445
|
+
content: `# FAQ
|
|
446
|
+
|
|
447
|
+
## What is SEO Autopilot?
|
|
448
|
+
|
|
449
|
+
SEO Autopilot is a CLI-first SEO tool.
|
|
450
|
+
|
|
451
|
+
## How does it work?
|
|
452
|
+
|
|
453
|
+
It runs 200+ checks on your website.`,
|
|
454
|
+
metaDescription: 'FAQ about SEO Autopilot.',
|
|
455
|
+
keywords: [],
|
|
456
|
+
hasFAQ: true,
|
|
457
|
+
});
|
|
458
|
+
|
|
459
|
+
const config: BlogPostConfig = {
|
|
460
|
+
topic: 'FAQ about SEO',
|
|
461
|
+
includeFAQ: true,
|
|
462
|
+
};
|
|
463
|
+
|
|
464
|
+
const result = await generateBlogPost(config, { llm: mockLLM });
|
|
465
|
+
|
|
466
|
+
expect(result.hasFAQ).toBe(true);
|
|
467
|
+
});
|
|
468
|
+
});
|
|
469
|
+
|
|
470
|
+
describe('integration scenarios', () => {
|
|
471
|
+
it('generates complete blog post with all SEO elements', async () => {
|
|
472
|
+
const mockLLM = vi.fn().mockResolvedValue({
|
|
473
|
+
title: 'Complete SEO Guide for Developers',
|
|
474
|
+
content: `# Complete SEO Guide for Developers
|
|
475
|
+
|
|
476
|
+
Learn everything about SEO optimization for your web applications.
|
|
477
|
+
|
|
478
|
+
## Introduction
|
|
479
|
+
|
|
480
|
+
SEO is crucial for modern web development.
|
|
481
|
+
|
|
482
|
+
## Key Concepts
|
|
483
|
+
|
|
484
|
+
Understanding these concepts will help you rank better.
|
|
485
|
+
|
|
486
|
+
## Conclusion
|
|
487
|
+
|
|
488
|
+
Start optimizing your sites today!`,
|
|
489
|
+
metaDescription: 'A complete guide to SEO for developers. Learn optimization techniques.',
|
|
490
|
+
keywords: ['seo', 'developers', 'guide', 'optimization'],
|
|
491
|
+
readingTime: '5 min',
|
|
492
|
+
wordCount: 150,
|
|
493
|
+
});
|
|
494
|
+
|
|
495
|
+
const config: BlogPostConfig = {
|
|
496
|
+
topic: 'SEO for developers',
|
|
497
|
+
keywords: ['seo', 'developers'],
|
|
498
|
+
tone: 'educational',
|
|
499
|
+
wordCount: { min: 100, max: 500 },
|
|
500
|
+
};
|
|
501
|
+
|
|
502
|
+
const result = await generateBlogPost(config, { llm: mockLLM });
|
|
503
|
+
|
|
504
|
+
expect(result.title).toBeDefined();
|
|
505
|
+
expect(result.content).toBeDefined();
|
|
506
|
+
expect(result.metaDescription).toBeDefined();
|
|
507
|
+
expect(result.keywords).toBeDefined();
|
|
508
|
+
expect(result.readingTime).toBeDefined();
|
|
509
|
+
});
|
|
510
|
+
|
|
511
|
+
it('generates release notes from conventional commits', async () => {
|
|
512
|
+
const mockExec = vi.fn().mockResolvedValue({
|
|
513
|
+
stdout: `feat(api): add new endpoint for user profiles
|
|
514
|
+
fix(auth): resolve token refresh issue
|
|
515
|
+
feat(cli): add --verbose flag
|
|
516
|
+
fix(ui): correct button alignment
|
|
517
|
+
docs: update API documentation
|
|
518
|
+
chore: bump dependencies`,
|
|
519
|
+
});
|
|
520
|
+
|
|
521
|
+
const config: ChangelogConfig = {
|
|
522
|
+
fromRef: 'v1.0.0',
|
|
523
|
+
toRef: 'v1.1.0',
|
|
524
|
+
groupByType: true,
|
|
525
|
+
includeBreaking: true,
|
|
526
|
+
};
|
|
527
|
+
|
|
528
|
+
const result = await generateChangelog(config, { exec: mockExec });
|
|
529
|
+
|
|
530
|
+
expect(result.sections.features).toHaveLength(2);
|
|
531
|
+
expect(result.sections.fixes).toHaveLength(2);
|
|
532
|
+
});
|
|
533
|
+
});
|
|
534
|
+
});
|