@npow/ghostwriter 0.1.0

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 (113) hide show
  1. package/dist/commands/connect.d.ts +30 -0
  2. package/dist/commands/connect.d.ts.map +1 -0
  3. package/dist/commands/connect.js +442 -0
  4. package/dist/commands/connect.js.map +1 -0
  5. package/dist/commands/create/index.d.ts +6 -0
  6. package/dist/commands/create/index.d.ts.map +1 -0
  7. package/dist/commands/create/index.js +89 -0
  8. package/dist/commands/create/index.js.map +1 -0
  9. package/dist/commands/create/prompts/example-article.d.ts +3 -0
  10. package/dist/commands/create/prompts/example-article.d.ts.map +1 -0
  11. package/dist/commands/create/prompts/example-article.js +32 -0
  12. package/dist/commands/create/prompts/example-article.js.map +1 -0
  13. package/dist/commands/create/prompts/intent-parsing.d.ts +3 -0
  14. package/dist/commands/create/prompts/intent-parsing.d.ts.map +1 -0
  15. package/dist/commands/create/prompts/intent-parsing.js +45 -0
  16. package/dist/commands/create/prompts/intent-parsing.js.map +1 -0
  17. package/dist/commands/create/prompts/page-content.d.ts +3 -0
  18. package/dist/commands/create/prompts/page-content.d.ts.map +1 -0
  19. package/dist/commands/create/prompts/page-content.js +56 -0
  20. package/dist/commands/create/prompts/page-content.js.map +1 -0
  21. package/dist/commands/create/prompts/source-discovery.d.ts +3 -0
  22. package/dist/commands/create/prompts/source-discovery.d.ts.map +1 -0
  23. package/dist/commands/create/prompts/source-discovery.js +46 -0
  24. package/dist/commands/create/prompts/source-discovery.js.map +1 -0
  25. package/dist/commands/create/prompts/voice-generation.d.ts +3 -0
  26. package/dist/commands/create/prompts/voice-generation.d.ts.map +1 -0
  27. package/dist/commands/create/prompts/voice-generation.js +43 -0
  28. package/dist/commands/create/prompts/voice-generation.js.map +1 -0
  29. package/dist/commands/create/steps/assemble-config.d.ts +5 -0
  30. package/dist/commands/create/steps/assemble-config.d.ts.map +1 -0
  31. package/dist/commands/create/steps/assemble-config.js +82 -0
  32. package/dist/commands/create/steps/assemble-config.js.map +1 -0
  33. package/dist/commands/create/steps/discover-sources.d.ts +5 -0
  34. package/dist/commands/create/steps/discover-sources.d.ts.map +1 -0
  35. package/dist/commands/create/steps/discover-sources.js +53 -0
  36. package/dist/commands/create/steps/discover-sources.js.map +1 -0
  37. package/dist/commands/create/steps/fingerprint-style.d.ts +5 -0
  38. package/dist/commands/create/steps/fingerprint-style.d.ts.map +1 -0
  39. package/dist/commands/create/steps/fingerprint-style.js +46 -0
  40. package/dist/commands/create/steps/fingerprint-style.js.map +1 -0
  41. package/dist/commands/create/steps/generate-example.d.ts +4 -0
  42. package/dist/commands/create/steps/generate-example.d.ts.map +1 -0
  43. package/dist/commands/create/steps/generate-example.js +23 -0
  44. package/dist/commands/create/steps/generate-example.js.map +1 -0
  45. package/dist/commands/create/steps/generate-voice.d.ts +4 -0
  46. package/dist/commands/create/steps/generate-voice.d.ts.map +1 -0
  47. package/dist/commands/create/steps/generate-voice.js +32 -0
  48. package/dist/commands/create/steps/generate-voice.js.map +1 -0
  49. package/dist/commands/create/steps/parse-intent.d.ts +4 -0
  50. package/dist/commands/create/steps/parse-intent.d.ts.map +1 -0
  51. package/dist/commands/create/steps/parse-intent.js +16 -0
  52. package/dist/commands/create/steps/parse-intent.js.map +1 -0
  53. package/dist/commands/create/steps/resolve-connection.d.ts +5 -0
  54. package/dist/commands/create/steps/resolve-connection.d.ts.map +1 -0
  55. package/dist/commands/create/steps/resolve-connection.js +100 -0
  56. package/dist/commands/create/steps/resolve-connection.js.map +1 -0
  57. package/dist/commands/create/steps/resolve-schedule.d.ts +13 -0
  58. package/dist/commands/create/steps/resolve-schedule.d.ts.map +1 -0
  59. package/dist/commands/create/steps/resolve-schedule.js +70 -0
  60. package/dist/commands/create/steps/resolve-schedule.js.map +1 -0
  61. package/dist/commands/create/steps/setup-site.d.ts +5 -0
  62. package/dist/commands/create/steps/setup-site.d.ts.map +1 -0
  63. package/dist/commands/create/steps/setup-site.js +44 -0
  64. package/dist/commands/create/steps/setup-site.js.map +1 -0
  65. package/dist/commands/create/steps/validate-and-summary.d.ts +4 -0
  66. package/dist/commands/create/steps/validate-and-summary.d.ts.map +1 -0
  67. package/dist/commands/create/steps/validate-and-summary.js +60 -0
  68. package/dist/commands/create/steps/validate-and-summary.js.map +1 -0
  69. package/dist/commands/create/steps/validate-site-access.d.ts +4 -0
  70. package/dist/commands/create/steps/validate-site-access.d.ts.map +1 -0
  71. package/dist/commands/create/steps/validate-site-access.js +25 -0
  72. package/dist/commands/create/steps/validate-site-access.js.map +1 -0
  73. package/dist/commands/create/types.d.ts +72 -0
  74. package/dist/commands/create/types.d.ts.map +1 -0
  75. package/dist/commands/create/types.js +2 -0
  76. package/dist/commands/create/types.js.map +1 -0
  77. package/dist/commands/fingerprint.d.ts +4 -0
  78. package/dist/commands/fingerprint.d.ts.map +1 -0
  79. package/dist/commands/fingerprint.js +31 -0
  80. package/dist/commands/fingerprint.js.map +1 -0
  81. package/dist/commands/init.d.ts +2 -0
  82. package/dist/commands/init.d.ts.map +1 -0
  83. package/dist/commands/init.js +117 -0
  84. package/dist/commands/init.js.map +1 -0
  85. package/dist/commands/list.d.ts +2 -0
  86. package/dist/commands/list.d.ts.map +1 -0
  87. package/dist/commands/list.js +36 -0
  88. package/dist/commands/list.js.map +1 -0
  89. package/dist/commands/logs.d.ts +6 -0
  90. package/dist/commands/logs.d.ts.map +1 -0
  91. package/dist/commands/logs.js +49 -0
  92. package/dist/commands/logs.js.map +1 -0
  93. package/dist/commands/patterns.d.ts +6 -0
  94. package/dist/commands/patterns.d.ts.map +1 -0
  95. package/dist/commands/patterns.js +51 -0
  96. package/dist/commands/patterns.js.map +1 -0
  97. package/dist/commands/run.d.ts +6 -0
  98. package/dist/commands/run.d.ts.map +1 -0
  99. package/dist/commands/run.js +139 -0
  100. package/dist/commands/run.js.map +1 -0
  101. package/dist/commands/status.d.ts +2 -0
  102. package/dist/commands/status.d.ts.map +1 -0
  103. package/dist/commands/status.js +66 -0
  104. package/dist/commands/status.js.map +1 -0
  105. package/dist/commands/validate.d.ts +2 -0
  106. package/dist/commands/validate.d.ts.map +1 -0
  107. package/dist/commands/validate.js +83 -0
  108. package/dist/commands/validate.js.map +1 -0
  109. package/dist/index.d.ts +3 -0
  110. package/dist/index.d.ts.map +1 -0
  111. package/dist/index.js +76 -0
  112. package/dist/index.js.map +1 -0
  113. package/package.json +38 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"parse-intent.js","sourceRoot":"","sources":["../../../../src/commands/create/steps/parse-intent.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,WAAW,EAAE,MAAM,+BAA+B,CAAC;AAE5D,OAAO,EACL,qBAAqB,EACrB,wBAAwB,GACzB,MAAM,8BAA8B,CAAC;AAEtC,MAAM,CAAC,KAAK,UAAU,WAAW,CAC/B,GAAkB,EAClB,OAAY;IAEZ,OAAO,CAAC,KAAK,CAAC,oCAAoC,CAAC,CAAC;IAEpD,MAAM,EAAE,IAAI,EAAE,IAAI,EAAE,GAAG,MAAM,WAAW,CACtC,QAAQ,EACR,qBAAqB,EACrB,wBAAwB,CAAC,GAAG,CAAC,cAAc,CAAC,EAC5C,EAAE,WAAW,EAAE,GAAG,EAAE,CACrB,CAAC;IAEF,GAAG,CAAC,SAAS,IAAI,IAAI,CAAC;IAEtB,uCAAuC;IACvC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS;SAC5B,WAAW,EAAE;SACb,OAAO,CAAC,aAAa,EAAE,GAAG,CAAC;SAC3B,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC;SACnB,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;IAEzB,OAAO,CAAC,OAAO,CACb,YAAY,IAAI,CAAC,WAAW,MAAM,IAAI,CAAC,WAAW,OAAO,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAC7E,CAAC;IAEF,OAAO,IAAI,CAAC;AACd,CAAC"}
