@contractspec/lib.content-gen 0.0.0-canary-20260113162409

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/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2025 Chaman Ventures, SASU
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,44 @@
1
+ # @contractspec/lib.content-gen
2
+
3
+ Website: https://contractspec.io/
4
+
5
+
6
+ Composable generators that turn ContractSpec briefs, specs, and telemetry into publish-ready marketing artifacts: blogs, landing pages, SEO metadata, email drips, and social posts.
7
+
8
+ ## Modules
9
+
10
+ - `BlogGenerator` – creates narrative posts from feature briefs and spec metadata.
11
+ - `LandingPageGenerator` – builds hero copy, feature bullets, proof points, and CTAs.
12
+ - `EmailCampaignGenerator` – drafts onboarding, announcement, and nurture emails.
13
+ - `SocialPostGenerator` – outputs multi-channel snippets with hashtags and CTAs.
14
+ - `SeoOptimizer` – extracts keywords, meta tags, slugs, and Schema.org markup.
15
+
16
+ Each generator accepts a `ContentBrief` that captures audience, value props, proof points, and compliance notes. If an `LLMProvider` is supplied, outputs blend AI creativity with deterministic templates; otherwise a deterministic fallback ensures consistent copy.
17
+
18
+ ## Quickstart
19
+
20
+ ```ts
21
+ import {
22
+ BlogGenerator,
23
+ LandingPageGenerator,
24
+ EmailCampaignGenerator,
25
+ SocialPostGenerator,
26
+ } from '@contractspec/lib.content-gen/generators';
27
+ import { SeoOptimizer } from '@contractspec/lib.content-gen/seo';
28
+
29
+ const brief = {
30
+ title: 'Policy-safe Workflow Automation',
31
+ summary: 'Automatically compiles intents into auditable workflows.',
32
+ problems: ['Manual reviews slow compliance', 'Engineers rebuild flows for every tenant'],
33
+ solutions: ['Spec-first workflows', 'Policy enforcement', 'Multi-tenant guardrails'],
34
+ audience: { role: 'COO', industry: 'Fintech', maturity: 'scaleup' },
35
+ };
36
+
37
+ const blog = await new BlogGenerator().generate(brief);
38
+ const landing = await new LandingPageGenerator().generate(brief);
39
+ const email = await new EmailCampaignGenerator().generate({ brief, variant: 'announcement' });
40
+ const social = await new SocialPostGenerator().generate(brief);
41
+ const seo = new SeoOptimizer().optimize(brief);
42
+ ```
43
+
44
+ Outputs return structured blocks that can be rendered directly in the marketing site or exported to CMS tools.
@@ -0,0 +1,16 @@
1
+ import { ContentBrief, GeneratedContent, GeneratorOptions } from "../types.js";
2
+
3
+ //#region src/generators/blog.d.ts
4
+ declare class BlogGenerator {
5
+ private readonly llm?;
6
+ private readonly model?;
7
+ private readonly temperature;
8
+ constructor(options?: GeneratorOptions);
9
+ generate(brief: ContentBrief): Promise<GeneratedContent>;
10
+ private generateWithLlm;
11
+ private generateDeterministic;
12
+ private renderWhyNow;
13
+ }
14
+ //#endregion
15
+ export { BlogGenerator };
16
+ //# sourceMappingURL=blog.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"blog.d.ts","names":[],"sources":["../../src/generators/blog.ts"],"sourcesContent":[],"mappings":";;;cAOa,aAAA;;EAAA,iBAAa,KAAA;EAKF,iBAAA,WAAA;EAMA,WAAA,CAAA,OAAA,CAAA,EANA,gBAMA;EAAuB,QAAA,CAAA,KAAA,EAAvB,YAAuB,CAAA,EAAR,OAAQ,CAAA,gBAAA,CAAA;EAAR,QAAA,eAAA;EAAO,QAAA,qBAAA"}
@@ -0,0 +1,70 @@
1
+ //#region src/generators/blog.ts
2
+ var BlogGenerator = class {
3
+ llm;
4
+ model;
5
+ temperature;
6
+ constructor(options) {
7
+ this.llm = options?.llm;
8
+ this.model = options?.model;
9
+ this.temperature = options?.temperature ?? .4;
10
+ }
11
+ async generate(brief) {
12
+ if (this.llm) return this.generateWithLlm(brief);
13
+ return this.generateDeterministic(brief);
14
+ }
15
+ async generateWithLlm(brief) {
16
+ if (!this.llm) return this.generateDeterministic(brief);
17
+ const jsonPart = (await this.llm.chat([{
18
+ role: "system",
19
+ content: [{
20
+ type: "text",
21
+ text: "You are a product marketing writer. Produce JSON with title, subtitle, intro, sections[].heading/body/bullets, outro."
22
+ }]
23
+ }, {
24
+ role: "user",
25
+ content: [{
26
+ type: "text",
27
+ text: JSON.stringify({ brief })
28
+ }]
29
+ }], {
30
+ responseFormat: "json",
31
+ model: this.model,
32
+ temperature: this.temperature
33
+ })).message.content.find((part) => "text" in part);
34
+ if (jsonPart && "text" in jsonPart) return JSON.parse(jsonPart.text);
35
+ return this.generateDeterministic(brief);
36
+ }
37
+ generateDeterministic(brief) {
38
+ const intro = `Operators like ${brief.audience.role} teams face ${brief.problems.slice(0, 2).join(" and ")}. ${brief.title} changes that by ${brief.summary}.`;
39
+ const sections = [
40
+ {
41
+ heading: "Why now",
42
+ body: this.renderWhyNow(brief)
43
+ },
44
+ {
45
+ heading: "What you get",
46
+ body: "A focused stack built for policy-safe automation.",
47
+ bullets: brief.solutions
48
+ },
49
+ {
50
+ heading: "Proof it works",
51
+ body: "Teams using the blueprint report measurable wins.",
52
+ bullets: brief.metrics ?? ["Launch workflows in minutes", "Cut review time by 60%"]
53
+ }
54
+ ];
55
+ return {
56
+ title: brief.title,
57
+ subtitle: brief.summary,
58
+ intro,
59
+ sections,
60
+ outro: brief.callToAction ?? "Ready to see it live? Spin up a sandbox in under 5 minutes."
61
+ };
62
+ }
63
+ renderWhyNow(brief) {
64
+ return `${`${brief.audience.role}${brief.audience.industry ? ` in ${brief.audience.industry}` : ""}`} teams are stuck with ${brief.problems.slice(0, 2).join("; ")}. ${brief.title} delivers guardrails without slowing shipping.`;
65
+ }
66
+ };
67
+
68
+ //#endregion
69
+ export { BlogGenerator };
70
+ //# sourceMappingURL=blog.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"blog.js","names":[],"sources":["../../src/generators/blog.ts"],"sourcesContent":["import type { LLMProvider } from '@contractspec/lib.contracts/integrations/providers/llm';\nimport type {\n ContentBrief,\n GeneratedContent,\n GeneratorOptions,\n} from '../types';\n\nexport class BlogGenerator {\n private readonly llm?: LLMProvider;\n private readonly model?: string;\n private readonly temperature: number;\n\n constructor(options?: GeneratorOptions) {\n this.llm = options?.llm;\n this.model = options?.model;\n this.temperature = options?.temperature ?? 0.4;\n }\n\n async generate(brief: ContentBrief): Promise<GeneratedContent> {\n if (this.llm) {\n return this.generateWithLlm(brief);\n }\n return this.generateDeterministic(brief);\n }\n\n private async generateWithLlm(\n brief: ContentBrief\n ): Promise<GeneratedContent> {\n if (!this.llm) {\n return this.generateDeterministic(brief);\n }\n const response = await this.llm.chat(\n [\n {\n role: 'system',\n content: [\n {\n type: 'text',\n text: 'You are a product marketing writer. Produce JSON with title, subtitle, intro, sections[].heading/body/bullets, outro.',\n },\n ],\n },\n {\n role: 'user',\n content: [\n {\n type: 'text',\n text: JSON.stringify({ brief }),\n },\n ],\n },\n ],\n {\n responseFormat: 'json',\n model: this.model,\n temperature: this.temperature,\n }\n );\n const jsonPart = response.message.content.find((part) => 'text' in part);\n if (jsonPart && 'text' in jsonPart) {\n return JSON.parse(jsonPart.text) as GeneratedContent;\n }\n return this.generateDeterministic(brief);\n }\n\n private generateDeterministic(brief: ContentBrief): GeneratedContent {\n const intro = `Operators like ${brief.audience.role} teams face ${brief.problems\n .slice(0, 2)\n .join(' and ')}. ${brief.title} changes that by ${brief.summary}.`;\n\n const sections = [\n {\n heading: 'Why now',\n body: this.renderWhyNow(brief),\n },\n {\n heading: 'What you get',\n body: 'A focused stack built for policy-safe automation.',\n bullets: brief.solutions,\n },\n {\n heading: 'Proof it works',\n body: 'Teams using the blueprint report measurable wins.',\n bullets: brief.metrics ?? [\n 'Launch workflows in minutes',\n 'Cut review time by 60%',\n ],\n },\n ];\n\n return {\n title: brief.title,\n subtitle: brief.summary,\n intro,\n sections,\n outro:\n brief.callToAction ??\n 'Ready to see it live? Spin up a sandbox in under 5 minutes.',\n };\n }\n\n private renderWhyNow(brief: ContentBrief): string {\n const audience = `${brief.audience.role}${brief.audience.industry ? ` in ${brief.audience.industry}` : ''}`;\n const pains = brief.problems.slice(0, 2).join('; ');\n return `${audience} teams are stuck with ${pains}. ${brief.title} delivers guardrails without slowing shipping.`;\n }\n}\n"],"mappings":";AAOA,IAAa,gBAAb,MAA2B;CACzB,AAAiB;CACjB,AAAiB;CACjB,AAAiB;CAEjB,YAAY,SAA4B;AACtC,OAAK,MAAM,SAAS;AACpB,OAAK,QAAQ,SAAS;AACtB,OAAK,cAAc,SAAS,eAAe;;CAG7C,MAAM,SAAS,OAAgD;AAC7D,MAAI,KAAK,IACP,QAAO,KAAK,gBAAgB,MAAM;AAEpC,SAAO,KAAK,sBAAsB,MAAM;;CAG1C,MAAc,gBACZ,OAC2B;AAC3B,MAAI,CAAC,KAAK,IACR,QAAO,KAAK,sBAAsB,MAAM;EA6B1C,MAAM,YA3BW,MAAM,KAAK,IAAI,KAC9B,CACE;GACE,MAAM;GACN,SAAS,CACP;IACE,MAAM;IACN,MAAM;IACP,CACF;GACF,EACD;GACE,MAAM;GACN,SAAS,CACP;IACE,MAAM;IACN,MAAM,KAAK,UAAU,EAAE,OAAO,CAAC;IAChC,CACF;GACF,CACF,EACD;GACE,gBAAgB;GAChB,OAAO,KAAK;GACZ,aAAa,KAAK;GACnB,CACF,EACyB,QAAQ,QAAQ,MAAM,SAAS,UAAU,KAAK;AACxE,MAAI,YAAY,UAAU,SACxB,QAAO,KAAK,MAAM,SAAS,KAAK;AAElC,SAAO,KAAK,sBAAsB,MAAM;;CAG1C,AAAQ,sBAAsB,OAAuC;EACnE,MAAM,QAAQ,kBAAkB,MAAM,SAAS,KAAK,cAAc,MAAM,SACrE,MAAM,GAAG,EAAE,CACX,KAAK,QAAQ,CAAC,IAAI,MAAM,MAAM,mBAAmB,MAAM,QAAQ;EAElE,MAAM,WAAW;GACf;IACE,SAAS;IACT,MAAM,KAAK,aAAa,MAAM;IAC/B;GACD;IACE,SAAS;IACT,MAAM;IACN,SAAS,MAAM;IAChB;GACD;IACE,SAAS;IACT,MAAM;IACN,SAAS,MAAM,WAAW,CACxB,+BACA,yBACD;IACF;GACF;AAED,SAAO;GACL,OAAO,MAAM;GACb,UAAU,MAAM;GAChB;GACA;GACA,OACE,MAAM,gBACN;GACH;;CAGH,AAAQ,aAAa,OAA6B;AAGhD,SAAO,GAFU,GAAG,MAAM,SAAS,OAAO,MAAM,SAAS,WAAW,OAAO,MAAM,SAAS,aAAa,KAEpF,wBADL,MAAM,SAAS,MAAM,GAAG,EAAE,CAAC,KAAK,KAAK,CACF,IAAI,MAAM,MAAM"}
@@ -0,0 +1,19 @@
1
+ import { EmailCampaignBrief, EmailDraft, GeneratorOptions } from "../types.js";
2
+
3
+ //#region src/generators/email.d.ts
4
+ declare class EmailCampaignGenerator {
5
+ private readonly llm?;
6
+ private readonly model?;
7
+ private readonly temperature;
8
+ constructor(options?: GeneratorOptions);
9
+ generate(input: EmailCampaignBrief): Promise<EmailDraft>;
10
+ private generateWithLlm;
11
+ private generateFallback;
12
+ private subjects;
13
+ private defaultPreview;
14
+ private renderBody;
15
+ private variantHook;
16
+ }
17
+ //#endregion
18
+ export { EmailCampaignGenerator };
19
+ //# sourceMappingURL=email.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"email.d.ts","names":[],"sources":["../../src/generators/email.ts"],"sourcesContent":[],"mappings":";;;cAOa,sBAAA;;EAAA,iBAAA,KAAA;EAKW,iBAAA,WAAA;EAMA,WAAA,CAAA,OAAA,CAAA,EANA,gBAMA;EAA6B,QAAA,CAAA,KAAA,EAA7B,kBAA6B,CAAA,EAAR,OAAQ,CAAA,UAAA,CAAA;EAAR,QAAA,eAAA;EAAO,QAAA,gBAAA"}
@@ -0,0 +1,102 @@
1
+ //#region src/generators/email.ts
2
+ var EmailCampaignGenerator = class {
3
+ llm;
4
+ model;
5
+ temperature;
6
+ constructor(options) {
7
+ this.llm = options?.llm;
8
+ this.model = options?.model;
9
+ this.temperature = options?.temperature ?? .6;
10
+ }
11
+ async generate(input) {
12
+ if (this.llm) {
13
+ const draft = await this.generateWithLlm(input);
14
+ if (draft) return draft;
15
+ }
16
+ return this.generateFallback(input);
17
+ }
18
+ async generateWithLlm(input) {
19
+ if (!this.llm) return null;
20
+ const jsonPart = (await this.llm.chat([{
21
+ role: "system",
22
+ content: [{
23
+ type: "text",
24
+ text: "Draft product marketing email as JSON {subject, previewText, body, cta}."
25
+ }]
26
+ }, {
27
+ role: "user",
28
+ content: [{
29
+ type: "text",
30
+ text: JSON.stringify(input)
31
+ }]
32
+ }], {
33
+ responseFormat: "json",
34
+ model: this.model,
35
+ temperature: this.temperature
36
+ })).message.content.find((chunk) => "text" in chunk);
37
+ if (!jsonPart || !("text" in jsonPart)) return null;
38
+ const parsed = JSON.parse(jsonPart.text);
39
+ if (!parsed.subject || !parsed.body || !parsed.cta) return null;
40
+ return {
41
+ subject: parsed.subject,
42
+ previewText: parsed.previewText ?? this.defaultPreview(input),
43
+ body: parsed.body,
44
+ cta: parsed.cta,
45
+ variant: input.variant
46
+ };
47
+ }
48
+ generateFallback(input) {
49
+ const { brief, variant } = input;
50
+ return {
51
+ subject: this.subjects(brief.title, variant)[0] ?? `${brief.title} update`,
52
+ previewText: this.defaultPreview(input),
53
+ body: this.renderBody(input),
54
+ cta: brief.callToAction ?? "Explore the sandbox",
55
+ variant
56
+ };
57
+ }
58
+ subjects(title, variant) {
59
+ switch (variant) {
60
+ case "announcement": return [
61
+ `Launch: ${title}`,
62
+ `${title} is live`,
63
+ `New: ${title}`
64
+ ];
65
+ case "onboarding": return [`Get started with ${title}`, `Your ${title} guide`];
66
+ case "nurture":
67
+ default: return [`How ${title} speeds ops`, `Proof ${title} works`];
68
+ }
69
+ }
70
+ defaultPreview(input) {
71
+ return `See how teams ${input.brief.metrics?.[0] ?? "ship faster without policy gaps"}.`;
72
+ }
73
+ renderBody(input) {
74
+ const { brief, variant } = input;
75
+ const greeting = "Hi there,";
76
+ const hook = this.variantHook(variant, brief);
77
+ const proof = brief.metrics?.map((metric) => `• ${metric}`).join("\n") ?? "";
78
+ return `${greeting}
79
+
80
+ ${hook}
81
+
82
+ Top reasons teams adopt ${brief.title}:
83
+ ${brief.solutions.map((solution) => `• ${solution}`).join("\n")}
84
+
85
+ ${proof}
86
+
87
+ ${brief.callToAction ?? "Spin up a sandbox"} → ${(input.cadenceDay ?? 0) + 1}
88
+ `;
89
+ }
90
+ variantHook(variant, brief) {
91
+ switch (variant) {
92
+ case "announcement": return `${brief.title} is live. ${brief.summary}`;
93
+ case "onboarding": return `Here is your next step to unlock ${brief.title}.`;
94
+ case "nurture":
95
+ default: return `Operators like ${brief.audience.role} keep asking how to automate policy checks. Here is what works.`;
96
+ }
97
+ }
98
+ };
99
+
100
+ //#endregion
101
+ export { EmailCampaignGenerator };
102
+ //# sourceMappingURL=email.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"email.js","names":[],"sources":["../../src/generators/email.ts"],"sourcesContent":["import type {\n EmailCampaignBrief,\n EmailDraft,\n GeneratorOptions,\n} from '../types';\nimport type { LLMProvider } from '@contractspec/lib.contracts/integrations/providers/llm';\n\nexport class EmailCampaignGenerator {\n private readonly llm?: LLMProvider;\n private readonly model?: string;\n private readonly temperature: number;\n\n constructor(options?: GeneratorOptions) {\n this.llm = options?.llm;\n this.model = options?.model;\n this.temperature = options?.temperature ?? 0.6;\n }\n\n async generate(input: EmailCampaignBrief): Promise<EmailDraft> {\n if (this.llm) {\n const draft = await this.generateWithLlm(input);\n if (draft) return draft;\n }\n return this.generateFallback(input);\n }\n\n private async generateWithLlm(\n input: EmailCampaignBrief\n ): Promise<EmailDraft | null> {\n if (!this.llm) return null;\n const response = await this.llm.chat(\n [\n {\n role: 'system',\n content: [\n {\n type: 'text',\n text: 'Draft product marketing email as JSON {subject, previewText, body, cta}.',\n },\n ],\n },\n {\n role: 'user',\n content: [{ type: 'text', text: JSON.stringify(input) }],\n },\n ],\n {\n responseFormat: 'json',\n model: this.model,\n temperature: this.temperature,\n }\n );\n const jsonPart = response.message.content.find((chunk) => 'text' in chunk);\n if (!jsonPart || !('text' in jsonPart)) return null;\n const parsed = JSON.parse(jsonPart.text) as Partial<EmailDraft>;\n if (!parsed.subject || !parsed.body || !parsed.cta) return null;\n return {\n subject: parsed.subject,\n previewText: parsed.previewText ?? this.defaultPreview(input),\n body: parsed.body,\n cta: parsed.cta,\n variant: input.variant,\n };\n }\n\n private generateFallback(input: EmailCampaignBrief): EmailDraft {\n const { brief, variant } = input;\n const subject =\n this.subjects(brief.title, variant)[0] ?? `${brief.title} update`;\n const previewText = this.defaultPreview(input);\n const body = this.renderBody(input);\n const cta = brief.callToAction ?? 'Explore the sandbox';\n return { subject, previewText, body, cta, variant };\n }\n\n private subjects(\n title: string,\n variant: EmailCampaignBrief['variant']\n ): string[] {\n switch (variant) {\n case 'announcement':\n return [`Launch: ${title}`, `${title} is live`, `New: ${title}`];\n case 'onboarding':\n return [`Get started with ${title}`, `Your ${title} guide`];\n case 'nurture':\n default:\n return [`How ${title} speeds ops`, `Proof ${title} works`];\n }\n }\n\n private defaultPreview(input: EmailCampaignBrief) {\n const win = input.brief.metrics?.[0] ?? 'ship faster without policy gaps';\n return `See how teams ${win}.`;\n }\n\n private renderBody(input: EmailCampaignBrief): string {\n const { brief, variant } = input;\n const greeting = 'Hi there,';\n const hook = this.variantHook(variant, brief);\n const proof =\n brief.metrics?.map((metric) => `• ${metric}`).join('\\n') ?? '';\n return `${greeting}\n\n${hook}\n\nTop reasons teams adopt ${brief.title}:\n${brief.solutions.map((solution) => `• ${solution}`).join('\\n')}\n\n${proof}\n\n${brief.callToAction ?? 'Spin up a sandbox'} → ${(input.cadenceDay ?? 0) + 1}\n`;\n }\n\n private variantHook(\n variant: EmailCampaignBrief['variant'],\n brief: EmailCampaignBrief['brief']\n ): string {\n switch (variant) {\n case 'announcement':\n return `${brief.title} is live. ${brief.summary}`;\n case 'onboarding':\n return `Here is your next step to unlock ${brief.title}.`;\n case 'nurture':\n default:\n return `Operators like ${brief.audience.role} keep asking how to automate policy checks. Here is what works.`;\n }\n }\n}\n"],"mappings":";AAOA,IAAa,yBAAb,MAAoC;CAClC,AAAiB;CACjB,AAAiB;CACjB,AAAiB;CAEjB,YAAY,SAA4B;AACtC,OAAK,MAAM,SAAS;AACpB,OAAK,QAAQ,SAAS;AACtB,OAAK,cAAc,SAAS,eAAe;;CAG7C,MAAM,SAAS,OAAgD;AAC7D,MAAI,KAAK,KAAK;GACZ,MAAM,QAAQ,MAAM,KAAK,gBAAgB,MAAM;AAC/C,OAAI,MAAO,QAAO;;AAEpB,SAAO,KAAK,iBAAiB,MAAM;;CAGrC,MAAc,gBACZ,OAC4B;AAC5B,MAAI,CAAC,KAAK,IAAK,QAAO;EAuBtB,MAAM,YAtBW,MAAM,KAAK,IAAI,KAC9B,CACE;GACE,MAAM;GACN,SAAS,CACP;IACE,MAAM;IACN,MAAM;IACP,CACF;GACF,EACD;GACE,MAAM;GACN,SAAS,CAAC;IAAE,MAAM;IAAQ,MAAM,KAAK,UAAU,MAAM;IAAE,CAAC;GACzD,CACF,EACD;GACE,gBAAgB;GAChB,OAAO,KAAK;GACZ,aAAa,KAAK;GACnB,CACF,EACyB,QAAQ,QAAQ,MAAM,UAAU,UAAU,MAAM;AAC1E,MAAI,CAAC,YAAY,EAAE,UAAU,UAAW,QAAO;EAC/C,MAAM,SAAS,KAAK,MAAM,SAAS,KAAK;AACxC,MAAI,CAAC,OAAO,WAAW,CAAC,OAAO,QAAQ,CAAC,OAAO,IAAK,QAAO;AAC3D,SAAO;GACL,SAAS,OAAO;GAChB,aAAa,OAAO,eAAe,KAAK,eAAe,MAAM;GAC7D,MAAM,OAAO;GACb,KAAK,OAAO;GACZ,SAAS,MAAM;GAChB;;CAGH,AAAQ,iBAAiB,OAAuC;EAC9D,MAAM,EAAE,OAAO,YAAY;AAM3B,SAAO;GAAE,SAJP,KAAK,SAAS,MAAM,OAAO,QAAQ,CAAC,MAAM,GAAG,MAAM,MAAM;GAIzC,aAHE,KAAK,eAAe,MAAM;GAGf,MAFlB,KAAK,WAAW,MAAM;GAEE,KADzB,MAAM,gBAAgB;GACQ;GAAS;;CAGrD,AAAQ,SACN,OACA,SACU;AACV,UAAQ,SAAR;GACE,KAAK,eACH,QAAO;IAAC,WAAW;IAAS,GAAG,MAAM;IAAW,QAAQ;IAAQ;GAClE,KAAK,aACH,QAAO,CAAC,oBAAoB,SAAS,QAAQ,MAAM,QAAQ;GAC7D,KAAK;GACL,QACE,QAAO,CAAC,OAAO,MAAM,cAAc,SAAS,MAAM,QAAQ;;;CAIhE,AAAQ,eAAe,OAA2B;AAEhD,SAAO,iBADK,MAAM,MAAM,UAAU,MAAM,kCACZ;;CAG9B,AAAQ,WAAW,OAAmC;EACpD,MAAM,EAAE,OAAO,YAAY;EAC3B,MAAM,WAAW;EACjB,MAAM,OAAO,KAAK,YAAY,SAAS,MAAM;EAC7C,MAAM,QACJ,MAAM,SAAS,KAAK,WAAW,KAAK,SAAS,CAAC,KAAK,KAAK,IAAI;AAC9D,SAAO,GAAG,SAAS;;EAErB,KAAK;;0BAEmB,MAAM,MAAM;EACpC,MAAM,UAAU,KAAK,aAAa,KAAK,WAAW,CAAC,KAAK,KAAK,CAAC;;EAE9D,MAAM;;EAEN,MAAM,gBAAgB,oBAAoB,MAAM,MAAM,cAAc,KAAK,EAAE;;;CAI3E,AAAQ,YACN,SACA,OACQ;AACR,UAAQ,SAAR;GACE,KAAK,eACH,QAAO,GAAG,MAAM,MAAM,YAAY,MAAM;GAC1C,KAAK,aACH,QAAO,oCAAoC,MAAM,MAAM;GACzD,KAAK;GACL,QACE,QAAO,kBAAkB,MAAM,SAAS,KAAK"}
@@ -0,0 +1,5 @@
1
+ import { BlogGenerator } from "./blog.js";
2
+ import { EmailCampaignGenerator } from "./email.js";
3
+ import { LandingPageCopy, LandingPageGenerator } from "./landing-page.js";
4
+ import { SocialPostGenerator } from "./social.js";
5
+ export { BlogGenerator, EmailCampaignGenerator, LandingPageCopy, LandingPageGenerator, SocialPostGenerator };
@@ -0,0 +1,6 @@
1
+ import { BlogGenerator } from "./blog.js";
2
+ import { LandingPageGenerator } from "./landing-page.js";
3
+ import { EmailCampaignGenerator } from "./email.js";
4
+ import { SocialPostGenerator } from "./social.js";
5
+
6
+ export { BlogGenerator, EmailCampaignGenerator, LandingPageGenerator, SocialPostGenerator };
@@ -0,0 +1,28 @@
1
+ import { ContentBlock, ContentBrief, GeneratorOptions } from "../types.js";
2
+
3
+ //#region src/generators/landing-page.d.ts
4
+ interface LandingPageCopy {
5
+ hero: {
6
+ eyebrow?: string;
7
+ title: string;
8
+ subtitle: string;
9
+ primaryCta: string;
10
+ secondaryCta?: string;
11
+ };
12
+ highlights: ContentBlock[];
13
+ socialProof: ContentBlock;
14
+ faq: ContentBlock[];
15
+ }
16
+ declare class LandingPageGenerator {
17
+ private readonly options?;
18
+ private readonly llm?;
19
+ private readonly model?;
20
+ constructor(options?: GeneratorOptions | undefined);
21
+ generate(brief: ContentBrief): Promise<LandingPageCopy>;
22
+ private generateWithLlm;
23
+ private generateFallback;
24
+ private buildFaq;
25
+ }
26
+ //#endregion
27
+ export { LandingPageCopy, LandingPageGenerator };
28
+ //# sourceMappingURL=landing-page.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"landing-page.d.ts","names":[],"sources":["../../src/generators/landing-page.ts"],"sourcesContent":[],"mappings":";;;UAGiB,eAAA;;IAAA,OAAA,CAAA,EAAA,MAAe;IAQlB,KAAA,EAAA,MAAA;IACC,QAAA,EAAA,MAAA;IACR,UAAA,EAAA,MAAA;IAAY,YAAA,CAAA,EAAA,MAAA;EAGN,CAAA;EAI4B,UAAA,EAT3B,YAS2B,EAAA;EAKjB,WAAA,EAbT,YAaS;EAAuB,GAAA,EAZxC,YAYwC,EAAA;;AAAD,cATjC,oBAAA,CASiC;;;;wBALL;kBAKjB,eAAe,QAAQ"}
@@ -0,0 +1,78 @@
1
+ //#region src/generators/landing-page.ts
2
+ var LandingPageGenerator = class {
3
+ llm;
4
+ model;
5
+ constructor(options) {
6
+ this.options = options;
7
+ this.llm = options?.llm;
8
+ this.model = options?.model;
9
+ }
10
+ async generate(brief) {
11
+ if (this.llm) return this.generateWithLlm(brief);
12
+ return this.generateFallback(brief);
13
+ }
14
+ async generateWithLlm(brief) {
15
+ if (!this.llm) return this.generateFallback(brief);
16
+ const part = (await this.llm.chat([{
17
+ role: "system",
18
+ content: [{
19
+ type: "text",
20
+ text: "Write JSON landing page copy with hero/highlights/socialProof/faq arrays."
21
+ }]
22
+ }, {
23
+ role: "user",
24
+ content: [{
25
+ type: "text",
26
+ text: JSON.stringify({ brief })
27
+ }]
28
+ }], {
29
+ responseFormat: "json",
30
+ model: this.model,
31
+ temperature: this.options?.temperature ?? .5
32
+ })).message.content.find((chunk) => "text" in chunk);
33
+ if (part && "text" in part) return JSON.parse(part.text);
34
+ return this.generateFallback(brief);
35
+ }
36
+ generateFallback(brief) {
37
+ return {
38
+ hero: {
39
+ eyebrow: `${brief.audience.industry ?? "Operations"} teams`,
40
+ title: brief.title,
41
+ subtitle: brief.summary,
42
+ primaryCta: brief.callToAction ?? "Launch a sandbox",
43
+ secondaryCta: "View docs"
44
+ },
45
+ highlights: brief.solutions.slice(0, 3).map((solution, index) => ({
46
+ heading: [
47
+ "Policy-safe by default",
48
+ "Auto-adapts per tenant",
49
+ "Launch-ready in days"
50
+ ][index] ?? "Key capability",
51
+ body: solution
52
+ })),
53
+ socialProof: {
54
+ heading: "Teams using ContractSpec",
55
+ body: brief.proofPoints?.join("\n") ?? "“We ship compliant workflows 5x faster while cutting ops toil in half.”"
56
+ },
57
+ faq: this.buildFaq(brief)
58
+ };
59
+ }
60
+ buildFaq(brief) {
61
+ const faqs = [{
62
+ heading: "How does this keep policies enforced?",
63
+ body: "All workflows compile from TypeScript specs and pass through PDP checks before execution, so no shadow logic slips through."
64
+ }, {
65
+ heading: "Will it fit our existing stack?",
66
+ body: "Runtime adapters plug into REST, GraphQL, or MCP. Integrations stay vendor agnostic."
67
+ }];
68
+ if (brief.complianceNotes?.length) faqs.push({
69
+ heading: "What about compliance requirements?",
70
+ body: brief.complianceNotes.join(" ")
71
+ });
72
+ return faqs;
73
+ }
74
+ };
75
+
76
+ //#endregion
77
+ export { LandingPageGenerator };
78
+ //# sourceMappingURL=landing-page.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"landing-page.js","names":[],"sources":["../../src/generators/landing-page.ts"],"sourcesContent":["import type { ContentBrief, ContentBlock, GeneratorOptions } from '../types';\nimport type { LLMProvider } from '@contractspec/lib.contracts/integrations/providers/llm';\n\nexport interface LandingPageCopy {\n hero: {\n eyebrow?: string;\n title: string;\n subtitle: string;\n primaryCta: string;\n secondaryCta?: string;\n };\n highlights: ContentBlock[];\n socialProof: ContentBlock;\n faq: ContentBlock[];\n}\n\nexport class LandingPageGenerator {\n private readonly llm?: LLMProvider;\n private readonly model?: string;\n\n constructor(private readonly options?: GeneratorOptions) {\n this.llm = options?.llm;\n this.model = options?.model;\n }\n\n async generate(brief: ContentBrief): Promise<LandingPageCopy> {\n if (this.llm) {\n return this.generateWithLlm(brief);\n }\n return this.generateFallback(brief);\n }\n\n private async generateWithLlm(brief: ContentBrief): Promise<LandingPageCopy> {\n if (!this.llm) {\n return this.generateFallback(brief);\n }\n const response = await this.llm.chat(\n [\n {\n role: 'system',\n content: [\n {\n type: 'text',\n text: 'Write JSON landing page copy with hero/highlights/socialProof/faq arrays.',\n },\n ],\n },\n {\n role: 'user',\n content: [{ type: 'text', text: JSON.stringify({ brief }) }],\n },\n ],\n {\n responseFormat: 'json',\n model: this.model,\n temperature: this.options?.temperature ?? 0.5,\n }\n );\n const part = response.message.content.find((chunk) => 'text' in chunk);\n if (part && 'text' in part) {\n return JSON.parse(part.text) as LandingPageCopy;\n }\n return this.generateFallback(brief);\n }\n\n private generateFallback(brief: ContentBrief): LandingPageCopy {\n return {\n hero: {\n eyebrow: `${brief.audience.industry ?? 'Operations'} teams`,\n title: brief.title,\n subtitle: brief.summary,\n primaryCta: brief.callToAction ?? 'Launch a sandbox',\n secondaryCta: 'View docs',\n },\n highlights: brief.solutions.slice(0, 3).map((solution, index) => ({\n heading:\n [\n 'Policy-safe by default',\n 'Auto-adapts per tenant',\n 'Launch-ready in days',\n ][index] ?? 'Key capability',\n body: solution,\n })),\n socialProof: {\n heading: 'Teams using ContractSpec',\n body:\n brief.proofPoints?.join('\\n') ??\n '“We ship compliant workflows 5x faster while cutting ops toil in half.”',\n },\n faq: this.buildFaq(brief),\n };\n }\n\n private buildFaq(brief: ContentBrief): ContentBlock[] {\n const faqs: ContentBlock[] = [\n {\n heading: 'How does this keep policies enforced?',\n body: 'All workflows compile from TypeScript specs and pass through PDP checks before execution, so no shadow logic slips through.',\n },\n {\n heading: 'Will it fit our existing stack?',\n body: 'Runtime adapters plug into REST, GraphQL, or MCP. Integrations stay vendor agnostic.',\n },\n ];\n if (brief.complianceNotes?.length) {\n faqs.push({\n heading: 'What about compliance requirements?',\n body: brief.complianceNotes.join(' '),\n });\n }\n return faqs;\n }\n}\n"],"mappings":";AAgBA,IAAa,uBAAb,MAAkC;CAChC,AAAiB;CACjB,AAAiB;CAEjB,YAAY,AAAiB,SAA4B;EAA5B;AAC3B,OAAK,MAAM,SAAS;AACpB,OAAK,QAAQ,SAAS;;CAGxB,MAAM,SAAS,OAA+C;AAC5D,MAAI,KAAK,IACP,QAAO,KAAK,gBAAgB,MAAM;AAEpC,SAAO,KAAK,iBAAiB,MAAM;;CAGrC,MAAc,gBAAgB,OAA+C;AAC3E,MAAI,CAAC,KAAK,IACR,QAAO,KAAK,iBAAiB,MAAM;EAwBrC,MAAM,QAtBW,MAAM,KAAK,IAAI,KAC9B,CACE;GACE,MAAM;GACN,SAAS,CACP;IACE,MAAM;IACN,MAAM;IACP,CACF;GACF,EACD;GACE,MAAM;GACN,SAAS,CAAC;IAAE,MAAM;IAAQ,MAAM,KAAK,UAAU,EAAE,OAAO,CAAC;IAAE,CAAC;GAC7D,CACF,EACD;GACE,gBAAgB;GAChB,OAAO,KAAK;GACZ,aAAa,KAAK,SAAS,eAAe;GAC3C,CACF,EACqB,QAAQ,QAAQ,MAAM,UAAU,UAAU,MAAM;AACtE,MAAI,QAAQ,UAAU,KACpB,QAAO,KAAK,MAAM,KAAK,KAAK;AAE9B,SAAO,KAAK,iBAAiB,MAAM;;CAGrC,AAAQ,iBAAiB,OAAsC;AAC7D,SAAO;GACL,MAAM;IACJ,SAAS,GAAG,MAAM,SAAS,YAAY,aAAa;IACpD,OAAO,MAAM;IACb,UAAU,MAAM;IAChB,YAAY,MAAM,gBAAgB;IAClC,cAAc;IACf;GACD,YAAY,MAAM,UAAU,MAAM,GAAG,EAAE,CAAC,KAAK,UAAU,WAAW;IAChE,SACE;KACE;KACA;KACA;KACD,CAAC,UAAU;IACd,MAAM;IACP,EAAE;GACH,aAAa;IACX,SAAS;IACT,MACE,MAAM,aAAa,KAAK,KAAK,IAC7B;IACH;GACD,KAAK,KAAK,SAAS,MAAM;GAC1B;;CAGH,AAAQ,SAAS,OAAqC;EACpD,MAAM,OAAuB,CAC3B;GACE,SAAS;GACT,MAAM;GACP,EACD;GACE,SAAS;GACT,MAAM;GACP,CACF;AACD,MAAI,MAAM,iBAAiB,OACzB,MAAK,KAAK;GACR,SAAS;GACT,MAAM,MAAM,gBAAgB,KAAK,IAAI;GACtC,CAAC;AAEJ,SAAO"}
@@ -0,0 +1,15 @@
1
+ import { ContentBrief, GeneratorOptions, SocialPost } from "../types.js";
2
+
3
+ //#region src/generators/social.d.ts
4
+ declare class SocialPostGenerator {
5
+ private readonly llm?;
6
+ private readonly model?;
7
+ constructor(options?: GeneratorOptions);
8
+ generate(brief: ContentBrief): Promise<SocialPost[]>;
9
+ private generateWithLlm;
10
+ private generateFallback;
11
+ private buildHashtags;
12
+ }
13
+ //#endregion
14
+ export { SocialPostGenerator };
15
+ //# sourceMappingURL=social.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"social.d.ts","names":[],"sources":["../../src/generators/social.ts"],"sourcesContent":[],"mappings":";;;cAGa,mBAAA;;EAAA,iBAAA,KAAmB;EAIR,WAAA,CAAA,OAAA,CAAA,EAAA,gBAAA;EAKA,QAAA,CAAA,KAAA,EAAA,YAAA,CAAA,EAAe,OAAf,CAAuB,UAAvB,EAAA,CAAA;EAAuB,QAAA,eAAA;EAAR,QAAA,gBAAA;EAAO,QAAA,aAAA"}
@@ -0,0 +1,75 @@
1
+ //#region src/generators/social.ts
2
+ var SocialPostGenerator = class {
3
+ llm;
4
+ model;
5
+ constructor(options) {
6
+ this.llm = options?.llm;
7
+ this.model = options?.model;
8
+ }
9
+ async generate(brief) {
10
+ if (this.llm) {
11
+ const posts = await this.generateWithLlm(brief);
12
+ if (posts.length) return posts;
13
+ }
14
+ return this.generateFallback(brief);
15
+ }
16
+ async generateWithLlm(brief) {
17
+ if (!this.llm) return [];
18
+ const part = (await this.llm.chat([{
19
+ role: "system",
20
+ content: [{
21
+ type: "text",
22
+ text: "Create JSON array of social posts for twitter/linkedin/threads with body, hashtags, cta."
23
+ }]
24
+ }, {
25
+ role: "user",
26
+ content: [{
27
+ type: "text",
28
+ text: JSON.stringify(brief)
29
+ }]
30
+ }], {
31
+ responseFormat: "json",
32
+ model: this.model
33
+ })).message.content.find((chunk) => "text" in chunk);
34
+ if (!part || !("text" in part)) return [];
35
+ return JSON.parse(part.text);
36
+ }
37
+ generateFallback(brief) {
38
+ const hashtags = this.buildHashtags(brief);
39
+ return [
40
+ {
41
+ channel: "linkedin",
42
+ body: `${brief.title}: ${brief.summary}\n${brief.problems[0]} → ${brief.solutions[0]}`,
43
+ hashtags,
44
+ cta: brief.callToAction ?? "Book a 15-min run-through"
45
+ },
46
+ {
47
+ channel: "twitter",
48
+ body: `${brief.solutions[0]} in <60s. ${brief.solutions[1] ?? ""}`.trim(),
49
+ hashtags: hashtags.slice(0, 3),
50
+ cta: "→ contractspec.io/sandbox"
51
+ },
52
+ {
53
+ channel: "threads",
54
+ body: `Ops + policy can move fast. ${brief.title} automates guardrails so teams ship daily.`,
55
+ hashtags: hashtags.slice(1, 4)
56
+ }
57
+ ];
58
+ }
59
+ buildHashtags(brief) {
60
+ const base = [
61
+ brief.audience.industry ? `#${camel(brief.audience.industry)}` : "#operations",
62
+ "#automation",
63
+ "#aiops",
64
+ "#compliance"
65
+ ];
66
+ return [...new Set(base.map((tag) => tag.replace(/\s+/g, "")))].slice(0, 5);
67
+ }
68
+ };
69
+ function camel(text) {
70
+ return text.split(/\s|-/).filter(Boolean).map((word) => word[0]?.toUpperCase() + word.slice(1)).join("");
71
+ }
72
+
73
+ //#endregion
74
+ export { SocialPostGenerator };
75
+ //# sourceMappingURL=social.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"social.js","names":[],"sources":["../../src/generators/social.ts"],"sourcesContent":["import type { ContentBrief, GeneratorOptions, SocialPost } from '../types';\nimport type { LLMProvider } from '@contractspec/lib.contracts/integrations/providers/llm';\n\nexport class SocialPostGenerator {\n private readonly llm?: LLMProvider;\n private readonly model?: string;\n\n constructor(options?: GeneratorOptions) {\n this.llm = options?.llm;\n this.model = options?.model;\n }\n\n async generate(brief: ContentBrief): Promise<SocialPost[]> {\n if (this.llm) {\n const posts = await this.generateWithLlm(brief);\n if (posts.length) return posts;\n }\n return this.generateFallback(brief);\n }\n\n private async generateWithLlm(brief: ContentBrief): Promise<SocialPost[]> {\n if (!this.llm) return [];\n const response = await this.llm.chat(\n [\n {\n role: 'system',\n content: [\n {\n type: 'text',\n text: 'Create JSON array of social posts for twitter/linkedin/threads with body, hashtags, cta.',\n },\n ],\n },\n {\n role: 'user',\n content: [{ type: 'text', text: JSON.stringify(brief) }],\n },\n ],\n { responseFormat: 'json', model: this.model }\n );\n const part = response.message.content.find((chunk) => 'text' in chunk);\n if (!part || !('text' in part)) return [];\n return JSON.parse(part.text) as SocialPost[];\n }\n\n private generateFallback(brief: ContentBrief): SocialPost[] {\n const hashtags = this.buildHashtags(brief);\n return [\n {\n channel: 'linkedin',\n body: `${brief.title}: ${brief.summary}\\n${brief.problems[0]} → ${brief.solutions[0]}`,\n hashtags,\n cta: brief.callToAction ?? 'Book a 15-min run-through',\n },\n {\n channel: 'twitter',\n body: `${brief.solutions[0]} in <60s. ${brief.solutions[1] ?? ''}`.trim(),\n hashtags: hashtags.slice(0, 3),\n cta: '→ contractspec.io/sandbox',\n },\n {\n channel: 'threads',\n body: `Ops + policy can move fast. ${brief.title} automates guardrails so teams ship daily.`,\n hashtags: hashtags.slice(1, 4),\n },\n ];\n }\n\n private buildHashtags(brief: ContentBrief): string[] {\n const base = [\n brief.audience.industry\n ? `#${camel(brief.audience.industry)}`\n : '#operations',\n '#automation',\n '#aiops',\n '#compliance',\n ];\n return [...new Set(base.map((tag) => tag.replace(/\\s+/g, '')))].slice(0, 5);\n }\n}\n\nfunction camel(text: string) {\n return text\n .split(/\\s|-/)\n .filter(Boolean)\n .map((word) => word[0]?.toUpperCase() + word.slice(1))\n .join('');\n}\n"],"mappings":";AAGA,IAAa,sBAAb,MAAiC;CAC/B,AAAiB;CACjB,AAAiB;CAEjB,YAAY,SAA4B;AACtC,OAAK,MAAM,SAAS;AACpB,OAAK,QAAQ,SAAS;;CAGxB,MAAM,SAAS,OAA4C;AACzD,MAAI,KAAK,KAAK;GACZ,MAAM,QAAQ,MAAM,KAAK,gBAAgB,MAAM;AAC/C,OAAI,MAAM,OAAQ,QAAO;;AAE3B,SAAO,KAAK,iBAAiB,MAAM;;CAGrC,MAAc,gBAAgB,OAA4C;AACxE,MAAI,CAAC,KAAK,IAAK,QAAO,EAAE;EAmBxB,MAAM,QAlBW,MAAM,KAAK,IAAI,KAC9B,CACE;GACE,MAAM;GACN,SAAS,CACP;IACE,MAAM;IACN,MAAM;IACP,CACF;GACF,EACD;GACE,MAAM;GACN,SAAS,CAAC;IAAE,MAAM;IAAQ,MAAM,KAAK,UAAU,MAAM;IAAE,CAAC;GACzD,CACF,EACD;GAAE,gBAAgB;GAAQ,OAAO,KAAK;GAAO,CAC9C,EACqB,QAAQ,QAAQ,MAAM,UAAU,UAAU,MAAM;AACtE,MAAI,CAAC,QAAQ,EAAE,UAAU,MAAO,QAAO,EAAE;AACzC,SAAO,KAAK,MAAM,KAAK,KAAK;;CAG9B,AAAQ,iBAAiB,OAAmC;EAC1D,MAAM,WAAW,KAAK,cAAc,MAAM;AAC1C,SAAO;GACL;IACE,SAAS;IACT,MAAM,GAAG,MAAM,MAAM,IAAI,MAAM,QAAQ,IAAI,MAAM,SAAS,GAAG,KAAK,MAAM,UAAU;IAClF;IACA,KAAK,MAAM,gBAAgB;IAC5B;GACD;IACE,SAAS;IACT,MAAM,GAAG,MAAM,UAAU,GAAG,YAAY,MAAM,UAAU,MAAM,KAAK,MAAM;IACzE,UAAU,SAAS,MAAM,GAAG,EAAE;IAC9B,KAAK;IACN;GACD;IACE,SAAS;IACT,MAAM,+BAA+B,MAAM,MAAM;IACjD,UAAU,SAAS,MAAM,GAAG,EAAE;IAC/B;GACF;;CAGH,AAAQ,cAAc,OAA+B;EACnD,MAAM,OAAO;GACX,MAAM,SAAS,WACX,IAAI,MAAM,MAAM,SAAS,SAAS,KAClC;GACJ;GACA;GACA;GACD;AACD,SAAO,CAAC,GAAG,IAAI,IAAI,KAAK,KAAK,QAAQ,IAAI,QAAQ,QAAQ,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,GAAG,EAAE;;;AAI/E,SAAS,MAAM,MAAc;AAC3B,QAAO,KACJ,MAAM,OAAO,CACb,OAAO,QAAQ,CACf,KAAK,SAAS,KAAK,IAAI,aAAa,GAAG,KAAK,MAAM,EAAE,CAAC,CACrD,KAAK,GAAG"}
@@ -0,0 +1,7 @@
1
+ import { AudienceProfile, ContentBlock, ContentBrief, EmailCampaignBrief, EmailDraft, GeneratedContent, GeneratorOptions, SeoMetadata, SocialPost } from "./types.js";
2
+ import { BlogGenerator } from "./generators/blog.js";
3
+ import { EmailCampaignGenerator } from "./generators/email.js";
4
+ import { LandingPageCopy, LandingPageGenerator } from "./generators/landing-page.js";
5
+ import { SocialPostGenerator } from "./generators/social.js";
6
+ import { SeoOptimizer } from "./seo/optimizer.js";
7
+ export { AudienceProfile, BlogGenerator, ContentBlock, ContentBrief, EmailCampaignBrief, EmailCampaignGenerator, EmailDraft, GeneratedContent, GeneratorOptions, LandingPageCopy, LandingPageGenerator, SeoMetadata, SeoOptimizer, SocialPost, SocialPostGenerator };
package/dist/index.js ADDED
@@ -0,0 +1,7 @@
1
+ import { BlogGenerator } from "./generators/blog.js";
2
+ import { LandingPageGenerator } from "./generators/landing-page.js";
3
+ import { EmailCampaignGenerator } from "./generators/email.js";
4
+ import { SocialPostGenerator } from "./generators/social.js";
5
+ import { SeoOptimizer } from "./seo/optimizer.js";
6
+
7
+ export { BlogGenerator, EmailCampaignGenerator, LandingPageGenerator, SeoOptimizer, SocialPostGenerator };
@@ -0,0 +1,2 @@
1
+ import { SeoOptimizer } from "./optimizer.js";
2
+ export { SeoOptimizer };
@@ -0,0 +1,3 @@
1
+ import { SeoOptimizer } from "./optimizer.js";
2
+
3
+ export { SeoOptimizer };
@@ -0,0 +1,12 @@
1
+ import { ContentBrief, SeoMetadata } from "../types.js";
2
+
3
+ //#region src/seo/optimizer.d.ts
4
+ declare class SeoOptimizer {
5
+ optimize(brief: ContentBrief): SeoMetadata;
6
+ private keywords;
7
+ private slugify;
8
+ private schema;
9
+ }
10
+ //#endregion
11
+ export { SeoOptimizer };
12
+ //# sourceMappingURL=optimizer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"optimizer.d.ts","names":[],"sources":["../../src/seo/optimizer.ts"],"sourcesContent":[],"mappings":";;;cAEa,YAAA;kBACK,eAAe;EADpB,QAAA,QAAY"}
@@ -0,0 +1,49 @@
1
+ //#region src/seo/optimizer.ts
2
+ var SeoOptimizer = class {
3
+ optimize(brief) {
4
+ const keywords = this.keywords(brief);
5
+ const metaTitle = `${brief.title} | ContractSpec`;
6
+ const metaDescription = `${brief.summary} — built for ${brief.audience.role}${brief.audience.industry ? ` in ${brief.audience.industry}` : ""}.`;
7
+ return {
8
+ metaTitle,
9
+ metaDescription,
10
+ keywords,
11
+ slug: this.slugify(brief.title),
12
+ schemaMarkup: this.schema(brief, metaDescription, keywords)
13
+ };
14
+ }
15
+ keywords(brief) {
16
+ const base = [
17
+ brief.title,
18
+ ...brief.problems,
19
+ ...brief.solutions
20
+ ];
21
+ return [...new Set(base.flatMap((entry) => entry.toLowerCase().split(/\s+/)))].filter((word) => word.length > 3).slice(0, 12);
22
+ }
23
+ slugify(text) {
24
+ return text.toLowerCase().replace(/[^a-z0-9]+/g, "-").replace(/^-+|-+$/g, "");
25
+ }
26
+ schema(brief, description, keywords) {
27
+ return {
28
+ "@context": "https://schema.org",
29
+ "@type": "Product",
30
+ name: brief.title,
31
+ description,
32
+ audience: {
33
+ "@type": "Audience",
34
+ audienceType: brief.audience.role,
35
+ industry: brief.audience.industry
36
+ },
37
+ offers: {
38
+ "@type": "Offer",
39
+ description: brief.callToAction ?? "Start building with ContractSpec"
40
+ },
41
+ keywords: keywords.join(", "),
42
+ citation: brief.references?.map((ref) => ref.url)
43
+ };
44
+ }
45
+ };
46
+
47
+ //#endregion
48
+ export { SeoOptimizer };
49
+ //# sourceMappingURL=optimizer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"optimizer.js","names":[],"sources":["../../src/seo/optimizer.ts"],"sourcesContent":["import type { ContentBrief, SeoMetadata } from '../types';\n\nexport class SeoOptimizer {\n optimize(brief: ContentBrief): SeoMetadata {\n const keywords = this.keywords(brief);\n const metaTitle = `${brief.title} | ContractSpec`;\n const metaDescription = `${brief.summary} — built for ${brief.audience.role}${brief.audience.industry ? ` in ${brief.audience.industry}` : ''}.`;\n const slug = this.slugify(brief.title);\n const schemaMarkup = this.schema(brief, metaDescription, keywords);\n return { metaTitle, metaDescription, keywords, slug, schemaMarkup };\n }\n\n private keywords(brief: ContentBrief): string[] {\n const base = [brief.title, ...brief.problems, ...brief.solutions];\n return [\n ...new Set(base.flatMap((entry) => entry.toLowerCase().split(/\\s+/))),\n ]\n .filter((word) => word.length > 3)\n .slice(0, 12);\n }\n\n private slugify(text: string): string {\n return text\n .toLowerCase()\n .replace(/[^a-z0-9]+/g, '-')\n .replace(/^-+|-+$/g, '');\n }\n\n private schema(\n brief: ContentBrief,\n description: string,\n keywords: string[]\n ): Record<string, unknown> {\n return {\n '@context': 'https://schema.org',\n '@type': 'Product',\n name: brief.title,\n description,\n audience: {\n '@type': 'Audience',\n audienceType: brief.audience.role,\n industry: brief.audience.industry,\n },\n offers: {\n '@type': 'Offer',\n description: brief.callToAction ?? 'Start building with ContractSpec',\n },\n keywords: keywords.join(', '),\n citation: brief.references?.map((ref) => ref.url),\n };\n }\n}\n"],"mappings":";AAEA,IAAa,eAAb,MAA0B;CACxB,SAAS,OAAkC;EACzC,MAAM,WAAW,KAAK,SAAS,MAAM;EACrC,MAAM,YAAY,GAAG,MAAM,MAAM;EACjC,MAAM,kBAAkB,GAAG,MAAM,QAAQ,eAAe,MAAM,SAAS,OAAO,MAAM,SAAS,WAAW,OAAO,MAAM,SAAS,aAAa,GAAG;AAG9I,SAAO;GAAE;GAAW;GAAiB;GAAU,MAFlC,KAAK,QAAQ,MAAM,MAAM;GAEe,cADhC,KAAK,OAAO,OAAO,iBAAiB,SAAS;GACC;;CAGrE,AAAQ,SAAS,OAA+B;EAC9C,MAAM,OAAO;GAAC,MAAM;GAAO,GAAG,MAAM;GAAU,GAAG,MAAM;GAAU;AACjE,SAAO,CACL,GAAG,IAAI,IAAI,KAAK,SAAS,UAAU,MAAM,aAAa,CAAC,MAAM,MAAM,CAAC,CAAC,CACtE,CACE,QAAQ,SAAS,KAAK,SAAS,EAAE,CACjC,MAAM,GAAG,GAAG;;CAGjB,AAAQ,QAAQ,MAAsB;AACpC,SAAO,KACJ,aAAa,CACb,QAAQ,eAAe,IAAI,CAC3B,QAAQ,YAAY,GAAG;;CAG5B,AAAQ,OACN,OACA,aACA,UACyB;AACzB,SAAO;GACL,YAAY;GACZ,SAAS;GACT,MAAM,MAAM;GACZ;GACA,UAAU;IACR,SAAS;IACT,cAAc,MAAM,SAAS;IAC7B,UAAU,MAAM,SAAS;IAC1B;GACD,QAAQ;IACN,SAAS;IACT,aAAa,MAAM,gBAAgB;IACpC;GACD,UAAU,SAAS,KAAK,KAAK;GAC7B,UAAU,MAAM,YAAY,KAAK,QAAQ,IAAI,IAAI;GAClD"}
@@ -0,0 +1,72 @@
1
+ import { LLMProvider } from "@contractspec/lib.contracts/integrations/providers/llm";
2
+
3
+ //#region src/types.d.ts
4
+ interface AudienceProfile {
5
+ role: string;
6
+ industry?: string;
7
+ region?: string;
8
+ maturity?: 'early' | 'scaleup' | 'enterprise';
9
+ painPoints?: string[];
10
+ }
11
+ interface ContentBrief {
12
+ title: string;
13
+ summary: string;
14
+ problems: string[];
15
+ solutions: string[];
16
+ metrics?: string[];
17
+ proofPoints?: string[];
18
+ complianceNotes?: string[];
19
+ audience: AudienceProfile;
20
+ callToAction?: string;
21
+ references?: {
22
+ label: string;
23
+ url: string;
24
+ }[];
25
+ }
26
+ interface ContentBlock {
27
+ heading: string;
28
+ body: string;
29
+ bullets?: string[];
30
+ cta?: string;
31
+ }
32
+ interface GeneratedContent {
33
+ title: string;
34
+ subtitle?: string;
35
+ intro: string;
36
+ sections: ContentBlock[];
37
+ outro?: string;
38
+ }
39
+ interface GeneratorOptions {
40
+ llm?: LLMProvider;
41
+ model?: string;
42
+ temperature?: number;
43
+ }
44
+ interface EmailCampaignBrief {
45
+ brief: ContentBrief;
46
+ variant: 'announcement' | 'onboarding' | 'nurture';
47
+ cadenceDay?: number;
48
+ }
49
+ interface EmailDraft {
50
+ subject: string;
51
+ previewText: string;
52
+ body: string;
53
+ cta: string;
54
+ variant: EmailCampaignBrief['variant'];
55
+ }
56
+ interface SocialPost {
57
+ channel: 'twitter' | 'linkedin' | 'threads';
58
+ body: string;
59
+ hashtags: string[];
60
+ cta?: string;
61
+ link?: string;
62
+ }
63
+ interface SeoMetadata {
64
+ metaTitle: string;
65
+ metaDescription: string;
66
+ keywords: string[];
67
+ slug: string;
68
+ schemaMarkup: Record<string, unknown>;
69
+ }
70
+ //#endregion
71
+ export { AudienceProfile, ContentBlock, ContentBrief, EmailCampaignBrief, EmailDraft, GeneratedContent, GeneratorOptions, SeoMetadata, SocialPost };
72
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","names":[],"sources":["../src/types.ts"],"sourcesContent":[],"mappings":";;;UAEiB,eAAA;;EAAA,QAAA,CAAA,EAAA,MAAA;EAQA,MAAA,CAAA,EAAA,MAAY;EAaZ,QAAA,CAAA,EAAA,OAAY,GAAA,SAAA,GAAA,YAAA;EAOZ,UAAA,CAAA,EAAA,MAAgB,EAAA;AAQjC;AAMiB,UAlCA,YAAA,CAkCkB;EAMlB,KAAA,EAAA,MAAU;EAQV,OAAA,EAAA,MAAU;EAQV,QAAA,EAAA,MAAW,EAAA;;;;;YAhDhB;;;;;;;UAKK,YAAA;;;;;;UAOA,gBAAA;;;;YAIL;;;UAIK,gBAAA;QACT;;;;UAKS,kBAAA;SACR;;;;UAKQ,UAAA;;;;;WAKN;;UAGM,UAAA;;;;;;;UAQA,WAAA;;;;;gBAKD"}
package/dist/types.js ADDED
File without changes
package/package.json ADDED
@@ -0,0 +1,76 @@
1
+ {
2
+ "name": "@contractspec/lib.content-gen",
3
+ "version": "0.0.0-canary-20260113162409",
4
+ "description": "AI-powered content generation for blog, email, and social",
5
+ "keywords": [
6
+ "contractspec",
7
+ "content",
8
+ "ai",
9
+ "generation",
10
+ "seo",
11
+ "typescript"
12
+ ],
13
+ "type": "module",
14
+ "types": "./dist/index.d.ts",
15
+ "files": [
16
+ "dist",
17
+ "README.md"
18
+ ],
19
+ "scripts": {
20
+ "publish:pkg": "bun publish --tolerate-republish --ignore-scripts --verbose",
21
+ "publish:pkg:canary": "bun publish:pkg --tag canary",
22
+ "build": "bun build:types && bun build:bundle",
23
+ "build:bundle": "tsdown",
24
+ "build:types": "tsc --noEmit",
25
+ "dev": "bun build:bundle --watch",
26
+ "clean": "rimraf dist .turbo",
27
+ "lint": "bun lint:fix",
28
+ "lint:fix": "eslint src --fix",
29
+ "lint:check": "eslint src",
30
+ "test": "bun test"
31
+ },
32
+ "dependencies": {
33
+ "@contractspec/lib.contracts": "0.0.0-canary-20260113162409"
34
+ },
35
+ "devDependencies": {
36
+ "@contractspec/tool.tsdown": "0.0.0-canary-20260113162409",
37
+ "@contractspec/tool.typescript": "0.0.0-canary-20260113162409",
38
+ "tsdown": "^0.19.0",
39
+ "typescript": "^5.9.3"
40
+ },
41
+ "exports": {
42
+ ".": "./dist/index.js",
43
+ "./generators": "./dist/generators/index.js",
44
+ "./generators/blog": "./dist/generators/blog.js",
45
+ "./generators/email": "./dist/generators/email.js",
46
+ "./generators/landing-page": "./dist/generators/landing-page.js",
47
+ "./generators/social": "./dist/generators/social.js",
48
+ "./seo": "./dist/seo/index.js",
49
+ "./seo/optimizer": "./dist/seo/optimizer.js",
50
+ "./types": "./dist/types.js",
51
+ "./*": "./*"
52
+ },
53
+ "publishConfig": {
54
+ "access": "public",
55
+ "exports": {
56
+ ".": "./dist/index.js",
57
+ "./generators": "./dist/generators/index.js",
58
+ "./generators/blog": "./dist/generators/blog.js",
59
+ "./generators/email": "./dist/generators/email.js",
60
+ "./generators/landing-page": "./dist/generators/landing-page.js",
61
+ "./generators/social": "./dist/generators/social.js",
62
+ "./seo": "./dist/seo/index.js",
63
+ "./seo/optimizer": "./dist/seo/optimizer.js",
64
+ "./types": "./dist/types.js",
65
+ "./*": "./*"
66
+ },
67
+ "registry": "https://registry.npmjs.org/"
68
+ },
69
+ "license": "MIT",
70
+ "repository": {
71
+ "type": "git",
72
+ "url": "https://github.com/lssm-tech/contractspec.git",
73
+ "directory": "packages/libs/content-gen"
74
+ },
75
+ "homepage": "https://contractspec.io"
76
+ }