adaria-ai 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.
- package/LICENSE +21 -0
- package/README.md +21 -0
- package/apps.example.yaml +65 -0
- package/dist/agent/audit.d.ts +16 -0
- package/dist/agent/audit.d.ts.map +1 -0
- package/dist/agent/audit.js +42 -0
- package/dist/agent/audit.js.map +1 -0
- package/dist/agent/claude.d.ts +62 -0
- package/dist/agent/claude.d.ts.map +1 -0
- package/dist/agent/claude.js +297 -0
- package/dist/agent/claude.js.map +1 -0
- package/dist/agent/conversation-summary.d.ts +29 -0
- package/dist/agent/conversation-summary.d.ts.map +1 -0
- package/dist/agent/conversation-summary.js +221 -0
- package/dist/agent/conversation-summary.js.map +1 -0
- package/dist/agent/core.d.ts +81 -0
- package/dist/agent/core.d.ts.map +1 -0
- package/dist/agent/core.js +527 -0
- package/dist/agent/core.js.map +1 -0
- package/dist/agent/mcp-launcher.d.ts +42 -0
- package/dist/agent/mcp-launcher.d.ts.map +1 -0
- package/dist/agent/mcp-launcher.js +38 -0
- package/dist/agent/mcp-launcher.js.map +1 -0
- package/dist/agent/mcp-manager.d.ts +81 -0
- package/dist/agent/mcp-manager.d.ts.map +1 -0
- package/dist/agent/mcp-manager.js +136 -0
- package/dist/agent/mcp-manager.js.map +1 -0
- package/dist/agent/memory.d.ts +10 -0
- package/dist/agent/memory.d.ts.map +1 -0
- package/dist/agent/memory.js +95 -0
- package/dist/agent/memory.js.map +1 -0
- package/dist/agent/safety.d.ts +45 -0
- package/dist/agent/safety.d.ts.map +1 -0
- package/dist/agent/safety.js +71 -0
- package/dist/agent/safety.js.map +1 -0
- package/dist/agent/session.d.ts +27 -0
- package/dist/agent/session.d.ts.map +1 -0
- package/dist/agent/session.js +124 -0
- package/dist/agent/session.js.map +1 -0
- package/dist/agent/tool-descriptions.d.ts +8 -0
- package/dist/agent/tool-descriptions.d.ts.map +1 -0
- package/dist/agent/tool-descriptions.js +26 -0
- package/dist/agent/tool-descriptions.js.map +1 -0
- package/dist/cli/analyze.d.ts +8 -0
- package/dist/cli/analyze.d.ts.map +1 -0
- package/dist/cli/analyze.js +114 -0
- package/dist/cli/analyze.js.map +1 -0
- package/dist/cli/daemon.d.ts +2 -0
- package/dist/cli/daemon.d.ts.map +1 -0
- package/dist/cli/daemon.js +91 -0
- package/dist/cli/daemon.js.map +1 -0
- package/dist/cli/doctor.d.ts +2 -0
- package/dist/cli/doctor.d.ts.map +1 -0
- package/dist/cli/doctor.js +198 -0
- package/dist/cli/doctor.js.map +1 -0
- package/dist/cli/init.d.ts +3 -0
- package/dist/cli/init.d.ts.map +1 -0
- package/dist/cli/init.js +459 -0
- package/dist/cli/init.js.map +1 -0
- package/dist/cli/logs.d.ts +4 -0
- package/dist/cli/logs.d.ts.map +1 -0
- package/dist/cli/logs.js +50 -0
- package/dist/cli/logs.js.map +1 -0
- package/dist/cli/monitor-cmd.d.ts +11 -0
- package/dist/cli/monitor-cmd.d.ts.map +1 -0
- package/dist/cli/monitor-cmd.js +59 -0
- package/dist/cli/monitor-cmd.js.map +1 -0
- package/dist/cli/start.d.ts +11 -0
- package/dist/cli/start.d.ts.map +1 -0
- package/dist/cli/start.js +103 -0
- package/dist/cli/start.js.map +1 -0
- package/dist/cli/status.d.ts +9 -0
- package/dist/cli/status.d.ts.map +1 -0
- package/dist/cli/status.js +49 -0
- package/dist/cli/status.js.map +1 -0
- package/dist/cli/stop.d.ts +2 -0
- package/dist/cli/stop.d.ts.map +1 -0
- package/dist/cli/stop.js +34 -0
- package/dist/cli/stop.js.map +1 -0
- package/dist/collectors/appstore.d.ts +51 -0
- package/dist/collectors/appstore.d.ts.map +1 -0
- package/dist/collectors/appstore.js +166 -0
- package/dist/collectors/appstore.js.map +1 -0
- package/dist/collectors/arden-tts.d.ts +60 -0
- package/dist/collectors/arden-tts.d.ts.map +1 -0
- package/dist/collectors/arden-tts.js +83 -0
- package/dist/collectors/arden-tts.js.map +1 -0
- package/dist/collectors/asomobile.d.ts +37 -0
- package/dist/collectors/asomobile.d.ts.map +1 -0
- package/dist/collectors/asomobile.js +88 -0
- package/dist/collectors/asomobile.js.map +1 -0
- package/dist/collectors/eodin-blog.d.ts +90 -0
- package/dist/collectors/eodin-blog.d.ts.map +1 -0
- package/dist/collectors/eodin-blog.js +238 -0
- package/dist/collectors/eodin-blog.js.map +1 -0
- package/dist/collectors/eodin-sdk.d.ts +60 -0
- package/dist/collectors/eodin-sdk.d.ts.map +1 -0
- package/dist/collectors/eodin-sdk.js +112 -0
- package/dist/collectors/eodin-sdk.js.map +1 -0
- package/dist/collectors/fridgify-recipes.d.ts +65 -0
- package/dist/collectors/fridgify-recipes.d.ts.map +1 -0
- package/dist/collectors/fridgify-recipes.js +111 -0
- package/dist/collectors/fridgify-recipes.js.map +1 -0
- package/dist/collectors/playstore.d.ts +46 -0
- package/dist/collectors/playstore.d.ts.map +1 -0
- package/dist/collectors/playstore.js +140 -0
- package/dist/collectors/playstore.js.map +1 -0
- package/dist/collectors/youtube.d.ts +44 -0
- package/dist/collectors/youtube.d.ts.map +1 -0
- package/dist/collectors/youtube.js +107 -0
- package/dist/collectors/youtube.js.map +1 -0
- package/dist/config/apps-schema.d.ts +94 -0
- package/dist/config/apps-schema.d.ts.map +1 -0
- package/dist/config/apps-schema.js +66 -0
- package/dist/config/apps-schema.js.map +1 -0
- package/dist/config/keychain.d.ts +14 -0
- package/dist/config/keychain.d.ts.map +1 -0
- package/dist/config/keychain.js +89 -0
- package/dist/config/keychain.js.map +1 -0
- package/dist/config/load-apps.d.ts +16 -0
- package/dist/config/load-apps.d.ts.map +1 -0
- package/dist/config/load-apps.js +38 -0
- package/dist/config/load-apps.js.map +1 -0
- package/dist/config/schema.d.ts +306 -0
- package/dist/config/schema.d.ts.map +1 -0
- package/dist/config/schema.js +220 -0
- package/dist/config/schema.js.map +1 -0
- package/dist/config/store.d.ts +38 -0
- package/dist/config/store.d.ts.map +1 -0
- package/dist/config/store.js +180 -0
- package/dist/config/store.js.map +1 -0
- package/dist/db/queries.d.ts +304 -0
- package/dist/db/queries.d.ts.map +1 -0
- package/dist/db/queries.js +327 -0
- package/dist/db/queries.js.map +1 -0
- package/dist/db/schema.d.ts +15 -0
- package/dist/db/schema.d.ts.map +1 -0
- package/dist/db/schema.js +252 -0
- package/dist/db/schema.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +86 -0
- package/dist/index.js.map +1 -0
- package/dist/messenger/adapter.d.ts +63 -0
- package/dist/messenger/adapter.d.ts.map +1 -0
- package/dist/messenger/adapter.js +7 -0
- package/dist/messenger/adapter.js.map +1 -0
- package/dist/messenger/factory.d.ts +12 -0
- package/dist/messenger/factory.d.ts.map +1 -0
- package/dist/messenger/factory.js +9 -0
- package/dist/messenger/factory.js.map +1 -0
- package/dist/messenger/slack.d.ts +30 -0
- package/dist/messenger/slack.d.ts.map +1 -0
- package/dist/messenger/slack.js +309 -0
- package/dist/messenger/slack.js.map +1 -0
- package/dist/messenger/split.d.ts +17 -0
- package/dist/messenger/split.d.ts.map +1 -0
- package/dist/messenger/split.js +56 -0
- package/dist/messenger/split.js.map +1 -0
- package/dist/orchestrator/dashboard.d.ts +67 -0
- package/dist/orchestrator/dashboard.d.ts.map +1 -0
- package/dist/orchestrator/dashboard.js +113 -0
- package/dist/orchestrator/dashboard.js.map +1 -0
- package/dist/orchestrator/monitor.d.ts +37 -0
- package/dist/orchestrator/monitor.d.ts.map +1 -0
- package/dist/orchestrator/monitor.js +236 -0
- package/dist/orchestrator/monitor.js.map +1 -0
- package/dist/orchestrator/types.d.ts +82 -0
- package/dist/orchestrator/types.d.ts.map +1 -0
- package/dist/orchestrator/types.js +12 -0
- package/dist/orchestrator/types.js.map +1 -0
- package/dist/orchestrator/weekly.d.ts +66 -0
- package/dist/orchestrator/weekly.d.ts.map +1 -0
- package/dist/orchestrator/weekly.js +376 -0
- package/dist/orchestrator/weekly.js.map +1 -0
- package/dist/prompts/loader.d.ts +18 -0
- package/dist/prompts/loader.d.ts.map +1 -0
- package/dist/prompts/loader.js +28 -0
- package/dist/prompts/loader.js.map +1 -0
- package/dist/security/auth.d.ts +14 -0
- package/dist/security/auth.d.ts.map +1 -0
- package/dist/security/auth.js +14 -0
- package/dist/security/auth.js.map +1 -0
- package/dist/security/prompt-guard.d.ts +21 -0
- package/dist/security/prompt-guard.d.ts.map +1 -0
- package/dist/security/prompt-guard.js +54 -0
- package/dist/security/prompt-guard.js.map +1 -0
- package/dist/skills/aso.d.ts +60 -0
- package/dist/skills/aso.d.ts.map +1 -0
- package/dist/skills/aso.js +322 -0
- package/dist/skills/aso.js.map +1 -0
- package/dist/skills/content.d.ts +25 -0
- package/dist/skills/content.d.ts.map +1 -0
- package/dist/skills/content.js +90 -0
- package/dist/skills/content.js.map +1 -0
- package/dist/skills/index.d.ts +65 -0
- package/dist/skills/index.d.ts.map +1 -0
- package/dist/skills/index.js +90 -0
- package/dist/skills/index.js.map +1 -0
- package/dist/skills/onboarding.d.ts +58 -0
- package/dist/skills/onboarding.d.ts.map +1 -0
- package/dist/skills/onboarding.js +274 -0
- package/dist/skills/onboarding.js.map +1 -0
- package/dist/skills/registry.d.ts +24 -0
- package/dist/skills/registry.d.ts.map +1 -0
- package/dist/skills/registry.js +66 -0
- package/dist/skills/registry.js.map +1 -0
- package/dist/skills/review.d.ts +33 -0
- package/dist/skills/review.d.ts.map +1 -0
- package/dist/skills/review.js +236 -0
- package/dist/skills/review.js.map +1 -0
- package/dist/skills/sdk-request.d.ts +30 -0
- package/dist/skills/sdk-request.d.ts.map +1 -0
- package/dist/skills/sdk-request.js +72 -0
- package/dist/skills/sdk-request.js.map +1 -0
- package/dist/skills/seo-blog.d.ts +64 -0
- package/dist/skills/seo-blog.d.ts.map +1 -0
- package/dist/skills/seo-blog.js +268 -0
- package/dist/skills/seo-blog.js.map +1 -0
- package/dist/skills/short-form.d.ts +28 -0
- package/dist/skills/short-form.d.ts.map +1 -0
- package/dist/skills/short-form.js +121 -0
- package/dist/skills/short-form.js.map +1 -0
- package/dist/skills/social-publish.d.ts +32 -0
- package/dist/skills/social-publish.d.ts.map +1 -0
- package/dist/skills/social-publish.js +133 -0
- package/dist/skills/social-publish.js.map +1 -0
- package/dist/social/base.d.ts +47 -0
- package/dist/social/base.d.ts.map +1 -0
- package/dist/social/base.js +26 -0
- package/dist/social/base.js.map +1 -0
- package/dist/social/facebook.d.ts +27 -0
- package/dist/social/facebook.d.ts.map +1 -0
- package/dist/social/facebook.js +166 -0
- package/dist/social/facebook.js.map +1 -0
- package/dist/social/factory.d.ts +26 -0
- package/dist/social/factory.d.ts.map +1 -0
- package/dist/social/factory.js +32 -0
- package/dist/social/factory.js.map +1 -0
- package/dist/social/linkedin.d.ts +26 -0
- package/dist/social/linkedin.d.ts.map +1 -0
- package/dist/social/linkedin.js +190 -0
- package/dist/social/linkedin.js.map +1 -0
- package/dist/social/threads.d.ts +21 -0
- package/dist/social/threads.d.ts.map +1 -0
- package/dist/social/threads.js +122 -0
- package/dist/social/threads.js.map +1 -0
- package/dist/social/tiktok.d.ts +23 -0
- package/dist/social/tiktok.d.ts.map +1 -0
- package/dist/social/tiktok.js +110 -0
- package/dist/social/tiktok.js.map +1 -0
- package/dist/social/twitter.d.ts +30 -0
- package/dist/social/twitter.d.ts.map +1 -0
- package/dist/social/twitter.js +189 -0
- package/dist/social/twitter.js.map +1 -0
- package/dist/social/youtube.d.ts +21 -0
- package/dist/social/youtube.d.ts.map +1 -0
- package/dist/social/youtube.js +108 -0
- package/dist/social/youtube.js.map +1 -0
- package/dist/tools/app-info.d.ts +7 -0
- package/dist/tools/app-info.d.ts.map +1 -0
- package/dist/tools/app-info.js +53 -0
- package/dist/tools/app-info.js.map +1 -0
- package/dist/tools/collector-fetch.d.ts +11 -0
- package/dist/tools/collector-fetch.d.ts.map +1 -0
- package/dist/tools/collector-fetch.js +101 -0
- package/dist/tools/collector-fetch.js.map +1 -0
- package/dist/tools/db-query.d.ts +29 -0
- package/dist/tools/db-query.d.ts.map +1 -0
- package/dist/tools/db-query.js +159 -0
- package/dist/tools/db-query.js.map +1 -0
- package/dist/tools/skill-result.d.ts +8 -0
- package/dist/tools/skill-result.d.ts.map +1 -0
- package/dist/tools/skill-result.js +63 -0
- package/dist/tools/skill-result.js.map +1 -0
- package/dist/tools/tool-host.d.ts +12 -0
- package/dist/tools/tool-host.d.ts.map +1 -0
- package/dist/tools/tool-host.js +124 -0
- package/dist/tools/tool-host.js.map +1 -0
- package/dist/types/collectors.d.ts +198 -0
- package/dist/types/collectors.d.ts.map +1 -0
- package/dist/types/collectors.js +28 -0
- package/dist/types/collectors.js.map +1 -0
- package/dist/types/skill.d.ts +60 -0
- package/dist/types/skill.d.ts.map +1 -0
- package/dist/types/skill.js +9 -0
- package/dist/types/skill.js.map +1 -0
- package/dist/utils/circuit-breaker.d.ts +26 -0
- package/dist/utils/circuit-breaker.d.ts.map +1 -0
- package/dist/utils/circuit-breaker.js +67 -0
- package/dist/utils/circuit-breaker.js.map +1 -0
- package/dist/utils/errors.d.ts +44 -0
- package/dist/utils/errors.d.ts.map +1 -0
- package/dist/utils/errors.js +75 -0
- package/dist/utils/errors.js.map +1 -0
- package/dist/utils/escape.d.ts +11 -0
- package/dist/utils/escape.d.ts.map +1 -0
- package/dist/utils/escape.js +19 -0
- package/dist/utils/escape.js.map +1 -0
- package/dist/utils/logger.d.ts +19 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +93 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/parse-json.d.ts +13 -0
- package/dist/utils/parse-json.d.ts.map +1 -0
- package/dist/utils/parse-json.js +61 -0
- package/dist/utils/parse-json.js.map +1 -0
- package/dist/utils/paths.d.ts +14 -0
- package/dist/utils/paths.d.ts.map +1 -0
- package/dist/utils/paths.js +19 -0
- package/dist/utils/paths.js.map +1 -0
- package/dist/utils/rate-limiter.d.ts +20 -0
- package/dist/utils/rate-limiter.d.ts.map +1 -0
- package/dist/utils/rate-limiter.js +47 -0
- package/dist/utils/rate-limiter.js.map +1 -0
- package/dist/utils/retry.d.ts +26 -0
- package/dist/utils/retry.d.ts.map +1 -0
- package/dist/utils/retry.js +61 -0
- package/dist/utils/retry.js.map +1 -0
- package/launchd/.gitkeep +0 -0
- package/launchd/com.adaria-ai.daemon.plist.template +62 -0
- package/launchd/com.adaria-ai.monitor.plist.template +41 -0
- package/launchd/com.adaria-ai.weekly.plist.template +43 -0
- package/package.json +72 -0
- package/prompts/aso-description.md +44 -0
- package/prompts/aso-inapp-events.md +20 -0
- package/prompts/aso-metadata.md +34 -0
- package/prompts/aso-screenshots.md +20 -0
- package/prompts/onboarding-hypotheses.md +38 -0
- package/prompts/onboarding-review-timing.md +24 -0
- package/prompts/review-clustering.md +19 -0
- package/prompts/review-replies.md +18 -0
- package/prompts/review-sentiment.md +16 -0
- package/prompts/seo-blog-fridgify-recipe.md +116 -0
- package/prompts/seo-blog.md +69 -0
- package/prompts/short-form-ideas.md +50 -0
- package/prompts/social-publish.md +46 -0
|
@@ -0,0 +1,268 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SEO Blog Skill — generates SEO-optimized blog posts and publishes
|
|
3
|
+
* to eodin.app/blogs. Includes Fridgify recipe integration with
|
|
4
|
+
* prompt injection sanitization.
|
|
5
|
+
*
|
|
6
|
+
* Ported from growth-agent `src/agents/seo-blog-agent.js`.
|
|
7
|
+
*/
|
|
8
|
+
import { parseAppNameFromCommand } from "./index.js";
|
|
9
|
+
import { insertBlogPost } from "../db/queries.js";
|
|
10
|
+
import { preparePrompt } from "../prompts/loader.js";
|
|
11
|
+
import { warn as logWarn, info as logInfo } from "../utils/logger.js";
|
|
12
|
+
const MAX_FIELD_LEN = 500;
|
|
13
|
+
const MAX_INGREDIENT_LEN = 120;
|
|
14
|
+
const MAX_INSTRUCTION_LEN = 400;
|
|
15
|
+
const NUTRITION_SENTINEL = { calories: 350, fat: 12, protein: 20, carbohydrates: 40 };
|
|
16
|
+
export class SeoBlogSkill {
|
|
17
|
+
name = "seo-blog";
|
|
18
|
+
commands = ["blog"];
|
|
19
|
+
deps;
|
|
20
|
+
constructor(deps) {
|
|
21
|
+
this.deps = deps;
|
|
22
|
+
}
|
|
23
|
+
async dispatch(ctx, text) {
|
|
24
|
+
const appName = parseAppNameFromCommand(text);
|
|
25
|
+
const app = appName
|
|
26
|
+
? ctx.apps.find((a) => a.id.toLowerCase() === appName)
|
|
27
|
+
: ctx.apps[0];
|
|
28
|
+
if (!app) {
|
|
29
|
+
return {
|
|
30
|
+
summary: appName ? `❌ App "${appName}" not found.` : "❌ No apps configured.",
|
|
31
|
+
alerts: [],
|
|
32
|
+
approvals: [],
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
return this.analyzeSeo(ctx, app);
|
|
36
|
+
}
|
|
37
|
+
async analyzeSeo(ctx, app) {
|
|
38
|
+
const approvals = [];
|
|
39
|
+
// 1. Get existing slugs
|
|
40
|
+
let existingSlugs = [];
|
|
41
|
+
if (this.deps.blogPublisher) {
|
|
42
|
+
try {
|
|
43
|
+
existingSlugs = await this.deps.blogPublisher.listSlugs();
|
|
44
|
+
}
|
|
45
|
+
catch (err) {
|
|
46
|
+
logWarn(`[seo-blog] Failed to list slugs: ${err instanceof Error ? err.message : String(err)}`);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
// 2. Fridgify recipe branch
|
|
50
|
+
let recipes = [];
|
|
51
|
+
let recipesPeriod = null;
|
|
52
|
+
let useRecipePrompt = false;
|
|
53
|
+
if (app.features.fridgifyRecipes && this.deps.recipesCollector) {
|
|
54
|
+
try {
|
|
55
|
+
const cascade = await this.deps.recipesCollector.getPopularWithCascade({
|
|
56
|
+
metric: "combined", limit: 10, minResults: 5,
|
|
57
|
+
});
|
|
58
|
+
if (cascade.satisfied) {
|
|
59
|
+
recipes = cascade.rows;
|
|
60
|
+
recipesPeriod = cascade.period;
|
|
61
|
+
useRecipePrompt = true;
|
|
62
|
+
}
|
|
63
|
+
else {
|
|
64
|
+
logInfo(`[seo-blog] Fridgify cascade unsatisfied (period=${cascade.period}) — using generic prompt`);
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
catch (err) {
|
|
68
|
+
logWarn(`[seo-blog] Fridgify recipes fetch failed: ${err instanceof Error ? err.message : String(err)}`);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
// 3. Build prompt
|
|
72
|
+
const promptName = useRecipePrompt ? "seo-blog-fridgify-recipe" : "seo-blog";
|
|
73
|
+
const baseVars = {
|
|
74
|
+
appName: app.name,
|
|
75
|
+
appDescription: "",
|
|
76
|
+
primaryKeywords: app.primaryKeywords.join(", "),
|
|
77
|
+
asoInsights: "Nothing notable",
|
|
78
|
+
reviewInsights: "Nothing notable",
|
|
79
|
+
seoKeywords: "No data",
|
|
80
|
+
blogPerformance: "No data (first week)",
|
|
81
|
+
trafficSources: "No data",
|
|
82
|
+
existingSlugs: existingSlugs.length > 0 ? existingSlugs.join(", ") : "none",
|
|
83
|
+
};
|
|
84
|
+
if (useRecipePrompt) {
|
|
85
|
+
baseVars["period"] = recipesPeriod ?? "";
|
|
86
|
+
baseVars["recipeCount"] = String(recipes.length);
|
|
87
|
+
baseVars["recipesContext"] = buildRecipesContext(recipes);
|
|
88
|
+
}
|
|
89
|
+
const prompt = preparePrompt(promptName, baseVars);
|
|
90
|
+
let posts = [];
|
|
91
|
+
try {
|
|
92
|
+
const raw = await ctx.runClaude(prompt);
|
|
93
|
+
const result = JSON.parse(raw);
|
|
94
|
+
posts = result.posts ?? [];
|
|
95
|
+
}
|
|
96
|
+
catch {
|
|
97
|
+
// Claude error is non-fatal
|
|
98
|
+
}
|
|
99
|
+
// 5. Validate and prepare posts
|
|
100
|
+
const inputRecipeIds = new Set(recipes.map((r) => r.id).filter((id) => typeof id === "string"));
|
|
101
|
+
const validPosts = [];
|
|
102
|
+
for (const post of posts) {
|
|
103
|
+
if (!post.slug || post.slug.length < 3 || existingSlugs.includes(post.slug))
|
|
104
|
+
continue;
|
|
105
|
+
if (!post.title)
|
|
106
|
+
continue;
|
|
107
|
+
if (Array.isArray(post.sourceRecipeIds)) {
|
|
108
|
+
post.sourceRecipeIds = post.sourceRecipeIds.filter((id) => typeof id === "string" && inputRecipeIds.has(id));
|
|
109
|
+
}
|
|
110
|
+
validPosts.push(post);
|
|
111
|
+
}
|
|
112
|
+
// 6. Create approval items for each post
|
|
113
|
+
for (const post of validPosts) {
|
|
114
|
+
approvals.push({
|
|
115
|
+
id: `blog-publish-${post.slug}`,
|
|
116
|
+
description: `Publish blog post: "${post.title}"`,
|
|
117
|
+
agent: "seo-blog",
|
|
118
|
+
payload: post,
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
// 7. Summary
|
|
122
|
+
const summary = this.buildSummary(app, validPosts, recipesPeriod);
|
|
123
|
+
return { summary, alerts: [], approvals };
|
|
124
|
+
}
|
|
125
|
+
/**
|
|
126
|
+
* Publish approved posts. Called after approval in orchestrator (M6).
|
|
127
|
+
*/
|
|
128
|
+
async publishApprovedPosts(ctx, posts) {
|
|
129
|
+
if (!this.deps.blogPublisher || !this.deps.markdownToHtml || !this.deps.estimateReadTime) {
|
|
130
|
+
return posts.map((p) => ({ slug: p.slug, status: "skipped" }));
|
|
131
|
+
}
|
|
132
|
+
if (process.env["ADARIA_DRY_RUN"] === "1") {
|
|
133
|
+
for (const post of posts) {
|
|
134
|
+
logInfo(`[seo-blog] DRY_RUN: would publish "${post.slug}"`);
|
|
135
|
+
}
|
|
136
|
+
return posts.map((p) => ({ slug: p.slug, status: "dry-run" }));
|
|
137
|
+
}
|
|
138
|
+
const published = [];
|
|
139
|
+
for (const post of posts) {
|
|
140
|
+
try {
|
|
141
|
+
const htmlContent = this.deps.markdownToHtml(post.body ?? "");
|
|
142
|
+
const readTime = this.deps.estimateReadTime(post.body ?? "");
|
|
143
|
+
await this.deps.blogPublisher.create({
|
|
144
|
+
slug: post.slug,
|
|
145
|
+
title: post.title,
|
|
146
|
+
description: post.description ?? "",
|
|
147
|
+
category: post.category ?? "Insights",
|
|
148
|
+
content: htmlContent,
|
|
149
|
+
readTime,
|
|
150
|
+
});
|
|
151
|
+
await this.deps.blogPublisher.publish(post.slug);
|
|
152
|
+
const keywordsPayload = {};
|
|
153
|
+
if (post.sourceRecipeIds?.length) {
|
|
154
|
+
keywordsPayload["_sourceRecipeIds"] = post.sourceRecipeIds;
|
|
155
|
+
}
|
|
156
|
+
insertBlogPost(ctx.db, {
|
|
157
|
+
app_id: "eodin",
|
|
158
|
+
title: post.title,
|
|
159
|
+
slug: post.slug,
|
|
160
|
+
keywords: JSON.stringify(keywordsPayload),
|
|
161
|
+
published_at: new Date().toISOString(),
|
|
162
|
+
});
|
|
163
|
+
published.push({ slug: post.slug, status: "published" });
|
|
164
|
+
logInfo(`[seo-blog] Published: ${post.slug}`);
|
|
165
|
+
}
|
|
166
|
+
catch (err) {
|
|
167
|
+
published.push({ slug: post.slug, status: "failed" });
|
|
168
|
+
logWarn(`[seo-blog] Publish failed for ${post.slug}: ${err instanceof Error ? err.message : String(err)}`);
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
return published;
|
|
172
|
+
}
|
|
173
|
+
buildSummary(app, posts, recipesPeriod) {
|
|
174
|
+
const lines = [`*📝 SEO Blog — ${app.name}*`];
|
|
175
|
+
if (recipesPeriod) {
|
|
176
|
+
lines.push(`• Based on trending Fridgify recipes (${recipesPeriod})`);
|
|
177
|
+
}
|
|
178
|
+
if (posts.length > 0) {
|
|
179
|
+
for (const post of posts) {
|
|
180
|
+
const sourceTag = post.sourceRecipeIds?.length
|
|
181
|
+
? ` [${String(post.sourceRecipeIds.length)} recipe${post.sourceRecipeIds.length > 1 ? "s" : ""}]`
|
|
182
|
+
: "";
|
|
183
|
+
lines.push(`• "${post.title}"${sourceTag}`);
|
|
184
|
+
}
|
|
185
|
+
lines.push(`• ${String(posts.length)} posts ready`);
|
|
186
|
+
}
|
|
187
|
+
else {
|
|
188
|
+
lines.push("• No posts generated this week");
|
|
189
|
+
}
|
|
190
|
+
return lines.join("\n");
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
// ── Recipe sanitization (prompt injection defense) ────────────
|
|
194
|
+
function isNutritionSentinel(n) {
|
|
195
|
+
if (!n || typeof n !== "object")
|
|
196
|
+
return false;
|
|
197
|
+
const obj = n;
|
|
198
|
+
return (obj["calories"] === NUTRITION_SENTINEL.calories &&
|
|
199
|
+
obj["fat"] === NUTRITION_SENTINEL.fat &&
|
|
200
|
+
obj["protein"] === NUTRITION_SENTINEL.protein &&
|
|
201
|
+
obj["carbohydrates"] === NUTRITION_SENTINEL.carbohydrates);
|
|
202
|
+
}
|
|
203
|
+
function sanitizeUserText(value, maxLen = MAX_FIELD_LEN) {
|
|
204
|
+
if (value == null)
|
|
205
|
+
return null;
|
|
206
|
+
const str = (typeof value === "string" ? value : JSON.stringify(value))
|
|
207
|
+
.replace(/<\/?[a-zA-Z][^>]*>/g, " ")
|
|
208
|
+
.replace(/\bignore (?:all )?previous (?:instructions|prompts?)\b/gi, "[filtered]")
|
|
209
|
+
.replace(/\b(?:system|assistant|user)\s*:/gi, "[filtered]:")
|
|
210
|
+
.replace(/\s+/g, " ")
|
|
211
|
+
.trim();
|
|
212
|
+
return str.length > maxLen ? `${str.slice(0, maxLen)}…` : str;
|
|
213
|
+
}
|
|
214
|
+
function sanitizeStringList(list, maxItems, maxLen) {
|
|
215
|
+
if (!Array.isArray(list))
|
|
216
|
+
return [];
|
|
217
|
+
return list
|
|
218
|
+
.slice(0, maxItems)
|
|
219
|
+
.map((item) => sanitizeUserText(item, maxLen))
|
|
220
|
+
.filter((s) => s != null && s.length > 0);
|
|
221
|
+
}
|
|
222
|
+
function buildRecipesContext(recipes) {
|
|
223
|
+
const compact = recipes.map((r, i) => {
|
|
224
|
+
const rec = r;
|
|
225
|
+
const tasteProfile = rec["tasteProfile"];
|
|
226
|
+
const taste = tasteProfile && typeof tasteProfile === "object"
|
|
227
|
+
? {
|
|
228
|
+
sweet: tasteProfile["sweet"],
|
|
229
|
+
salty: tasteProfile["salty"],
|
|
230
|
+
spicy: tasteProfile["spicy"],
|
|
231
|
+
umami: tasteProfile["umami"],
|
|
232
|
+
sour: tasteProfile["sour"],
|
|
233
|
+
texture: sanitizeUserText(tasteProfile["texture"], 120),
|
|
234
|
+
aroma: sanitizeUserText(tasteProfile["aroma"], 120),
|
|
235
|
+
beverage_pairing: sanitizeUserText(tasteProfile["beverage_pairing"], 200),
|
|
236
|
+
side_dishes: sanitizeUserText(tasteProfile["side_dishes"], 200),
|
|
237
|
+
}
|
|
238
|
+
: null;
|
|
239
|
+
const cuisineData = rec["cuisineData"];
|
|
240
|
+
return {
|
|
241
|
+
rank: i + 1,
|
|
242
|
+
id: typeof r.id === "string" ? r.id : null,
|
|
243
|
+
name: sanitizeUserText(r.name, 200),
|
|
244
|
+
cuisine: sanitizeUserText(rec["cuisine"] ?? cuisineData?.["name"], 80),
|
|
245
|
+
cuisineTags: Array.isArray(rec["cuisineTags"])
|
|
246
|
+
? rec["cuisineTags"].slice(0, 10).map((t) => sanitizeUserText(t, 60))
|
|
247
|
+
: [],
|
|
248
|
+
difficulty: sanitizeUserText(r.difficulty, 40),
|
|
249
|
+
estimatedTime: typeof r.estimatedTime === "number" ? r.estimatedTime : null,
|
|
250
|
+
estimatedCost: typeof rec["estimatedCost"] === "number" ? rec["estimatedCost"] : null,
|
|
251
|
+
servings: typeof r.servings === "number" ? r.servings : null,
|
|
252
|
+
ingredients: sanitizeStringList(r.ingredients, 15, MAX_INGREDIENT_LEN),
|
|
253
|
+
instructions: sanitizeStringList(r.instructions, 12, MAX_INSTRUCTION_LEN),
|
|
254
|
+
aiDescription: sanitizeUserText(rec["aiDescription"], MAX_FIELD_LEN),
|
|
255
|
+
tasteProfile: taste,
|
|
256
|
+
nutritionInfo: isNutritionSentinel(r.nutritionInfo) ? null : (r.nutritionInfo ?? null),
|
|
257
|
+
imageUrl: typeof rec["imageUrl"] === "string" ? rec["imageUrl"] : null,
|
|
258
|
+
periodScore: typeof rec["periodScore"] === "number" ? rec["periodScore"] : null,
|
|
259
|
+
stats: {
|
|
260
|
+
upvotes: rec["stats"]?.["upvotes"] ?? 0,
|
|
261
|
+
comments: rec["stats"]?.["comments"] ?? 0,
|
|
262
|
+
saves: rec["stats"]?.["saves"] ?? 0,
|
|
263
|
+
},
|
|
264
|
+
};
|
|
265
|
+
});
|
|
266
|
+
return JSON.stringify(compact, null, 2);
|
|
267
|
+
}
|
|
268
|
+
//# sourceMappingURL=seo-blog.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"seo-blog.js","sourceRoot":"","sources":["../../src/skills/seo-blog.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Short-form Skill — generates video ideas + AI prompts, collects
|
|
3
|
+
* YouTube performance data.
|
|
4
|
+
*
|
|
5
|
+
* Ported from growth-agent `src/agents/short-form-agent.js`.
|
|
6
|
+
*/
|
|
7
|
+
import type { Skill } from "./index.js";
|
|
8
|
+
import type { SkillContext, SkillResult } from "../types/skill.js";
|
|
9
|
+
import type { AppConfig } from "../config/apps-schema.js";
|
|
10
|
+
import type { YouTubeVideoStats } from "../types/collectors.js";
|
|
11
|
+
export interface ShortFormSkillDeps {
|
|
12
|
+
youtube?: {
|
|
13
|
+
getRecentShorts: (channelId: string, limit: number) => Promise<YouTubeVideoStats[]>;
|
|
14
|
+
};
|
|
15
|
+
}
|
|
16
|
+
export declare class ShortFormSkill implements Skill {
|
|
17
|
+
readonly name = "short-form";
|
|
18
|
+
readonly commands: readonly ["shortform", "short-form"];
|
|
19
|
+
private readonly deps;
|
|
20
|
+
constructor(deps: ShortFormSkillDeps);
|
|
21
|
+
dispatch(ctx: SkillContext, text: string): Promise<SkillResult>;
|
|
22
|
+
analyzeShortForm(ctx: SkillContext, app: AppConfig): Promise<SkillResult>;
|
|
23
|
+
private collectYouTubePerformance;
|
|
24
|
+
private formatPerformanceData;
|
|
25
|
+
private findTopPatterns;
|
|
26
|
+
private buildSummary;
|
|
27
|
+
}
|
|
28
|
+
//# sourceMappingURL=short-form.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"short-form.d.ts","sourceRoot":"","sources":["../../src/skills/short-form.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,YAAY,CAAC;AAExC,OAAO,KAAK,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AACnE,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,0BAA0B,CAAC;AAC1D,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,wBAAwB,CAAC;AAQhE,MAAM,WAAW,kBAAkB;IACjC,OAAO,CAAC,EAAE;QACR,eAAe,EAAE,CAAC,SAAS,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,KAAK,OAAO,CAAC,iBAAiB,EAAE,CAAC,CAAC;KACrF,CAAC;CACH;AAED,qBAAa,cAAe,YAAW,KAAK;IAC1C,QAAQ,CAAC,IAAI,gBAAgB;IAC7B,QAAQ,CAAC,QAAQ,uCAAwC;IAEzD,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAqB;gBAE9B,IAAI,EAAE,kBAAkB;IAI9B,QAAQ,CAAC,GAAG,EAAE,YAAY,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,WAAW,CAAC;IAiB/D,gBAAgB,CAAC,GAAG,EAAE,YAAY,EAAE,GAAG,EAAE,SAAS,GAAG,OAAO,CAAC,WAAW,CAAC;YAuCjE,yBAAyB;IAwBvC,OAAO,CAAC,qBAAqB;IAW7B,OAAO,CAAC,eAAe;IAOvB,OAAO,CAAC,YAAY;CAqBrB"}
|
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Short-form Skill — generates video ideas + AI prompts, collects
|
|
3
|
+
* YouTube performance data.
|
|
4
|
+
*
|
|
5
|
+
* Ported from growth-agent `src/agents/short-form-agent.js`.
|
|
6
|
+
*/
|
|
7
|
+
import { parseAppNameFromCommand } from "./index.js";
|
|
8
|
+
import { upsertShortFormPerformance, getRecentShortFormPerformance, } from "../db/queries.js";
|
|
9
|
+
import { preparePrompt } from "../prompts/loader.js";
|
|
10
|
+
import { warn as logWarn } from "../utils/logger.js";
|
|
11
|
+
export class ShortFormSkill {
|
|
12
|
+
name = "short-form";
|
|
13
|
+
commands = ["shortform", "short-form"];
|
|
14
|
+
deps;
|
|
15
|
+
constructor(deps) {
|
|
16
|
+
this.deps = deps;
|
|
17
|
+
}
|
|
18
|
+
async dispatch(ctx, text) {
|
|
19
|
+
const appName = parseAppNameFromCommand(text);
|
|
20
|
+
const app = appName
|
|
21
|
+
? ctx.apps.find((a) => a.id.toLowerCase() === appName)
|
|
22
|
+
: ctx.apps[0];
|
|
23
|
+
if (!app) {
|
|
24
|
+
return {
|
|
25
|
+
summary: appName ? `❌ App "${appName}" not found.` : "❌ No apps configured.",
|
|
26
|
+
alerts: [],
|
|
27
|
+
approvals: [],
|
|
28
|
+
};
|
|
29
|
+
}
|
|
30
|
+
return this.analyzeShortForm(ctx, app);
|
|
31
|
+
}
|
|
32
|
+
async analyzeShortForm(ctx, app) {
|
|
33
|
+
// 1. Collect YouTube Shorts performance
|
|
34
|
+
let performanceData = [];
|
|
35
|
+
if (app.youtubeChannelId && this.deps.youtube) {
|
|
36
|
+
performanceData = await this.collectYouTubePerformance(ctx, app);
|
|
37
|
+
}
|
|
38
|
+
// 2. Get recent performance from DB
|
|
39
|
+
const recentPerformance = getRecentShortFormPerformance(ctx.db, app.id, 14);
|
|
40
|
+
const lastWeekPerformance = this.formatPerformanceData(recentPerformance);
|
|
41
|
+
const topPerformingPatterns = this.findTopPatterns(recentPerformance);
|
|
42
|
+
// 3. Generate ideas via Claude
|
|
43
|
+
const prompt = preparePrompt("short-form-ideas", {
|
|
44
|
+
appName: app.name,
|
|
45
|
+
appDescription: "",
|
|
46
|
+
primaryKeywords: app.primaryKeywords.join(", "),
|
|
47
|
+
lastWeekPerformance,
|
|
48
|
+
topPerformingPatterns,
|
|
49
|
+
asoInsights: "Nothing notable",
|
|
50
|
+
reviewInsights: "Nothing notable",
|
|
51
|
+
webTrafficImpact: "No data",
|
|
52
|
+
});
|
|
53
|
+
let ideas = [];
|
|
54
|
+
try {
|
|
55
|
+
const raw = await ctx.runClaude(prompt);
|
|
56
|
+
const result = JSON.parse(raw);
|
|
57
|
+
ideas = result.ideas ?? [];
|
|
58
|
+
}
|
|
59
|
+
catch {
|
|
60
|
+
// Claude error is non-fatal
|
|
61
|
+
}
|
|
62
|
+
// 4. Build summary
|
|
63
|
+
const summary = this.buildSummary(app, performanceData, ideas.length);
|
|
64
|
+
return { summary, alerts: [], approvals: [] };
|
|
65
|
+
}
|
|
66
|
+
async collectYouTubePerformance(ctx, app) {
|
|
67
|
+
try {
|
|
68
|
+
const shorts = await this.deps.youtube.getRecentShorts(app.youtubeChannelId, 10);
|
|
69
|
+
for (const video of shorts) {
|
|
70
|
+
upsertShortFormPerformance(ctx.db, {
|
|
71
|
+
app_id: app.id,
|
|
72
|
+
platform: "youtube",
|
|
73
|
+
video_id: video.videoId,
|
|
74
|
+
title: video.title,
|
|
75
|
+
views: video.views,
|
|
76
|
+
likes: video.likes,
|
|
77
|
+
comments: video.comments,
|
|
78
|
+
});
|
|
79
|
+
}
|
|
80
|
+
return shorts;
|
|
81
|
+
}
|
|
82
|
+
catch (err) {
|
|
83
|
+
logWarn(`[short-form] YouTube collection failed: ${err instanceof Error ? err.message : String(err)}`);
|
|
84
|
+
return [];
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
formatPerformanceData(data) {
|
|
88
|
+
if (!data.length)
|
|
89
|
+
return "No performance data (first week)";
|
|
90
|
+
const totalViews = data.reduce((sum, d) => sum + d.views, 0);
|
|
91
|
+
const totalLikes = data.reduce((sum, d) => sum + d.likes, 0);
|
|
92
|
+
const lines = [`${String(data.length)} videos | ${totalViews.toLocaleString()} views | ${totalLikes.toLocaleString()} likes`];
|
|
93
|
+
for (const video of data.slice(0, 5)) {
|
|
94
|
+
lines.push(`- "${video.title ?? "Untitled"}" — ${video.views.toLocaleString()} views / ${String(video.likes)} likes`);
|
|
95
|
+
}
|
|
96
|
+
return lines.join("\n");
|
|
97
|
+
}
|
|
98
|
+
findTopPatterns(data) {
|
|
99
|
+
if (!data.length)
|
|
100
|
+
return "Insufficient data for pattern analysis";
|
|
101
|
+
const sorted = [...data].sort((a, b) => b.views - a.views);
|
|
102
|
+
const top = sorted[0];
|
|
103
|
+
return `Best performer: "${top.title ?? "Untitled"}" (${top.views.toLocaleString()} views, ${String(top.likes)} likes)`;
|
|
104
|
+
}
|
|
105
|
+
buildSummary(app, performanceData, ideaCount) {
|
|
106
|
+
const lines = [`*🎬 Short-form — ${app.name}*`];
|
|
107
|
+
if (performanceData.length > 0) {
|
|
108
|
+
const totalViews = performanceData.reduce((sum, d) => sum + d.views, 0);
|
|
109
|
+
const totalLikes = performanceData.reduce((sum, d) => sum + d.likes, 0);
|
|
110
|
+
lines.push(`• Last week: ${totalViews.toLocaleString()} views · ${totalLikes.toLocaleString()} likes`);
|
|
111
|
+
}
|
|
112
|
+
else {
|
|
113
|
+
lines.push("• No performance data (first week or channel not configured)");
|
|
114
|
+
}
|
|
115
|
+
if (ideaCount > 0) {
|
|
116
|
+
lines.push(`• ${String(ideaCount)} new ideas + AI prompts ready`);
|
|
117
|
+
}
|
|
118
|
+
return lines.join("\n");
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
//# sourceMappingURL=short-form.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"short-form.js","sourceRoot":"","sources":["../../src/skills/short-form.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Social publish skill.
|
|
3
|
+
*
|
|
4
|
+
* Generates platform-optimised marketing content via Claude and produces
|
|
5
|
+
* ApprovalItem[] — one per enabled platform. On approval, the platform
|
|
6
|
+
* client posts and the result is recorded in the social_posts DB table.
|
|
7
|
+
*/
|
|
8
|
+
import type { Skill } from "./index.js";
|
|
9
|
+
import type { SkillContext, SkillResult } from "../types/skill.js";
|
|
10
|
+
import type { SocialPlatform, SocialPostContent } from "../social/base.js";
|
|
11
|
+
import { type SocialConfigs } from "../social/factory.js";
|
|
12
|
+
export interface SocialPublishSkillDeps {
|
|
13
|
+
socialConfigs: SocialConfigs;
|
|
14
|
+
}
|
|
15
|
+
export declare class SocialPublishSkill implements Skill {
|
|
16
|
+
private readonly deps;
|
|
17
|
+
readonly name = "social-publish";
|
|
18
|
+
readonly commands: readonly ["social", "소셜", "sns"];
|
|
19
|
+
constructor(deps: SocialPublishSkillDeps);
|
|
20
|
+
dispatch(ctx: SkillContext, text: string): Promise<SkillResult>;
|
|
21
|
+
/**
|
|
22
|
+
* Execute the actual post after approval. Called by the approval
|
|
23
|
+
* callback handler with the payload from the ApprovalItem.
|
|
24
|
+
*/
|
|
25
|
+
executePost(ctx: SkillContext, payload: {
|
|
26
|
+
platform: SocialPlatform;
|
|
27
|
+
appId: string;
|
|
28
|
+
content: SocialPostContent;
|
|
29
|
+
}): Promise<void>;
|
|
30
|
+
private generateContent;
|
|
31
|
+
}
|
|
32
|
+
//# sourceMappingURL=social-publish.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"social-publish.d.ts","sourceRoot":"","sources":["../../src/skills/social-publish.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,YAAY,CAAC;AACxC,OAAO,KAAK,EAAE,YAAY,EAAE,WAAW,EAAgB,MAAM,mBAAmB,CAAC;AAEjF,OAAO,KAAK,EAAE,cAAc,EAAE,iBAAiB,EAAE,MAAM,mBAAmB,CAAC;AAC3E,OAAO,EAAsB,KAAK,aAAa,EAAE,MAAM,sBAAsB,CAAC;AAsB9E,MAAM,WAAW,sBAAsB;IACrC,aAAa,EAAE,aAAa,CAAC;CAC9B;AAED,qBAAa,kBAAmB,YAAW,KAAK;IAIlC,OAAO,CAAC,QAAQ,CAAC,IAAI;IAHjC,QAAQ,CAAC,IAAI,oBAAoB;IACjC,QAAQ,CAAC,QAAQ,mCAA8C;gBAElC,IAAI,EAAE,sBAAsB;IAEnD,QAAQ,CAAC,GAAG,EAAE,YAAY,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,WAAW,CAAC;IAsErE;;;OAGG;IACG,WAAW,CACf,GAAG,EAAE,YAAY,EACjB,OAAO,EAAE;QACP,QAAQ,EAAE,cAAc,CAAC;QACzB,KAAK,EAAE,MAAM,CAAC;QACd,OAAO,EAAE,iBAAiB,CAAC;KAC5B,GACA,OAAO,CAAC,IAAI,CAAC;YAoCF,eAAe;CAkC9B"}
|
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Social publish skill.
|
|
3
|
+
*
|
|
4
|
+
* Generates platform-optimised marketing content via Claude and produces
|
|
5
|
+
* ApprovalItem[] — one per enabled platform. On approval, the platform
|
|
6
|
+
* client posts and the result is recorded in the social_posts DB table.
|
|
7
|
+
*/
|
|
8
|
+
import { createSocialClient } from "../social/factory.js";
|
|
9
|
+
import { insertSocialPost } from "../db/queries.js";
|
|
10
|
+
import { parseAppNameFromCommand } from "./index.js";
|
|
11
|
+
import { preparePrompt } from "../prompts/loader.js";
|
|
12
|
+
import { parseJsonResponse } from "../utils/parse-json.js";
|
|
13
|
+
import * as logger from "../utils/logger.js";
|
|
14
|
+
const ALL_PLATFORMS = [
|
|
15
|
+
"twitter",
|
|
16
|
+
"facebook",
|
|
17
|
+
"threads",
|
|
18
|
+
"tiktok",
|
|
19
|
+
"youtube",
|
|
20
|
+
"linkedin",
|
|
21
|
+
];
|
|
22
|
+
export class SocialPublishSkill {
|
|
23
|
+
deps;
|
|
24
|
+
name = "social-publish";
|
|
25
|
+
commands = ["social", "\uc18c\uc15c", "sns"];
|
|
26
|
+
constructor(deps) {
|
|
27
|
+
this.deps = deps;
|
|
28
|
+
}
|
|
29
|
+
async dispatch(ctx, text) {
|
|
30
|
+
const appName = parseAppNameFromCommand(text);
|
|
31
|
+
const app = appName
|
|
32
|
+
? ctx.apps.find((a) => a.name.toLowerCase() === appName.toLowerCase())
|
|
33
|
+
: ctx.apps[0];
|
|
34
|
+
if (!app) {
|
|
35
|
+
return {
|
|
36
|
+
summary: `App not found: ${appName ?? "(none)"}`,
|
|
37
|
+
alerts: [],
|
|
38
|
+
approvals: [],
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
// Determine which platforms are enabled for this app
|
|
42
|
+
const enabledPlatforms = ALL_PLATFORMS.filter((p) => app.social[p] === true);
|
|
43
|
+
if (enabledPlatforms.length === 0) {
|
|
44
|
+
return {
|
|
45
|
+
summary: `No social platforms enabled for ${app.name}. Enable platforms in apps.yaml under \`social:\`.`,
|
|
46
|
+
alerts: [],
|
|
47
|
+
approvals: [],
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
// Generate platform-specific content via Claude
|
|
51
|
+
const platformContents = await this.generateContent(ctx, app, enabledPlatforms);
|
|
52
|
+
if (platformContents.length === 0) {
|
|
53
|
+
return {
|
|
54
|
+
summary: `Failed to generate social content for ${app.name}.`,
|
|
55
|
+
alerts: [],
|
|
56
|
+
approvals: [],
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
// Build approval items — one per platform
|
|
60
|
+
const approvals = platformContents.map((pc) => ({
|
|
61
|
+
id: `social-${pc.platform}-${app.id}-${Date.now()}`,
|
|
62
|
+
description: `[${pc.platform.toUpperCase()}] ${pc.text.slice(0, 100)}${pc.text.length > 100 ? "..." : ""}`,
|
|
63
|
+
agent: "social-publish",
|
|
64
|
+
payload: {
|
|
65
|
+
platform: pc.platform,
|
|
66
|
+
appId: app.id,
|
|
67
|
+
content: {
|
|
68
|
+
text: pc.text,
|
|
69
|
+
hashtags: pc.hashtags,
|
|
70
|
+
},
|
|
71
|
+
},
|
|
72
|
+
}));
|
|
73
|
+
const platformList = platformContents
|
|
74
|
+
.map((pc) => `\u2022 *${pc.platform}*: ${pc.text.slice(0, 80)}...`)
|
|
75
|
+
.join("\n");
|
|
76
|
+
return {
|
|
77
|
+
summary: `Generated social content for ${app.name} on ${String(enabledPlatforms.length)} platform(s):\n${platformList}\n\n_Approve each platform to publish._`,
|
|
78
|
+
alerts: [],
|
|
79
|
+
approvals,
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
/**
|
|
83
|
+
* Execute the actual post after approval. Called by the approval
|
|
84
|
+
* callback handler with the payload from the ApprovalItem.
|
|
85
|
+
*/
|
|
86
|
+
async executePost(ctx, payload) {
|
|
87
|
+
const client = createSocialClient(payload.platform, this.deps.socialConfigs);
|
|
88
|
+
if (!client) {
|
|
89
|
+
logger.error(`[social-publish] No client for ${payload.platform} — credentials missing`);
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
const result = await client.post(payload.content);
|
|
93
|
+
insertSocialPost(ctx.db, {
|
|
94
|
+
app_id: payload.appId,
|
|
95
|
+
platform: payload.platform,
|
|
96
|
+
post_id: result.postId ?? null,
|
|
97
|
+
post_url: result.postUrl ?? null,
|
|
98
|
+
content: payload.content.text,
|
|
99
|
+
image_url: payload.content.imageUrl ?? null,
|
|
100
|
+
status: result.success ? "posted" : "failed",
|
|
101
|
+
});
|
|
102
|
+
if (result.success) {
|
|
103
|
+
logger.info(`[social-publish] Posted to ${payload.platform}: ${result.postUrl ?? result.postId ?? "ok"}`);
|
|
104
|
+
}
|
|
105
|
+
else {
|
|
106
|
+
logger.error(`[social-publish] Failed to post to ${payload.platform}: ${result.error ?? "unknown"}`);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
async generateContent(ctx, app, platforms) {
|
|
110
|
+
try {
|
|
111
|
+
const prompt = preparePrompt("social-publish", {
|
|
112
|
+
appName: app.name,
|
|
113
|
+
platforms: platforms.join(", "),
|
|
114
|
+
keywords: app.primaryKeywords.join(", ") || "N/A",
|
|
115
|
+
});
|
|
116
|
+
const response = await ctx.runClaude(prompt);
|
|
117
|
+
const parsed = parseJsonResponse(response);
|
|
118
|
+
if (!Array.isArray(parsed)) {
|
|
119
|
+
logger.error("[social-publish] Claude did not return a JSON array");
|
|
120
|
+
return [];
|
|
121
|
+
}
|
|
122
|
+
return parsed.filter((pc) => typeof pc.platform === "string" &&
|
|
123
|
+
typeof pc.text === "string" &&
|
|
124
|
+
pc.text.trim().length > 0 &&
|
|
125
|
+
platforms.includes(pc.platform));
|
|
126
|
+
}
|
|
127
|
+
catch (err) {
|
|
128
|
+
logger.error(`[social-publish] Content generation failed: ${err instanceof Error ? err.message : String(err)}`);
|
|
129
|
+
return [];
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
//# sourceMappingURL=social-publish.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"social-publish.js","sourceRoot":"","sources":["../../src/skills/social-publish.ts"],"names":[],"mappings":"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"}
|