@@ -0,0 +1,5 @@
1
+ import type { Ora } from "ora";
2
+ import { type ConnectionEntry } from "@ghostwriter/core";
3
+ import type { CreateContext } from "../types.js";
4
+ export declare function resolveConnection(ctx: CreateContext, spinner: Ora): Promise<ConnectionEntry>;
5
+ //# sourceMappingURL=resolve-connection.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolve-connection.d.ts","sourceRoot":"","sources":["../../../../src/commands/create/steps/resolve-connection.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,KAAK,CAAC;AAG/B,OAAO,EAIL,KAAK,eAAe,EACrB,MAAM,mBAAmB,CAAC;AAC3B,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAGjD,wBAAsB,iBAAiB,CACrC,GAAG,EAAE,aAAa,EAClB,OAAO,EAAE,GAAG,GACX,OAAO,CAAC,eAAe,CAAC,CAiH1B"}
@@ -0,0 +1,100 @@
1
+ import chalk from "chalk";
2
+ import { input, password, select } from "@inquirer/prompts";
3
+ import { getConnectionsForPlatform, getConnection, saveConnection, } from "@ghostwriter/core";
4
+ import { connectWordPressCom } from "../../connect.js";
5
+ export async function resolveConnection(ctx, spinner) {
6
+ spinner.start("Resolving WordPress connection...");
7
+ const intent = ctx.intent;
8
+ // If intent specifies a connectionId, look it up (try both platforms)
9
+ if (intent.connectionId) {
10
+ const conn = (await getConnection(intent.connectionId, "wordpress")) ??
11
+ (await getConnection(intent.connectionId, "wordpress-com"));
12
+ if (conn) {
13
+ spinner.succeed(`Using existing connection: ${conn.id} (${conn.url})`);
14
+ return conn;
15
+ }
16
+ spinner.warn(`Connection "${intent.connectionId}" not found, searching for alternatives...`);
17
+ }
18
+ // Find existing WordPress connections (both self-hosted and WordPress.com)
19
+ const wpSelfHosted = await getConnectionsForPlatform("wordpress");
20
+ const wpCom = await getConnectionsForPlatform("wordpress-com");
21
+ const wpConnections = [...wpSelfHosted, ...wpCom];
22
+ // If there's a site URL from intent, try to match it
23
+ if (intent.siteUrl && wpConnections.length > 0) {
24
+ const normalized = intent.siteUrl.replace(/^https?:\/\//, "").replace(/\/$/, "");
25
+ const match = wpConnections.find((c) => {
26
+ const connUrl = (c.url ?? "").replace(/^https?:\/\//, "").replace(/\/$/, "");
27
+ return connUrl === normalized || connUrl.includes(normalized);
28
+ });
29
+ if (match) {
30
+ spinner.succeed(`Matched connection: ${match.id} (${match.url})`);
31
+ return match;
32
+ }
33
+ }
34
+ // Use first available connection
35
+ if (wpConnections.length > 0) {
36
+ const conn = wpConnections[0];
37
+ spinner.succeed(`Using connection: ${conn.id} (${conn.url})`);
38
+ return conn;
39
+ }
40
+ // No existing connection — create one inline
41
+ spinner.stop();
42
+ console.log(chalk.yellow("\n No WordPress connection found. Let's set one up.\n"));
43
+ const method = await select({
44
+ message: "How do you want to connect?",
45
+ choices: [
46
+ {
47
+ value: "wordpress-com",
48
+ name: "WordPress.com (OAuth) — easiest, opens browser to authorize",
49
+ },
50
+ {
51
+ value: "wordpress",
52
+ name: "Self-hosted WordPress (Application Password)",
53
+ },
54
+ ],
55
+ });
56
+ if (method === "wordpress-com") {
57
+ const result = await connectWordPressCom();
58
+ if (!result) {
59
+ throw new Error("WordPress.com connection was not completed");
60
+ }
61
+ return result;
62
+ }
63
+ // Self-hosted flow (unchanged)
64
+ const siteUrl = intent.siteUrl
65
+ ? `https://${intent.siteUrl.replace(/^https?:\/\//, "")}`
66
+ : await input({
67
+ message: "WordPress site URL:",
68
+ validate: (val) => {
69
+ try {
70
+ new URL(val);
71
+ return true;
72
+ }
73
+ catch {
74
+ return "Enter a valid URL";
75
+ }
76
+ },
77
+ });
78
+ const baseUrl = siteUrl.replace(/\/$/, "");
79
+ const username = await input({ message: "WordPress username:" });
80
+ const appPassword = await password({
81
+ message: "Application password:",
82
+ mask: "*",
83
+ });
84
+ const connectionId = intent.channelId + "-wp";
85
+ const conn = {
86
+ id: connectionId,
87
+ platform: "wordpress",
88
+ url: baseUrl,
89
+ credentials: {
90
+ url: baseUrl,
91
+ username,
92
+ password: appPassword.trim(),
93
+ },
94
+ createdAt: new Date().toISOString(),
95
+ };
96
+ await saveConnection(conn);
97
+ console.log(chalk.green(` Saved connection "${connectionId}"\n`));
98
+ return conn;
99
+ }
100
+ //# sourceMappingURL=resolve-connection.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolve-connection.js","sourceRoot":"","sources":["../../../../src/commands/create/steps/resolve-connection.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,13 @@
1
+ import type { Ora } from "ora";
2
+ import type { CreateContext } from "../types.js";
3
+ interface ResolvedSchedule {
4
+ cron: string;
5
+ timezone: string;
6
+ }
7
+ /**
8
+ * Resolve schedule from intent to cron expression.
9
+ * Pure function — no LLM call needed.
10
+ */
11
+ export declare function resolveSchedule(ctx: CreateContext, spinner: Ora): ResolvedSchedule;
12
+ export {};
13
+ //# sourceMappingURL=resolve-schedule.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolve-schedule.d.ts","sourceRoot":"","sources":["../../../../src/commands/create/steps/resolve-schedule.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,KAAK,CAAC;AAC/B,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAEjD,UAAU,gBAAgB;IACxB,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,MAAM,CAAC;CAClB;AAmBD;;;GAGG;AACH,wBAAgB,eAAe,CAC7B,GAAG,EAAE,aAAa,EAClB,OAAO,EAAE,GAAG,GACX,gBAAgB,CAqDlB"}
@@ -0,0 +1,70 @@
1
+ const DAY_MAP = {
2
+ monday: "MON",
3
+ tuesday: "TUE",
4
+ wednesday: "WED",
5
+ thursday: "THU",
6
+ friday: "FRI",
7
+ saturday: "SAT",
8
+ sunday: "SUN",
9
+ mon: "MON",
10
+ tue: "TUE",
11
+ wed: "WED",
12
+ thu: "THU",
13
+ fri: "FRI",
14
+ sat: "SAT",
15
+ sun: "SUN",
16
+ };
17
+ /**
18
+ * Resolve schedule from intent to cron expression.
19
+ * Pure function — no LLM call needed.
20
+ */
21
+ export function resolveSchedule(ctx, spinner) {
22
+ spinner.start("Resolving schedule...");
23
+ const schedule = ctx.intent?.schedule;
24
+ const timezone = schedule?.timezone ?? "America/New_York";
25
+ // Parse time (default 9:00 for daily, 10:00 for weekly)
26
+ let hour = 9;
27
+ let minute = 0;
28
+ if (schedule?.time) {
29
+ const timeMatch = schedule.time.match(/(\d{1,2})(?::(\d{2}))?\s*(am|pm)?/i);
30
+ if (timeMatch) {
31
+ hour = parseInt(timeMatch[1], 10);
32
+ minute = parseInt(timeMatch[2] ?? "0", 10);
33
+ if (timeMatch[3]?.toLowerCase() === "pm" && hour < 12)
34
+ hour += 12;
35
+ if (timeMatch[3]?.toLowerCase() === "am" && hour === 12)
36
+ hour = 0;
37
+ }
38
+ }
39
+ let cron;
40
+ const frequency = schedule?.frequency ?? "weekly";
41
+ switch (frequency) {
42
+ case "daily":
43
+ cron = `${minute} ${hour} * * *`;
44
+ break;
45
+ case "weekly": {
46
+ const day = schedule?.dayOfWeek
47
+ ? DAY_MAP[schedule.dayOfWeek.toLowerCase()] ?? "SAT"
48
+ : "SAT";
49
+ hour = schedule?.time ? hour : 10; // Default to 10am for weekly
50
+ cron = `${minute} ${hour} * * ${day}`;
51
+ break;
52
+ }
53
+ case "biweekly": {
54
+ // Approximate: run on 1st and 15th
55
+ hour = schedule?.time ? hour : 10;
56
+ cron = `${minute} ${hour} 1,15 * *`;
57
+ break;
58
+ }
59
+ case "monthly": {
60
+ hour = schedule?.time ? hour : 10;
61
+ cron = `${minute} ${hour} 1 * *`;
62
+ break;
63
+ }
64
+ default:
65
+ cron = `${minute} ${hour} * * SAT`;
66
+ }
67
+ spinner.succeed(`Schedule: ${cron} (${timezone})`);
68
+ return { cron, timezone };
69
+ }
70
+ //# sourceMappingURL=resolve-schedule.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolve-schedule.js","sourceRoot":"","sources":["../../../../src/commands/create/steps/resolve-schedule.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ import type { Ora } from "ora";
2
+ import { type SiteSetupResult } from "@ghostwriter/site-setup";
3
+ import type { CreateContext } from "../types.js";
4
+ export declare function setupSite(ctx: CreateContext, spinner: Ora): Promise<SiteSetupResult>;
5
+ //# sourceMappingURL=setup-site.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"setup-site.d.ts","sourceRoot":"","sources":["../../../../src/commands/create/steps/setup-site.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,KAAK,CAAC;AAE/B,OAAO,EAAwC,KAAK,eAAe,EAAE,MAAM,yBAAyB,CAAC;AACrG,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AA4BjD,wBAAsB,SAAS,CAC7B,GAAG,EAAE,aAAa,EAClB,OAAO,EAAE,GAAG,GACX,OAAO,CAAC,eAAe,CAAC,CAgE1B"}
@@ -0,0 +1,44 @@
1
+ import { callLlmJson } from "@ghostwriter/content-pipeline";
2
+ import { executeSiteSetup } from "@ghostwriter/site-setup";
3
+ import { PAGE_CONTENT_SYSTEM, buildPageContentPrompt, } from "../prompts/page-content.js";
4
+ export async function setupSite(ctx, spinner) {
5
+ spinner.start("Planning site setup...");
6
+ const intent = ctx.intent;
7
+ const voice = ctx.voice;
8
+ // Use LLM to generate site setup content
9
+ const { data, cost } = await callLlmJson("sonnet", PAGE_CONTENT_SYSTEM, buildPageContentPrompt(intent.channelName, intent.topic.domain, intent.topic.focus, intent.toneDescription, voice.persona), { temperature: 0.5 });
10
+ ctx.totalCost += cost;
11
+ if (ctx.options.dryRun) {
12
+ spinner.succeed(`Site plan: ${data.categories.length} categories, ${data.tags.length} tags, ` +
13
+ `${data.pages.length} pages (dry run — not applying)`);
14
+ return {
15
+ categories: data.categories.map((c) => ({ name: c.name, id: 0, created: false })),
16
+ tags: data.tags.map((t) => ({ name: t.name, id: 0, created: false })),
17
+ pages: data.pages.map((p) => ({ title: p.title, id: 0, url: "" })),
18
+ settings: { applied: false, changes: [] },
19
+ menus: data.menus.map((m) => ({ name: m.name, id: 0 })),
20
+ errors: [],
21
+ };
22
+ }
23
+ // Build the site setup plan
24
+ const siteId = intent.siteUrl
25
+ ?? ctx.connection?.url?.replace(/^https?:\/\//, "")
26
+ ?? intent.channelId;
27
+ const plan = {
28
+ platform: "wordpress-com",
29
+ siteId,
30
+ siteIdentity: data.siteIdentity,
31
+ categories: data.categories,
32
+ tags: data.tags,
33
+ pages: data.pages,
34
+ menus: data.menus,
35
+ };
36
+ spinner.text = "Configuring WordPress site...";
37
+ const result = await executeSiteSetup(plan, ctx.connection);
38
+ const created = result.categories.filter((c) => c.created).length;
39
+ const errors = result.errors.length;
40
+ spinner.succeed(`Site configured: ${created} categories created, ${result.pages.length} pages, ${result.menus.length} menus` +
41
+ (errors > 0 ? ` (${errors} errors)` : ""));
42
+ return result;
43
+ }
44
+ //# sourceMappingURL=setup-site.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"setup-site.js","sourceRoot":"","sources":["../../../../src/commands/create/steps/setup-site.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { Ora } from "ora";
2
+ import type { CreateContext } from "../types.js";
3
+ export declare function validateAndSummary(ctx: CreateContext, spinner: Ora): Promise<void>;
4
+ //# sourceMappingURL=validate-and-summary.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-and-summary.d.ts","sourceRoot":"","sources":["../../../../src/commands/create/steps/validate-and-summary.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,KAAK,CAAC;AAG/B,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAEjD,wBAAsB,kBAAkB,CACtC,GAAG,EAAE,aAAa,EAClB,OAAO,EAAE,GAAG,GACX,OAAO,CAAC,IAAI,CAAC,CA2Ef"}
@@ -0,0 +1,60 @@
1
+ import chalk from "chalk";
2
+ import { ChannelConfigSchema } from "@ghostwriter/core";
3
+ export async function validateAndSummary(ctx, spinner) {
4
+ spinner.start("Validating configuration...");
5
+ const config = ctx.config;
6
+ // Validate the config against the schema
7
+ const result = ChannelConfigSchema.safeParse(config);
8
+ if (!result.success) {
9
+ const errors = result.error.issues.map((i) => ` ${i.path.join(".")}: ${i.message}`);
10
+ spinner.fail("Config validation failed:");
11
+ console.log(chalk.red(errors.join("\n")));
12
+ return;
13
+ }
14
+ spinner.succeed("Config validated successfully");
15
+ // Summary
16
+ console.log("");
17
+ console.log(chalk.blue.bold(" Channel Created"));
18
+ console.log(chalk.blue(" " + "─".repeat(50)));
19
+ console.log("");
20
+ console.log(` ${chalk.dim("Name:")} ${config.name}`);
21
+ console.log(` ${chalk.dim("ID:")} ${config.id}`);
22
+ console.log(` ${chalk.dim("Type:")} ${config.contentType}`);
23
+ console.log(` ${chalk.dim("Topic:")} ${config.topic.focus}`);
24
+ console.log(` ${chalk.dim("Voice:")} ${config.voice.name} (${config.voice.tone})`);
25
+ console.log(` ${chalk.dim("Sources:")} ${config.dataSources.length} data source(s)`);
26
+ console.log(` ${chalk.dim("Schedule:")} ${config.schedule.cron} (${config.schedule.timezone})`);
27
+ console.log(` ${chalk.dim("Word count:")} ${config.targetWordCount}`);
28
+ if (ctx.siteResult) {
29
+ const sr = ctx.siteResult;
30
+ console.log("");
31
+ console.log(` ${chalk.dim("Site setup:")}`);
32
+ if (sr.categories.length > 0) {
33
+ console.log(` Categories: ${sr.categories.map((c) => c.name).join(", ")}`);
34
+ }
35
+ if (sr.pages.length > 0) {
36
+ console.log(` Pages: ${sr.pages.map((p) => p.title).join(", ")}`);
37
+ }
38
+ if (sr.errors.length > 0) {
39
+ console.log(chalk.yellow(` Errors: ${sr.errors.length}`));
40
+ for (const err of sr.errors) {
41
+ console.log(chalk.yellow(` - ${err}`));
42
+ }
43
+ }
44
+ }
45
+ console.log("");
46
+ console.log(` ${chalk.dim("LLM cost:")} $${ctx.totalCost.toFixed(4)}`);
47
+ console.log("");
48
+ if (ctx.options.dryRun) {
49
+ console.log(chalk.yellow(" Dry run complete — no files were written."));
50
+ console.log(chalk.dim(` Run without --dry-run to create the channel.\n`));
51
+ }
52
+ else {
53
+ console.log(chalk.green(" Next steps:"));
54
+ console.log(chalk.dim(` ghostwriter run ${config.id} --dry-run # Test the pipeline`));
55
+ console.log(chalk.dim(` ghostwriter validate ${config.id} # Validate config + API keys`));
56
+ console.log(chalk.dim(` ghostwriter run ${config.id} # Run for real`));
57
+ console.log("");
58
+ }
59
+ }
60
+ //# sourceMappingURL=validate-and-summary.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-and-summary.js","sourceRoot":"","sources":["../../../../src/commands/create/steps/validate-and-summary.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { Ora } from "ora";
2
+ import type { CreateContext } from "../types.js";
3
+ export declare function validateSiteAccess(ctx: CreateContext, spinner: Ora): Promise<void>;
4
+ //# sourceMappingURL=validate-site-access.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-site-access.d.ts","sourceRoot":"","sources":["../../../../src/commands/create/steps/validate-site-access.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,KAAK,CAAC;AAE/B,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AASjD,wBAAsB,kBAAkB,CACtC,GAAG,EAAE,aAAa,EAClB,OAAO,EAAE,GAAG,GACX,OAAO,CAAC,IAAI,CAAC,CA8Bf"}
@@ -0,0 +1,25 @@
1
+ import { WpComClient } from "@ghostwriter/site-setup";
2
+ export async function validateSiteAccess(ctx, spinner) {
3
+ const conn = ctx.connection;
4
+ const siteLabel = conn.url ?? conn.id;
5
+ spinner.start(`Verifying access to ${siteLabel}...`);
6
+ const client = WpComClient.fromConnection(conn);
7
+ let user;
8
+ try {
9
+ user = await client.get("/users/me?context=edit");
10
+ }
11
+ catch (err) {
12
+ const msg = err instanceof Error ? err.message : String(err);
13
+ if (msg.includes("401") || msg.includes("403")) {
14
+ throw new Error(`Could not access ${siteLabel} — check your credentials (${conn.id})`);
15
+ }
16
+ throw new Error(`Could not access ${siteLabel}: ${msg}`);
17
+ }
18
+ if (!user.capabilities?.publish_posts) {
19
+ const role = user.roles?.[0] ?? "unknown role";
20
+ throw new Error(`User "${user.name}" (${role}) does not have publish permission on ${siteLabel}`);
21
+ }
22
+ const role = user.roles?.[0] ?? "unknown role";
23
+ spinner.succeed(`Verified access as "${user.name}" (${role})`);
24
+ }
25
+ //# sourceMappingURL=validate-site-access.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-site-access.js","sourceRoot":"","sources":["../../../../src/commands/create/steps/validate-site-access.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,WAAW,EAAE,MAAM,yBAAyB,CAAC;AAUtD,MAAM,CAAC,KAAK,UAAU,kBAAkB,CACtC,GAAkB,EAClB,OAAY;IAEZ,MAAM,IAAI,GAAG,GAAG,CAAC,UAAW,CAAC;IAC7B,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC;IAEtC,OAAO,CAAC,KAAK,CAAC,uBAAuB,SAAS,KAAK,CAAC,CAAC;IAErD,MAAM,MAAM,GAAG,WAAW,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;IAEhD,IAAI,IAAY,CAAC;IACjB,IAAI,CAAC;QACH,IAAI,GAAG,MAAM,MAAM,CAAC,GAAG,CAAS,wBAAwB,CAAC,CAAC;IAC5D,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,MAAM,GAAG,GAAG,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QAC7D,IAAI,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC;YAC/C,MAAM,IAAI,KAAK,CACb,oBAAoB,SAAS,8BAA8B,IAAI,CAAC,EAAE,GAAG,CACtE,CAAC;QACJ,CAAC;QACD,MAAM,IAAI,KAAK,CAAC,oBAAoB,SAAS,KAAK,GAAG,EAAE,CAAC,CAAC;IAC3D,CAAC;IAED,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,aAAa,EAAE,CAAC;QACtC,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,cAAc,CAAC;QAC/C,MAAM,IAAI,KAAK,CACb,SAAS,IAAI,CAAC,IAAI,MAAM,IAAI,yCAAyC,SAAS,EAAE,CACjF,CAAC;IACJ,CAAC;IAED,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,cAAc,CAAC;IAC/C,OAAO,CAAC,OAAO,CAAC,uBAAuB,IAAI,CAAC,IAAI,MAAM,IAAI,GAAG,CAAC,CAAC;AACjE,CAAC"}
@@ -0,0 +1,72 @@
1
+ import type { ChannelConfig, ConnectionEntry, ContentType, DataSource } from "@ghostwriter/core";
2
+ import type { StyleProfile } from "@ghostwriter/style-fingerprint";
3
+ import type { SiteSetupResult } from "@ghostwriter/site-setup";
4
+ export interface ParsedIntent {
5
+ channelId: string;
6
+ channelName: string;
7
+ contentType: ContentType;
8
+ topic: {
9
+ domain: string;
10
+ focus: string;
11
+ keywords: string[];
12
+ constraints?: string;
13
+ };
14
+ toneDescription: string;
15
+ styleReferences: string[];
16
+ publishPlatform: "wordpress-com";
17
+ siteUrl?: string;
18
+ connectionId?: string;
19
+ schedule?: {
20
+ frequency: "daily" | "weekly" | "biweekly" | "monthly";
21
+ dayOfWeek?: string;
22
+ time?: string;
23
+ timezone?: string;
24
+ };
25
+ targetWordCount?: number;
26
+ }
27
+ export interface GeneratedVoice {
28
+ name: string;
29
+ persona: string;
30
+ age?: number;
31
+ backstory?: string;
32
+ opinions?: string[];
33
+ verbalTics?: string[];
34
+ vocabulary: {
35
+ preferred: string[];
36
+ forbidden: string[];
37
+ };
38
+ tone: "conversational" | "professional" | "academic" | "casual" | "authoritative" | "humorous" | "warm";
39
+ }
40
+ export interface DiscoveredSource {
41
+ type: "rss" | "api";
42
+ url: string;
43
+ name: string;
44
+ description: string;
45
+ requiresApiKey?: boolean;
46
+ apiKeyEnvVar?: string;
47
+ }
48
+ export interface DiscoveredSources {
49
+ sources: DiscoveredSource[];
50
+ }
51
+ export interface CreateOptions {
52
+ interactive: boolean;
53
+ siteSetup: boolean;
54
+ dryRun: boolean;
55
+ }
56
+ export interface CreateContext {
57
+ rawDescription: string;
58
+ options: CreateOptions;
59
+ intent?: ParsedIntent;
60
+ connection?: ConnectionEntry;
61
+ styleProfile?: StyleProfile;
62
+ voice?: GeneratedVoice;
63
+ dataSources?: DataSource[];
64
+ schedule?: {
65
+ cron: string;
66
+ timezone: string;
67
+ };
68
+ config?: ChannelConfig;
69
+ siteResult?: SiteSetupResult;
70
+ totalCost: number;
71
+ }
72
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../src/commands/create/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,aAAa,EAAE,eAAe,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AACjG,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,gCAAgC,CAAC;AACnE,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,yBAAyB,CAAC;AAI/D,MAAM,WAAW,YAAY;IAC3B,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,WAAW,CAAC;IACzB,KAAK,EAAE;QACL,MAAM,EAAE,MAAM,CAAC;QACf,KAAK,EAAE,MAAM,CAAC;QACd,QAAQ,EAAE,MAAM,EAAE,CAAC;QACnB,WAAW,CAAC,EAAE,MAAM,CAAC;KACtB,CAAC;IACF,eAAe,EAAE,MAAM,CAAC;IACxB,eAAe,EAAE,MAAM,EAAE,CAAC;IAC1B,eAAe,EAAE,eAAe,CAAC;IACjC,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,EAAE;QACT,SAAS,EAAE,OAAO,GAAG,QAAQ,GAAG,UAAU,GAAG,SAAS,CAAC;QACvD,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,IAAI,CAAC,EAAE,MAAM,CAAC;QACd,QAAQ,CAAC,EAAE,MAAM,CAAC;KACnB,CAAC;IACF,eAAe,CAAC,EAAE,MAAM,CAAC;CAC1B;AAID,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,QAAQ,CAAC,EAAE,MAAM,EAAE,CAAC;IACpB,UAAU,CAAC,EAAE,MAAM,EAAE,CAAC;IACtB,UAAU,EAAE;QACV,SAAS,EAAE,MAAM,EAAE,CAAC;QACpB,SAAS,EAAE,MAAM,EAAE,CAAC;KACrB,CAAC;IACF,IAAI,EAAE,gBAAgB,GAAG,cAAc,GAAG,UAAU,GAAG,QAAQ,GAAG,eAAe,GAAG,UAAU,GAAG,MAAM,CAAC;CACzG;AAID,MAAM,WAAW,gBAAgB;IAC/B,IAAI,EAAE,KAAK,GAAG,KAAK,CAAC;IACpB,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,cAAc,CAAC,EAAE,OAAO,CAAC;IACzB,YAAY,CAAC,EAAE,MAAM,CAAC;CACvB;AAED,MAAM,WAAW,iBAAiB;IAChC,OAAO,EAAE,gBAAgB,EAAE,CAAC;CAC7B;AAID,MAAM,WAAW,aAAa;IAC5B,WAAW,EAAE,OAAO,CAAC;IACrB,SAAS,EAAE,OAAO,CAAC;IACnB,MAAM,EAAE,OAAO,CAAC;CACjB;AAED,MAAM,WAAW,aAAa;IAC5B,cAAc,EAAE,MAAM,CAAC;IACvB,OAAO,EAAE,aAAa,CAAC;IACvB,MAAM,CAAC,EAAE,YAAY,CAAC;IACtB,UAAU,CAAC,EAAE,eAAe,CAAC;IAC7B,YAAY,CAAC,EAAE,YAAY,CAAC;IAC5B,KAAK,CAAC,EAAE,cAAc,CAAC;IACvB,WAAW,CAAC,EAAE,UAAU,EAAE,CAAC;IAC3B,QAAQ,CAAC,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,QAAQ,EAAE,MAAM,CAAA;KAAE,CAAC;IAC9C,MAAM,CAAC,EAAE,aAAa,CAAC;IACvB,UAAU,CAAC,EAAE,eAAe,CAAC;IAC7B,SAAS,EAAE,MAAM,CAAC;CACnB"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/commands/create/types.ts"],"names":[],"mappings":""}
@@ -0,0 +1,4 @@
1
+ export declare function fingerprintCommand(input: string, options: {
2
+ output?: string;
3
+ }): Promise<void>;
4
+ //# sourceMappingURL=fingerprint.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fingerprint.d.ts","sourceRoot":"","sources":["../../src/commands/fingerprint.ts"],"names":[],"mappings":"AASA,wBAAsB,kBAAkB,CACtC,KAAK,EAAE,MAAM,EACb,OAAO,EAAE;IAAE,MAAM,CAAC,EAAE,MAAM,CAAA;CAAE,GAC3B,OAAO,CAAC,IAAI,CAAC,CA0Bf"}
@@ -0,0 +1,31 @@
1
+ import { readFileSync } from "node:fs";
2
+ import { resolve } from "node:path";
3
+ import { analyzeStyle, analyzeUrl, formatStyleProfile, } from "@ghostwriter/style-fingerprint";
4
+ export async function fingerprintCommand(input, options) {
5
+ const mode = (options.output ?? "prompt");
6
+ const isUrl = input.startsWith("http://") || input.startsWith("https://");
7
+ try {
8
+ let profile;
9
+ if (isUrl) {
10
+ console.log(`Fetching ${input}...`);
11
+ profile = await analyzeUrl(input);
12
+ }
13
+ else {
14
+ const filePath = resolve(input);
15
+ const text = readFileSync(filePath, "utf-8");
16
+ profile = analyzeStyle([text]);
17
+ }
18
+ if (mode === "json") {
19
+ console.log(JSON.stringify(profile, null, 2));
20
+ }
21
+ else {
22
+ console.log(formatStyleProfile(profile, mode));
23
+ }
24
+ }
25
+ catch (err) {
26
+ const message = err instanceof Error ? err.message : String(err);
27
+ console.error(`Error: ${message}`);
28
+ process.exitCode = 1;
29
+ }
30
+ }
31
+ //# sourceMappingURL=fingerprint.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fingerprint.js","sourceRoot":"","sources":["../../src/commands/fingerprint.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AACvC,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EACL,YAAY,EACZ,UAAU,EACV,kBAAkB,GACnB,MAAM,gCAAgC,CAAC;AAGxC,MAAM,CAAC,KAAK,UAAU,kBAAkB,CACtC,KAAa,EACb,OAA4B;IAE5B,MAAM,IAAI,GAAG,CAAC,OAAO,CAAC,MAAM,IAAI,QAAQ,CAAwB,CAAC;IACjE,MAAM,KAAK,GAAG,KAAK,CAAC,UAAU,CAAC,SAAS,CAAC,IAAI,KAAK,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;IAE1E,IAAI,CAAC;QACH,IAAI,OAAO,CAAC;QAEZ,IAAI,KAAK,EAAE,CAAC;YACV,OAAO,CAAC,GAAG,CAAC,YAAY,KAAK,KAAK,CAAC,CAAC;YACpC,OAAO,GAAG,MAAM,UAAU,CAAC,KAAK,CAAC,CAAC;QACpC,CAAC;aAAM,CAAC;YACN,MAAM,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC;YAChC,MAAM,IAAI,GAAG,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;YAC7C,OAAO,GAAG,YAAY,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QACjC,CAAC;QAED,IAAI,IAAI,KAAK,MAAM,EAAE,CAAC;YACpB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAChD,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC,OAAO,EAAE,IAAkB,CAAC,CAAC,CAAC;QAC/D,CAAC;IACH,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,MAAM,OAAO,GAAG,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QACjE,OAAO,CAAC,KAAK,CAAC,UAAU,OAAO,EAAE,CAAC,CAAC;QACnC,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;IACvB,CAAC;AACH,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function initCommand(channelName: string): Promise<void>;
2
+ //# sourceMappingURL=init.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"init.d.ts","sourceRoot":"","sources":["../../src/commands/init.ts"],"names":[],"mappings":"AA4FA,wBAAsB,WAAW,CAAC,WAAW,EAAE,MAAM,iBAwCpD"}