@timmeck/marketing-brain 0.2.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/.mcp.json +9 -0
- package/README.md +342 -0
- package/dashboard.html +666 -0
- package/dist/api/server.d.ts +15 -0
- package/dist/api/server.js +73 -0
- package/dist/api/server.js.map +1 -0
- package/dist/cli/colors.d.ts +43 -0
- package/dist/cli/colors.js +54 -0
- package/dist/cli/colors.js.map +1 -0
- package/dist/cli/commands/campaign.d.ts +2 -0
- package/dist/cli/commands/campaign.js +62 -0
- package/dist/cli/commands/campaign.js.map +1 -0
- package/dist/cli/commands/config.d.ts +2 -0
- package/dist/cli/commands/config.js +164 -0
- package/dist/cli/commands/config.js.map +1 -0
- package/dist/cli/commands/dashboard.d.ts +2 -0
- package/dist/cli/commands/dashboard.js +147 -0
- package/dist/cli/commands/dashboard.js.map +1 -0
- package/dist/cli/commands/doctor.d.ts +2 -0
- package/dist/cli/commands/doctor.js +111 -0
- package/dist/cli/commands/doctor.js.map +1 -0
- package/dist/cli/commands/export.d.ts +2 -0
- package/dist/cli/commands/export.js +37 -0
- package/dist/cli/commands/export.js.map +1 -0
- package/dist/cli/commands/import.d.ts +2 -0
- package/dist/cli/commands/import.js +76 -0
- package/dist/cli/commands/import.js.map +1 -0
- package/dist/cli/commands/insights.d.ts +2 -0
- package/dist/cli/commands/insights.js +41 -0
- package/dist/cli/commands/insights.js.map +1 -0
- package/dist/cli/commands/learn.d.ts +2 -0
- package/dist/cli/commands/learn.js +22 -0
- package/dist/cli/commands/learn.js.map +1 -0
- package/dist/cli/commands/network.d.ts +2 -0
- package/dist/cli/commands/network.js +66 -0
- package/dist/cli/commands/network.js.map +1 -0
- package/dist/cli/commands/post.d.ts +2 -0
- package/dist/cli/commands/post.js +45 -0
- package/dist/cli/commands/post.js.map +1 -0
- package/dist/cli/commands/query.d.ts +2 -0
- package/dist/cli/commands/query.js +96 -0
- package/dist/cli/commands/query.js.map +1 -0
- package/dist/cli/commands/rules.d.ts +2 -0
- package/dist/cli/commands/rules.js +25 -0
- package/dist/cli/commands/rules.js.map +1 -0
- package/dist/cli/commands/start.d.ts +2 -0
- package/dist/cli/commands/start.js +91 -0
- package/dist/cli/commands/start.js.map +1 -0
- package/dist/cli/commands/status.d.ts +2 -0
- package/dist/cli/commands/status.js +63 -0
- package/dist/cli/commands/status.js.map +1 -0
- package/dist/cli/commands/stop.d.ts +2 -0
- package/dist/cli/commands/stop.js +34 -0
- package/dist/cli/commands/stop.js.map +1 -0
- package/dist/cli/commands/suggest.d.ts +2 -0
- package/dist/cli/commands/suggest.js +57 -0
- package/dist/cli/commands/suggest.js.map +1 -0
- package/dist/cli/ipc-helper.d.ts +2 -0
- package/dist/cli/ipc-helper.js +26 -0
- package/dist/cli/ipc-helper.js.map +1 -0
- package/dist/cli/update-check.d.ts +2 -0
- package/dist/cli/update-check.js +58 -0
- package/dist/cli/update-check.js.map +1 -0
- package/dist/config.d.ts +2 -0
- package/dist/config.js +111 -0
- package/dist/config.js.map +1 -0
- package/dist/dashboard/renderer.d.ts +11 -0
- package/dist/dashboard/renderer.js +112 -0
- package/dist/dashboard/renderer.js.map +1 -0
- package/dist/dashboard/server.d.ts +15 -0
- package/dist/dashboard/server.js +122 -0
- package/dist/dashboard/server.js.map +1 -0
- package/dist/db/connection.d.ts +2 -0
- package/dist/db/connection.js +19 -0
- package/dist/db/connection.js.map +1 -0
- package/dist/db/migrations/001_core_schema.d.ts +2 -0
- package/dist/db/migrations/001_core_schema.js +62 -0
- package/dist/db/migrations/001_core_schema.js.map +1 -0
- package/dist/db/migrations/002_learning_schema.d.ts +2 -0
- package/dist/db/migrations/002_learning_schema.js +45 -0
- package/dist/db/migrations/002_learning_schema.js.map +1 -0
- package/dist/db/migrations/003_synapse_schema.d.ts +2 -0
- package/dist/db/migrations/003_synapse_schema.js +26 -0
- package/dist/db/migrations/003_synapse_schema.js.map +1 -0
- package/dist/db/migrations/004_insights_schema.d.ts +2 -0
- package/dist/db/migrations/004_insights_schema.js +37 -0
- package/dist/db/migrations/004_insights_schema.js.map +1 -0
- package/dist/db/migrations/005_fts_indexes.d.ts +2 -0
- package/dist/db/migrations/005_fts_indexes.js +76 -0
- package/dist/db/migrations/005_fts_indexes.js.map +1 -0
- package/dist/db/migrations/index.d.ts +2 -0
- package/dist/db/migrations/index.js +47 -0
- package/dist/db/migrations/index.js.map +1 -0
- package/dist/db/repositories/audience.repository.d.ts +18 -0
- package/dist/db/repositories/audience.repository.js +45 -0
- package/dist/db/repositories/audience.repository.js.map +1 -0
- package/dist/db/repositories/campaign.repository.d.ts +15 -0
- package/dist/db/repositories/campaign.repository.js +58 -0
- package/dist/db/repositories/campaign.repository.js.map +1 -0
- package/dist/db/repositories/engagement.repository.d.ts +26 -0
- package/dist/db/repositories/engagement.repository.js +83 -0
- package/dist/db/repositories/engagement.repository.js.map +1 -0
- package/dist/db/repositories/insight.repository.d.ts +18 -0
- package/dist/db/repositories/insight.repository.js +87 -0
- package/dist/db/repositories/insight.repository.js.map +1 -0
- package/dist/db/repositories/post.repository.d.ts +21 -0
- package/dist/db/repositories/post.repository.js +105 -0
- package/dist/db/repositories/post.repository.js.map +1 -0
- package/dist/db/repositories/rule.repository.d.ts +16 -0
- package/dist/db/repositories/rule.repository.js +71 -0
- package/dist/db/repositories/rule.repository.js.map +1 -0
- package/dist/db/repositories/strategy.repository.d.ts +16 -0
- package/dist/db/repositories/strategy.repository.js +69 -0
- package/dist/db/repositories/strategy.repository.js.map +1 -0
- package/dist/db/repositories/synapse.repository.d.ts +25 -0
- package/dist/db/repositories/synapse.repository.js +115 -0
- package/dist/db/repositories/synapse.repository.js.map +1 -0
- package/dist/db/repositories/template.repository.d.ts +16 -0
- package/dist/db/repositories/template.repository.js +61 -0
- package/dist/db/repositories/template.repository.js.map +1 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +62 -0
- package/dist/index.js.map +1 -0
- package/dist/ipc/client.d.ts +13 -0
- package/dist/ipc/client.js +93 -0
- package/dist/ipc/client.js.map +1 -0
- package/dist/ipc/protocol.d.ts +8 -0
- package/dist/ipc/protocol.js +29 -0
- package/dist/ipc/protocol.js.map +1 -0
- package/dist/ipc/router.d.ts +30 -0
- package/dist/ipc/router.js +88 -0
- package/dist/ipc/router.js.map +1 -0
- package/dist/ipc/server.d.ts +14 -0
- package/dist/ipc/server.js +130 -0
- package/dist/ipc/server.js.map +1 -0
- package/dist/learning/confidence-scorer.d.ts +17 -0
- package/dist/learning/confidence-scorer.js +26 -0
- package/dist/learning/confidence-scorer.js.map +1 -0
- package/dist/learning/learning-engine.d.ts +33 -0
- package/dist/learning/learning-engine.js +211 -0
- package/dist/learning/learning-engine.js.map +1 -0
- package/dist/marketing-core.d.ts +17 -0
- package/dist/marketing-core.js +233 -0
- package/dist/marketing-core.js.map +1 -0
- package/dist/mcp/server.d.ts +1 -0
- package/dist/mcp/server.js +67 -0
- package/dist/mcp/server.js.map +1 -0
- package/dist/mcp/tools.d.ts +3 -0
- package/dist/mcp/tools.js +138 -0
- package/dist/mcp/tools.js.map +1 -0
- package/dist/research/research-engine.d.ts +28 -0
- package/dist/research/research-engine.js +211 -0
- package/dist/research/research-engine.js.map +1 -0
- package/dist/services/analytics.service.d.ts +116 -0
- package/dist/services/analytics.service.js +69 -0
- package/dist/services/analytics.service.js.map +1 -0
- package/dist/services/audience.service.d.ts +20 -0
- package/dist/services/audience.service.js +30 -0
- package/dist/services/audience.service.js.map +1 -0
- package/dist/services/campaign.service.d.ts +27 -0
- package/dist/services/campaign.service.js +65 -0
- package/dist/services/campaign.service.js.map +1 -0
- package/dist/services/insight.service.d.ts +18 -0
- package/dist/services/insight.service.js +40 -0
- package/dist/services/insight.service.js.map +1 -0
- package/dist/services/post.service.d.ts +48 -0
- package/dist/services/post.service.js +93 -0
- package/dist/services/post.service.js.map +1 -0
- package/dist/services/rule.service.d.ts +29 -0
- package/dist/services/rule.service.js +67 -0
- package/dist/services/rule.service.js.map +1 -0
- package/dist/services/strategy.service.d.ts +17 -0
- package/dist/services/strategy.service.js +39 -0
- package/dist/services/strategy.service.js.map +1 -0
- package/dist/services/synapse.service.d.ts +22 -0
- package/dist/services/synapse.service.js +22 -0
- package/dist/services/synapse.service.js.map +1 -0
- package/dist/services/template.service.d.ts +17 -0
- package/dist/services/template.service.js +37 -0
- package/dist/services/template.service.js.map +1 -0
- package/dist/synapses/activation.d.ts +13 -0
- package/dist/synapses/activation.js +50 -0
- package/dist/synapses/activation.js.map +1 -0
- package/dist/synapses/decay.d.ts +11 -0
- package/dist/synapses/decay.js +27 -0
- package/dist/synapses/decay.js.map +1 -0
- package/dist/synapses/hebbian.d.ts +13 -0
- package/dist/synapses/hebbian.js +35 -0
- package/dist/synapses/hebbian.js.map +1 -0
- package/dist/synapses/pathfinder.d.ts +14 -0
- package/dist/synapses/pathfinder.js +50 -0
- package/dist/synapses/pathfinder.js.map +1 -0
- package/dist/synapses/synapse-manager.d.ts +32 -0
- package/dist/synapses/synapse-manager.js +76 -0
- package/dist/synapses/synapse-manager.js.map +1 -0
- package/dist/types/config.types.d.ts +69 -0
- package/dist/types/config.types.js +2 -0
- package/dist/types/config.types.js.map +1 -0
- package/dist/types/ipc.types.d.ts +11 -0
- package/dist/types/ipc.types.js +2 -0
- package/dist/types/ipc.types.js.map +1 -0
- package/dist/types/post.types.d.ts +141 -0
- package/dist/types/post.types.js +2 -0
- package/dist/types/post.types.js.map +1 -0
- package/dist/types/synapse.types.d.ts +23 -0
- package/dist/types/synapse.types.js +2 -0
- package/dist/types/synapse.types.js.map +1 -0
- package/dist/utils/events.d.ts +57 -0
- package/dist/utils/events.js +23 -0
- package/dist/utils/events.js.map +1 -0
- package/dist/utils/hash.d.ts +1 -0
- package/dist/utils/hash.js +5 -0
- package/dist/utils/hash.js.map +1 -0
- package/dist/utils/logger.d.ts +8 -0
- package/dist/utils/logger.js +39 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/paths.d.ts +3 -0
- package/dist/utils/paths.js +18 -0
- package/dist/utils/paths.js.map +1 -0
- package/package.json +40 -0
- package/seed-data.json +78 -0
- package/src/api/server.ts +86 -0
- package/src/cli/colors.ts +59 -0
- package/src/cli/commands/campaign.ts +66 -0
- package/src/cli/commands/config.ts +168 -0
- package/src/cli/commands/dashboard.ts +165 -0
- package/src/cli/commands/doctor.ts +110 -0
- package/src/cli/commands/export.ts +40 -0
- package/src/cli/commands/import.ts +84 -0
- package/src/cli/commands/insights.ts +44 -0
- package/src/cli/commands/learn.ts +24 -0
- package/src/cli/commands/network.ts +71 -0
- package/src/cli/commands/post.ts +47 -0
- package/src/cli/commands/query.ts +108 -0
- package/src/cli/commands/rules.ts +27 -0
- package/src/cli/commands/start.ts +100 -0
- package/src/cli/commands/status.ts +73 -0
- package/src/cli/commands/stop.ts +33 -0
- package/src/cli/commands/suggest.ts +64 -0
- package/src/cli/ipc-helper.ts +22 -0
- package/src/cli/update-check.ts +63 -0
- package/src/config.ts +110 -0
- package/src/dashboard/renderer.ts +136 -0
- package/src/dashboard/server.ts +140 -0
- package/src/db/connection.ts +22 -0
- package/src/db/migrations/001_core_schema.ts +63 -0
- package/src/db/migrations/002_learning_schema.ts +46 -0
- package/src/db/migrations/003_synapse_schema.ts +27 -0
- package/src/db/migrations/004_insights_schema.ts +38 -0
- package/src/db/migrations/005_fts_indexes.ts +77 -0
- package/src/db/migrations/index.ts +62 -0
- package/src/db/repositories/audience.repository.ts +53 -0
- package/src/db/repositories/campaign.repository.ts +72 -0
- package/src/db/repositories/engagement.repository.ts +108 -0
- package/src/db/repositories/insight.repository.ts +100 -0
- package/src/db/repositories/post.repository.ts +123 -0
- package/src/db/repositories/rule.repository.ts +87 -0
- package/src/db/repositories/strategy.repository.ts +82 -0
- package/src/db/repositories/synapse.repository.ts +148 -0
- package/src/db/repositories/template.repository.ts +76 -0
- package/src/index.ts +69 -0
- package/src/ipc/client.ts +110 -0
- package/src/ipc/protocol.ts +35 -0
- package/src/ipc/router.ts +126 -0
- package/src/ipc/server.ts +140 -0
- package/src/learning/confidence-scorer.ts +36 -0
- package/src/learning/learning-engine.ts +254 -0
- package/src/marketing-core.ts +285 -0
- package/src/mcp/server.ts +72 -0
- package/src/mcp/tools.ts +216 -0
- package/src/research/research-engine.ts +226 -0
- package/src/services/analytics.service.ts +73 -0
- package/src/services/audience.service.ts +40 -0
- package/src/services/campaign.service.ts +80 -0
- package/src/services/insight.service.ts +54 -0
- package/src/services/post.service.ts +116 -0
- package/src/services/rule.service.ts +90 -0
- package/src/services/strategy.service.ts +53 -0
- package/src/services/synapse.service.ts +32 -0
- package/src/services/template.service.ts +50 -0
- package/src/synapses/activation.ts +80 -0
- package/src/synapses/decay.ts +38 -0
- package/src/synapses/hebbian.ts +68 -0
- package/src/synapses/pathfinder.ts +81 -0
- package/src/synapses/synapse-manager.ts +115 -0
- package/src/types/config.types.ts +79 -0
- package/src/types/ipc.types.ts +8 -0
- package/src/types/post.types.ts +156 -0
- package/src/types/synapse.types.ts +43 -0
- package/src/utils/events.ts +44 -0
- package/src/utils/hash.ts +5 -0
- package/src/utils/logger.ts +48 -0
- package/src/utils/paths.ts +19 -0
- package/tsconfig.json +18 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/db/migrations/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,SAAS,EAAE,MAAM,uBAAuB,CAAC;AAClD,OAAO,EAAE,EAAE,IAAI,UAAU,EAAE,MAAM,sBAAsB,CAAC;AACxD,OAAO,EAAE,EAAE,IAAI,cAAc,EAAE,MAAM,0BAA0B,CAAC;AAChE,OAAO,EAAE,EAAE,IAAI,aAAa,EAAE,MAAM,yBAAyB,CAAC;AAC9D,OAAO,EAAE,EAAE,IAAI,cAAc,EAAE,MAAM,0BAA0B,CAAC;AAChE,OAAO,EAAE,EAAE,IAAI,UAAU,EAAE,MAAM,sBAAsB,CAAC;AAQxD,MAAM,UAAU,GAAgB;IAC9B,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,iBAAiB,EAAE,EAAE,EAAE,UAAU,EAAE;IACvD,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,qBAAqB,EAAE,EAAE,EAAE,cAAc,EAAE;IAC/D,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,oBAAoB,EAAE,EAAE,EAAE,aAAa,EAAE;IAC7D,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,qBAAqB,EAAE,EAAE,EAAE,cAAc,EAAE;IAC/D,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,iBAAiB,EAAE,EAAE,EAAE,UAAU,EAAE;CACxD,CAAC;AAEF,SAAS,qBAAqB,CAAC,EAAqB;IAClD,EAAE,CAAC,IAAI,CAAC;;;;;;GAMP,CAAC,CAAC;AACL,CAAC;AAED,SAAS,iBAAiB,CAAC,EAAqB;IAC9C,MAAM,GAAG,GAAG,EAAE,CAAC,OAAO,CAAC,gDAAgD,CAAC,CAAC,GAAG,EAA4C,CAAC;IACzH,OAAO,GAAG,EAAE,OAAO,IAAI,CAAC,CAAC;AAC3B,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,EAAqB;IACjD,MAAM,MAAM,GAAG,SAAS,EAAE,CAAC;IAC3B,qBAAqB,CAAC,EAAE,CAAC,CAAC;IAE1B,MAAM,cAAc,GAAG,iBAAiB,CAAC,EAAE,CAAC,CAAC;IAC7C,MAAM,OAAO,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,GAAG,cAAc,CAAC,CAAC;IAEnE,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACzB,MAAM,CAAC,IAAI,CAAC,wBAAwB,CAAC,CAAC;QACtC,OAAO;IACT,CAAC;IAED,MAAM,CAAC,IAAI,CAAC,WAAW,OAAO,CAAC,MAAM,8BAA8B,cAAc,EAAE,CAAC,CAAC;IAErF,MAAM,MAAM,GAAG,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE;QACjC,KAAK,MAAM,SAAS,IAAI,OAAO,EAAE,CAAC;YAChC,MAAM,CAAC,IAAI,CAAC,sBAAsB,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC;YACpD,SAAS,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;YACjB,EAAE,CAAC,OAAO,CAAC,sDAAsD,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,OAAO,EAAE,SAAS,CAAC,IAAI,CAAC,CAAC;QAC5G,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,MAAM,EAAE,CAAC;IACT,MAAM,CAAC,IAAI,CAAC,uCAAuC,OAAO,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAE,CAAC,OAAO,EAAE,CAAC,CAAC;AAC7F,CAAC"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import type Database from 'better-sqlite3';
|
|
2
|
+
import type { Audience } from '../../types/post.types.js';
|
|
3
|
+
export declare class AudienceRepository {
|
|
4
|
+
private db;
|
|
5
|
+
private stmts;
|
|
6
|
+
constructor(db: Database.Database);
|
|
7
|
+
create(data: {
|
|
8
|
+
name: string;
|
|
9
|
+
platform?: string;
|
|
10
|
+
demographics?: string;
|
|
11
|
+
interests?: string;
|
|
12
|
+
}): number;
|
|
13
|
+
getById(id: number): Audience | undefined;
|
|
14
|
+
getByName(name: string): Audience | undefined;
|
|
15
|
+
listAll(): Audience[];
|
|
16
|
+
countAll(): number;
|
|
17
|
+
delete(id: number): boolean;
|
|
18
|
+
}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
export class AudienceRepository {
|
|
2
|
+
db;
|
|
3
|
+
stmts;
|
|
4
|
+
constructor(db) {
|
|
5
|
+
this.db = db;
|
|
6
|
+
this.stmts = {
|
|
7
|
+
create: db.prepare(`
|
|
8
|
+
INSERT INTO audiences (name, platform, demographics, interests)
|
|
9
|
+
VALUES (@name, @platform, @demographics, @interests)
|
|
10
|
+
`),
|
|
11
|
+
getById: db.prepare('SELECT * FROM audiences WHERE id = ?'),
|
|
12
|
+
getByName: db.prepare('SELECT * FROM audiences WHERE name = ?'),
|
|
13
|
+
listAll: db.prepare('SELECT * FROM audiences ORDER BY created_at DESC'),
|
|
14
|
+
countAll: db.prepare('SELECT COUNT(*) as count FROM audiences'),
|
|
15
|
+
delete: db.prepare('DELETE FROM audiences WHERE id = ?'),
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
create(data) {
|
|
19
|
+
const result = this.stmts.create.run({
|
|
20
|
+
name: data.name,
|
|
21
|
+
platform: data.platform ?? null,
|
|
22
|
+
demographics: data.demographics ?? null,
|
|
23
|
+
interests: data.interests ?? null,
|
|
24
|
+
});
|
|
25
|
+
return result.lastInsertRowid;
|
|
26
|
+
}
|
|
27
|
+
getById(id) {
|
|
28
|
+
return this.stmts.getById.get(id);
|
|
29
|
+
}
|
|
30
|
+
getByName(name) {
|
|
31
|
+
return this.stmts.getByName.get(name);
|
|
32
|
+
}
|
|
33
|
+
listAll() {
|
|
34
|
+
return this.stmts.listAll.all();
|
|
35
|
+
}
|
|
36
|
+
countAll() {
|
|
37
|
+
const row = this.stmts.countAll.get();
|
|
38
|
+
return row.count;
|
|
39
|
+
}
|
|
40
|
+
delete(id) {
|
|
41
|
+
const result = this.stmts.delete.run(id);
|
|
42
|
+
return result.changes > 0;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
//# sourceMappingURL=audience.repository.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"audience.repository.js","sourceRoot":"","sources":["../../../src/db/repositories/audience.repository.ts"],"names":[],"mappings":"AAIA,MAAM,OAAO,kBAAkB;IAGT;IAFZ,KAAK,CAA4B;IAEzC,YAAoB,EAAqB;QAArB,OAAE,GAAF,EAAE,CAAmB;QACvC,IAAI,CAAC,KAAK,GAAG;YACX,MAAM,EAAE,EAAE,CAAC,OAAO,CAAC;;;OAGlB,CAAC;YACF,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,sCAAsC,CAAC;YAC3D,SAAS,EAAE,EAAE,CAAC,OAAO,CAAC,wCAAwC,CAAC;YAC/D,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,kDAAkD,CAAC;YACvE,QAAQ,EAAE,EAAE,CAAC,OAAO,CAAC,yCAAyC,CAAC;YAC/D,MAAM,EAAE,EAAE,CAAC,OAAO,CAAC,oCAAoC,CAAC;SACzD,CAAC;IACJ,CAAC;IAED,MAAM,CAAC,IAAoF;QACzF,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC;YACnC,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,QAAQ,EAAE,IAAI,CAAC,QAAQ,IAAI,IAAI;YAC/B,YAAY,EAAE,IAAI,CAAC,YAAY,IAAI,IAAI;YACvC,SAAS,EAAE,IAAI,CAAC,SAAS,IAAI,IAAI;SAClC,CAAC,CAAC;QACH,OAAO,MAAM,CAAC,eAAyB,CAAC;IAC1C,CAAC;IAED,OAAO,CAAC,EAAU;QAChB,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAyB,CAAC;IAC5D,CAAC;IAED,SAAS,CAAC,IAAY;QACpB,OAAO,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAyB,CAAC;IAChE,CAAC;IAED,OAAO;QACL,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,EAAgB,CAAC;IAChD,CAAC;IAED,QAAQ;QACN,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,EAAuB,CAAC;QAC3D,OAAO,GAAG,CAAC,KAAK,CAAC;IACnB,CAAC;IAED,MAAM,CAAC,EAAU;QACf,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QACzC,OAAO,MAAM,CAAC,OAAO,GAAG,CAAC,CAAC;IAC5B,CAAC;CACF"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import type Database from 'better-sqlite3';
|
|
2
|
+
import type { Campaign, CampaignCreate } from '../../types/post.types.js';
|
|
3
|
+
export declare class CampaignRepository {
|
|
4
|
+
private db;
|
|
5
|
+
private stmts;
|
|
6
|
+
constructor(db: Database.Database);
|
|
7
|
+
create(data: CampaignCreate): number;
|
|
8
|
+
getById(id: number): Campaign | undefined;
|
|
9
|
+
getByName(name: string): Campaign | undefined;
|
|
10
|
+
listAll(): Campaign[];
|
|
11
|
+
listActive(): Campaign[];
|
|
12
|
+
update(id: number, data: Partial<Campaign>): boolean;
|
|
13
|
+
countAll(): number;
|
|
14
|
+
delete(id: number): boolean;
|
|
15
|
+
}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
export class CampaignRepository {
|
|
2
|
+
db;
|
|
3
|
+
stmts;
|
|
4
|
+
constructor(db) {
|
|
5
|
+
this.db = db;
|
|
6
|
+
this.stmts = {
|
|
7
|
+
create: db.prepare(`
|
|
8
|
+
INSERT INTO campaigns (name, brand, goal, platform)
|
|
9
|
+
VALUES (@name, @brand, @goal, @platform)
|
|
10
|
+
`),
|
|
11
|
+
getById: db.prepare('SELECT * FROM campaigns WHERE id = ?'),
|
|
12
|
+
getByName: db.prepare('SELECT * FROM campaigns WHERE name = ?'),
|
|
13
|
+
listAll: db.prepare('SELECT * FROM campaigns ORDER BY created_at DESC'),
|
|
14
|
+
listActive: db.prepare(`SELECT * FROM campaigns WHERE status = 'active' ORDER BY created_at DESC`),
|
|
15
|
+
countAll: db.prepare('SELECT COUNT(*) as count FROM campaigns'),
|
|
16
|
+
delete: db.prepare('DELETE FROM campaigns WHERE id = ?'),
|
|
17
|
+
};
|
|
18
|
+
}
|
|
19
|
+
create(data) {
|
|
20
|
+
const result = this.stmts.create.run({
|
|
21
|
+
name: data.name,
|
|
22
|
+
brand: data.brand ?? null,
|
|
23
|
+
goal: data.goal ?? null,
|
|
24
|
+
platform: data.platform ?? null,
|
|
25
|
+
});
|
|
26
|
+
return result.lastInsertRowid;
|
|
27
|
+
}
|
|
28
|
+
getById(id) {
|
|
29
|
+
return this.stmts.getById.get(id);
|
|
30
|
+
}
|
|
31
|
+
getByName(name) {
|
|
32
|
+
return this.stmts.getByName.get(name);
|
|
33
|
+
}
|
|
34
|
+
listAll() {
|
|
35
|
+
return this.stmts.listAll.all();
|
|
36
|
+
}
|
|
37
|
+
listActive() {
|
|
38
|
+
return this.stmts.listActive.all();
|
|
39
|
+
}
|
|
40
|
+
update(id, data) {
|
|
41
|
+
const fields = Object.keys(data).filter((key) => key !== 'id' && key !== 'created_at' && data[key] !== undefined);
|
|
42
|
+
if (fields.length === 0)
|
|
43
|
+
return false;
|
|
44
|
+
const setClauses = fields.map((f) => `${f} = @${f}`).join(', ');
|
|
45
|
+
const stmt = this.db.prepare(`UPDATE campaigns SET ${setClauses}, updated_at = datetime('now') WHERE id = @id`);
|
|
46
|
+
const result = stmt.run({ ...data, id });
|
|
47
|
+
return result.changes > 0;
|
|
48
|
+
}
|
|
49
|
+
countAll() {
|
|
50
|
+
const row = this.stmts.countAll.get();
|
|
51
|
+
return row.count;
|
|
52
|
+
}
|
|
53
|
+
delete(id) {
|
|
54
|
+
const result = this.stmts.delete.run(id);
|
|
55
|
+
return result.changes > 0;
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
//# sourceMappingURL=campaign.repository.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"campaign.repository.js","sourceRoot":"","sources":["../../../src/db/repositories/campaign.repository.ts"],"names":[],"mappings":"AAIA,MAAM,OAAO,kBAAkB;IAGT;IAFZ,KAAK,CAA4B;IAEzC,YAAoB,EAAqB;QAArB,OAAE,GAAF,EAAE,CAAmB;QACvC,IAAI,CAAC,KAAK,GAAG;YACX,MAAM,EAAE,EAAE,CAAC,OAAO,CAAC;;;OAGlB,CAAC;YACF,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,sCAAsC,CAAC;YAC3D,SAAS,EAAE,EAAE,CAAC,OAAO,CAAC,wCAAwC,CAAC;YAC/D,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,kDAAkD,CAAC;YACvE,UAAU,EAAE,EAAE,CAAC,OAAO,CAAC,0EAA0E,CAAC;YAClG,QAAQ,EAAE,EAAE,CAAC,OAAO,CAAC,yCAAyC,CAAC;YAC/D,MAAM,EAAE,EAAE,CAAC,OAAO,CAAC,oCAAoC,CAAC;SACzD,CAAC;IACJ,CAAC;IAED,MAAM,CAAC,IAAoB;QACzB,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC;YACnC,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,KAAK,EAAE,IAAI,CAAC,KAAK,IAAI,IAAI;YACzB,IAAI,EAAE,IAAI,CAAC,IAAI,IAAI,IAAI;YACvB,QAAQ,EAAE,IAAI,CAAC,QAAQ,IAAI,IAAI;SAChC,CAAC,CAAC;QACH,OAAO,MAAM,CAAC,eAAyB,CAAC;IAC1C,CAAC;IAED,OAAO,CAAC,EAAU;QAChB,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAyB,CAAC;IAC5D,CAAC;IAED,SAAS,CAAC,IAAY;QACpB,OAAO,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAyB,CAAC;IAChE,CAAC;IAED,OAAO;QACL,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,EAAgB,CAAC;IAChD,CAAC;IAED,UAAU;QACR,OAAO,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,GAAG,EAAgB,CAAC;IACnD,CAAC;IAED,MAAM,CAAC,EAAU,EAAE,IAAuB;QACxC,MAAM,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,MAAM,CACrC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,KAAK,IAAI,IAAI,GAAG,KAAK,YAAY,IAAK,IAAgC,CAAC,GAAG,CAAC,KAAK,SAAS,CACtG,CAAC;QACF,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,KAAK,CAAC;QAEtC,MAAM,UAAU,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAChE,MAAM,IAAI,GAAG,IAAI,CAAC,EAAE,CAAC,OAAO,CAC1B,wBAAwB,UAAU,+CAA+C,CAClF,CAAC;QACF,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,GAAG,IAAI,EAAE,EAAE,EAAE,CAAC,CAAC;QACzC,OAAO,MAAM,CAAC,OAAO,GAAG,CAAC,CAAC;IAC5B,CAAC;IAED,QAAQ;QACN,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,EAAuB,CAAC;QAC3D,OAAO,GAAG,CAAC,KAAK,CAAC;IACnB,CAAC;IAED,MAAM,CAAC,EAAU;QACf,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QACzC,OAAO,MAAM,CAAC,OAAO,GAAG,CAAC,CAAC;IAC5B,CAAC;CACF"}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import type Database from 'better-sqlite3';
|
|
2
|
+
import type { Engagement, EngagementCreate } from '../../types/post.types.js';
|
|
3
|
+
export declare class EngagementRepository {
|
|
4
|
+
private db;
|
|
5
|
+
private stmts;
|
|
6
|
+
constructor(db: Database.Database);
|
|
7
|
+
create(data: EngagementCreate): number;
|
|
8
|
+
getById(id: number): Engagement | undefined;
|
|
9
|
+
getLatestByPost(postId: number): Engagement | undefined;
|
|
10
|
+
listByPost(postId: number): Engagement[];
|
|
11
|
+
topPosts(limit?: number): Array<Engagement & {
|
|
12
|
+
platform: string;
|
|
13
|
+
content: string;
|
|
14
|
+
published_at: string;
|
|
15
|
+
}>;
|
|
16
|
+
avgByPlatform(): Array<{
|
|
17
|
+
platform: string;
|
|
18
|
+
avg_likes: number;
|
|
19
|
+
avg_comments: number;
|
|
20
|
+
avg_shares: number;
|
|
21
|
+
avg_impressions: number;
|
|
22
|
+
avg_clicks: number;
|
|
23
|
+
post_count: number;
|
|
24
|
+
}>;
|
|
25
|
+
delete(id: number): boolean;
|
|
26
|
+
}
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
export class EngagementRepository {
|
|
2
|
+
db;
|
|
3
|
+
stmts;
|
|
4
|
+
constructor(db) {
|
|
5
|
+
this.db = db;
|
|
6
|
+
this.stmts = {
|
|
7
|
+
create: db.prepare(`
|
|
8
|
+
INSERT INTO engagement (post_id, likes, comments, shares, impressions, clicks, saves, reach)
|
|
9
|
+
VALUES (@post_id, @likes, @comments, @shares, @impressions, @clicks, @saves, @reach)
|
|
10
|
+
`),
|
|
11
|
+
getById: db.prepare('SELECT * FROM engagement WHERE id = ?'),
|
|
12
|
+
getByPost: db.prepare('SELECT * FROM engagement WHERE post_id = ? ORDER BY timestamp DESC LIMIT 1'),
|
|
13
|
+
listByPost: db.prepare('SELECT * FROM engagement WHERE post_id = ? ORDER BY timestamp DESC'),
|
|
14
|
+
getLatestPerPost: db.prepare(`
|
|
15
|
+
SELECT e.* FROM engagement e
|
|
16
|
+
INNER JOIN (
|
|
17
|
+
SELECT post_id, MAX(timestamp) as max_ts FROM engagement GROUP BY post_id
|
|
18
|
+
) latest ON e.post_id = latest.post_id AND e.timestamp = latest.max_ts
|
|
19
|
+
ORDER BY (e.likes + e.comments + e.shares + e.impressions) DESC
|
|
20
|
+
LIMIT ?
|
|
21
|
+
`),
|
|
22
|
+
topPosts: db.prepare(`
|
|
23
|
+
SELECT e.*, p.platform, p.content, p.published_at FROM engagement e
|
|
24
|
+
INNER JOIN posts p ON p.id = e.post_id
|
|
25
|
+
INNER JOIN (
|
|
26
|
+
SELECT post_id, MAX(timestamp) as max_ts FROM engagement GROUP BY post_id
|
|
27
|
+
) latest ON e.post_id = latest.post_id AND e.timestamp = latest.max_ts
|
|
28
|
+
WHERE p.status = 'published'
|
|
29
|
+
ORDER BY (e.likes + e.comments * 3 + e.shares * 5 + e.clicks * 2) DESC
|
|
30
|
+
LIMIT ?
|
|
31
|
+
`),
|
|
32
|
+
avgByPlatform: db.prepare(`
|
|
33
|
+
SELECT p.platform,
|
|
34
|
+
AVG(e.likes) as avg_likes,
|
|
35
|
+
AVG(e.comments) as avg_comments,
|
|
36
|
+
AVG(e.shares) as avg_shares,
|
|
37
|
+
AVG(e.impressions) as avg_impressions,
|
|
38
|
+
AVG(e.clicks) as avg_clicks,
|
|
39
|
+
COUNT(DISTINCT e.post_id) as post_count
|
|
40
|
+
FROM engagement e
|
|
41
|
+
INNER JOIN posts p ON p.id = e.post_id
|
|
42
|
+
INNER JOIN (
|
|
43
|
+
SELECT post_id, MAX(timestamp) as max_ts FROM engagement GROUP BY post_id
|
|
44
|
+
) latest ON e.post_id = latest.post_id AND e.timestamp = latest.max_ts
|
|
45
|
+
GROUP BY p.platform
|
|
46
|
+
`),
|
|
47
|
+
delete: db.prepare('DELETE FROM engagement WHERE id = ?'),
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
create(data) {
|
|
51
|
+
const result = this.stmts.create.run({
|
|
52
|
+
post_id: data.post_id,
|
|
53
|
+
likes: data.likes ?? 0,
|
|
54
|
+
comments: data.comments ?? 0,
|
|
55
|
+
shares: data.shares ?? 0,
|
|
56
|
+
impressions: data.impressions ?? 0,
|
|
57
|
+
clicks: data.clicks ?? 0,
|
|
58
|
+
saves: data.saves ?? 0,
|
|
59
|
+
reach: data.reach ?? 0,
|
|
60
|
+
});
|
|
61
|
+
return result.lastInsertRowid;
|
|
62
|
+
}
|
|
63
|
+
getById(id) {
|
|
64
|
+
return this.stmts.getById.get(id);
|
|
65
|
+
}
|
|
66
|
+
getLatestByPost(postId) {
|
|
67
|
+
return this.stmts.getByPost.get(postId);
|
|
68
|
+
}
|
|
69
|
+
listByPost(postId) {
|
|
70
|
+
return this.stmts.listByPost.all(postId);
|
|
71
|
+
}
|
|
72
|
+
topPosts(limit = 10) {
|
|
73
|
+
return this.stmts.topPosts.all(limit);
|
|
74
|
+
}
|
|
75
|
+
avgByPlatform() {
|
|
76
|
+
return this.stmts.avgByPlatform.all();
|
|
77
|
+
}
|
|
78
|
+
delete(id) {
|
|
79
|
+
const result = this.stmts.delete.run(id);
|
|
80
|
+
return result.changes > 0;
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
//# sourceMappingURL=engagement.repository.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"engagement.repository.js","sourceRoot":"","sources":["../../../src/db/repositories/engagement.repository.ts"],"names":[],"mappings":"AAIA,MAAM,OAAO,oBAAoB;IAGX;IAFZ,KAAK,CAA4B;IAEzC,YAAoB,EAAqB;QAArB,OAAE,GAAF,EAAE,CAAmB;QACvC,IAAI,CAAC,KAAK,GAAG;YACX,MAAM,EAAE,EAAE,CAAC,OAAO,CAAC;;;OAGlB,CAAC;YACF,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,uCAAuC,CAAC;YAC5D,SAAS,EAAE,EAAE,CAAC,OAAO,CAAC,4EAA4E,CAAC;YACnG,UAAU,EAAE,EAAE,CAAC,OAAO,CAAC,oEAAoE,CAAC;YAC5F,gBAAgB,EAAE,EAAE,CAAC,OAAO,CAAC;;;;;;;OAO5B,CAAC;YACF,QAAQ,EAAE,EAAE,CAAC,OAAO,CAAC;;;;;;;;;OASpB,CAAC;YACF,aAAa,EAAE,EAAE,CAAC,OAAO,CAAC;;;;;;;;;;;;;;OAczB,CAAC;YACF,MAAM,EAAE,EAAE,CAAC,OAAO,CAAC,qCAAqC,CAAC;SAC1D,CAAC;IACJ,CAAC;IAED,MAAM,CAAC,IAAsB;QAC3B,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC;YACnC,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,KAAK,EAAE,IAAI,CAAC,KAAK,IAAI,CAAC;YACtB,QAAQ,EAAE,IAAI,CAAC,QAAQ,IAAI,CAAC;YAC5B,MAAM,EAAE,IAAI,CAAC,MAAM,IAAI,CAAC;YACxB,WAAW,EAAE,IAAI,CAAC,WAAW,IAAI,CAAC;YAClC,MAAM,EAAE,IAAI,CAAC,MAAM,IAAI,CAAC;YACxB,KAAK,EAAE,IAAI,CAAC,KAAK,IAAI,CAAC;YACtB,KAAK,EAAE,IAAI,CAAC,KAAK,IAAI,CAAC;SACvB,CAAC,CAAC;QACH,OAAO,MAAM,CAAC,eAAyB,CAAC;IAC1C,CAAC;IAED,OAAO,CAAC,EAAU;QAChB,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAA2B,CAAC;IAC9D,CAAC;IAED,eAAe,CAAC,MAAc;QAC5B,OAAO,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,GAAG,CAAC,MAAM,CAA2B,CAAC;IACpE,CAAC;IAED,UAAU,CAAC,MAAc;QACvB,OAAO,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC,MAAM,CAAiB,CAAC;IAC3D,CAAC;IAED,QAAQ,CAAC,QAAgB,EAAE;QACzB,OAAO,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAoF,CAAC;IAC3H,CAAC;IAED,aAAa;QASX,OAAO,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,GAAG,EAQjC,CAAC;IACL,CAAC;IAED,MAAM,CAAC,EAAU;QACf,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QACzC,OAAO,MAAM,CAAC,OAAO,GAAG,CAAC,CAAC;IAC5B,CAAC;CACF"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import type Database from 'better-sqlite3';
|
|
2
|
+
import type { Insight, InsightCreate } from '../../types/post.types.js';
|
|
3
|
+
export declare class InsightRepository {
|
|
4
|
+
private db;
|
|
5
|
+
private stmts;
|
|
6
|
+
constructor(db: Database.Database);
|
|
7
|
+
create(data: InsightCreate): number;
|
|
8
|
+
getById(id: number): Insight | undefined;
|
|
9
|
+
listActive(limit?: number): Insight[];
|
|
10
|
+
listAll(limit?: number): Insight[];
|
|
11
|
+
listByType(type: string, limit?: number): Insight[];
|
|
12
|
+
listByCampaign(campaignId: number): Insight[];
|
|
13
|
+
deactivate(id: number): void;
|
|
14
|
+
expireOld(): number;
|
|
15
|
+
countActive(): number;
|
|
16
|
+
countAll(): number;
|
|
17
|
+
delete(id: number): boolean;
|
|
18
|
+
}
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
export class InsightRepository {
|
|
2
|
+
db;
|
|
3
|
+
stmts;
|
|
4
|
+
constructor(db) {
|
|
5
|
+
this.db = db;
|
|
6
|
+
this.stmts = {
|
|
7
|
+
create: db.prepare(`
|
|
8
|
+
INSERT INTO insights (type, title, description, confidence, priority, campaign_id, expires_at)
|
|
9
|
+
VALUES (@type, @title, @description, @confidence, @priority, @campaign_id, @expires_at)
|
|
10
|
+
`),
|
|
11
|
+
getById: db.prepare('SELECT * FROM insights WHERE id = ?'),
|
|
12
|
+
listActive: db.prepare(`
|
|
13
|
+
SELECT * FROM insights
|
|
14
|
+
WHERE active = 1 AND (expires_at IS NULL OR expires_at > datetime('now'))
|
|
15
|
+
ORDER BY priority DESC, confidence DESC
|
|
16
|
+
LIMIT ?
|
|
17
|
+
`),
|
|
18
|
+
listAll: db.prepare('SELECT * FROM insights ORDER BY created_at DESC LIMIT ?'),
|
|
19
|
+
listByType: db.prepare(`
|
|
20
|
+
SELECT * FROM insights WHERE type = ? AND active = 1
|
|
21
|
+
ORDER BY priority DESC LIMIT ?
|
|
22
|
+
`),
|
|
23
|
+
listByCampaign: db.prepare(`
|
|
24
|
+
SELECT * FROM insights WHERE campaign_id = ? AND active = 1
|
|
25
|
+
ORDER BY priority DESC
|
|
26
|
+
`),
|
|
27
|
+
countActive: db.prepare(`
|
|
28
|
+
SELECT COUNT(*) as count FROM insights
|
|
29
|
+
WHERE active = 1 AND (expires_at IS NULL OR expires_at > datetime('now'))
|
|
30
|
+
`),
|
|
31
|
+
countAll: db.prepare('SELECT COUNT(*) as count FROM insights'),
|
|
32
|
+
deactivate: db.prepare('UPDATE insights SET active = 0 WHERE id = ?'),
|
|
33
|
+
expireOld: db.prepare(`
|
|
34
|
+
UPDATE insights SET active = 0
|
|
35
|
+
WHERE active = 1 AND expires_at IS NOT NULL AND expires_at <= datetime('now')
|
|
36
|
+
`),
|
|
37
|
+
delete: db.prepare('DELETE FROM insights WHERE id = ?'),
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
create(data) {
|
|
41
|
+
const result = this.stmts.create.run({
|
|
42
|
+
type: data.type,
|
|
43
|
+
title: data.title,
|
|
44
|
+
description: data.description,
|
|
45
|
+
confidence: data.confidence ?? 0.5,
|
|
46
|
+
priority: data.priority ?? 0,
|
|
47
|
+
campaign_id: data.campaign_id ?? null,
|
|
48
|
+
expires_at: data.expires_at ?? null,
|
|
49
|
+
});
|
|
50
|
+
return result.lastInsertRowid;
|
|
51
|
+
}
|
|
52
|
+
getById(id) {
|
|
53
|
+
return this.stmts.getById.get(id);
|
|
54
|
+
}
|
|
55
|
+
listActive(limit = 20) {
|
|
56
|
+
return this.stmts.listActive.all(limit);
|
|
57
|
+
}
|
|
58
|
+
listAll(limit = 50) {
|
|
59
|
+
return this.stmts.listAll.all(limit);
|
|
60
|
+
}
|
|
61
|
+
listByType(type, limit = 20) {
|
|
62
|
+
return this.stmts.listByType.all(type, limit);
|
|
63
|
+
}
|
|
64
|
+
listByCampaign(campaignId) {
|
|
65
|
+
return this.stmts.listByCampaign.all(campaignId);
|
|
66
|
+
}
|
|
67
|
+
deactivate(id) {
|
|
68
|
+
this.stmts.deactivate.run(id);
|
|
69
|
+
}
|
|
70
|
+
expireOld() {
|
|
71
|
+
const result = this.stmts.expireOld.run();
|
|
72
|
+
return result.changes;
|
|
73
|
+
}
|
|
74
|
+
countActive() {
|
|
75
|
+
const row = this.stmts.countActive.get();
|
|
76
|
+
return row.count;
|
|
77
|
+
}
|
|
78
|
+
countAll() {
|
|
79
|
+
const row = this.stmts.countAll.get();
|
|
80
|
+
return row.count;
|
|
81
|
+
}
|
|
82
|
+
delete(id) {
|
|
83
|
+
const result = this.stmts.delete.run(id);
|
|
84
|
+
return result.changes > 0;
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
//# sourceMappingURL=insight.repository.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"insight.repository.js","sourceRoot":"","sources":["../../../src/db/repositories/insight.repository.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import type Database from 'better-sqlite3';
|
|
2
|
+
import type { Post, PostCreate } from '../../types/post.types.js';
|
|
3
|
+
export declare class PostRepository {
|
|
4
|
+
private db;
|
|
5
|
+
private stmts;
|
|
6
|
+
constructor(db: Database.Database);
|
|
7
|
+
create(data: PostCreate): number;
|
|
8
|
+
getById(id: number): Post | undefined;
|
|
9
|
+
getByFingerprint(fingerprint: string): Post | undefined;
|
|
10
|
+
listAll(limit?: number): Post[];
|
|
11
|
+
listByPlatform(platform: string, limit?: number): Post[];
|
|
12
|
+
listByCampaign(campaignId: number, limit?: number): Post[];
|
|
13
|
+
listPublished(limit?: number): Post[];
|
|
14
|
+
recentPublished(sinceDate: string): Post[];
|
|
15
|
+
update(id: number, data: Partial<Post>): boolean;
|
|
16
|
+
delete(id: number): boolean;
|
|
17
|
+
search(query: string, limit?: number): Post[];
|
|
18
|
+
countAll(): number;
|
|
19
|
+
countByPlatform(): Record<string, number>;
|
|
20
|
+
countByStatus(): Record<string, number>;
|
|
21
|
+
}
|
|
@@ -0,0 +1,105 @@
|
|
|
1
|
+
import { sha256 } from '../../utils/hash.js';
|
|
2
|
+
export class PostRepository {
|
|
3
|
+
db;
|
|
4
|
+
stmts;
|
|
5
|
+
constructor(db) {
|
|
6
|
+
this.db = db;
|
|
7
|
+
this.stmts = {
|
|
8
|
+
create: db.prepare(`
|
|
9
|
+
INSERT INTO posts (campaign_id, platform, content, format, hashtags, url, published_at, fingerprint, status)
|
|
10
|
+
VALUES (@campaign_id, @platform, @content, @format, @hashtags, @url, @published_at, @fingerprint, @status)
|
|
11
|
+
`),
|
|
12
|
+
getById: db.prepare('SELECT * FROM posts WHERE id = ?'),
|
|
13
|
+
getByFingerprint: db.prepare('SELECT * FROM posts WHERE fingerprint = ?'),
|
|
14
|
+
listAll: db.prepare('SELECT * FROM posts ORDER BY created_at DESC LIMIT ?'),
|
|
15
|
+
listByPlatform: db.prepare('SELECT * FROM posts WHERE platform = ? ORDER BY created_at DESC LIMIT ?'),
|
|
16
|
+
listByCampaign: db.prepare('SELECT * FROM posts WHERE campaign_id = ? ORDER BY created_at DESC LIMIT ?'),
|
|
17
|
+
listPublished: db.prepare(`SELECT * FROM posts WHERE status = 'published' ORDER BY published_at DESC LIMIT ?`),
|
|
18
|
+
countAll: db.prepare('SELECT COUNT(*) as count FROM posts'),
|
|
19
|
+
countByPlatform: db.prepare('SELECT platform, COUNT(*) as count FROM posts GROUP BY platform'),
|
|
20
|
+
countByStatus: db.prepare('SELECT status, COUNT(*) as count FROM posts GROUP BY status'),
|
|
21
|
+
delete: db.prepare('DELETE FROM posts WHERE id = ?'),
|
|
22
|
+
search: db.prepare(`
|
|
23
|
+
SELECT p.* FROM posts p
|
|
24
|
+
JOIN posts_fts f ON f.rowid = p.id
|
|
25
|
+
WHERE posts_fts MATCH ?
|
|
26
|
+
ORDER BY rank
|
|
27
|
+
LIMIT ?
|
|
28
|
+
`),
|
|
29
|
+
recentPublished: db.prepare(`
|
|
30
|
+
SELECT * FROM posts WHERE status = 'published' AND published_at > ? ORDER BY published_at DESC
|
|
31
|
+
`),
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
create(data) {
|
|
35
|
+
const fingerprint = sha256(`${data.platform}:${data.content}`);
|
|
36
|
+
const result = this.stmts.create.run({
|
|
37
|
+
campaign_id: data.campaign_id ?? null,
|
|
38
|
+
platform: data.platform,
|
|
39
|
+
content: data.content,
|
|
40
|
+
format: data.format ?? 'text',
|
|
41
|
+
hashtags: data.hashtags ?? null,
|
|
42
|
+
url: data.url ?? null,
|
|
43
|
+
published_at: data.published_at ?? null,
|
|
44
|
+
fingerprint,
|
|
45
|
+
status: data.status ?? 'draft',
|
|
46
|
+
});
|
|
47
|
+
return result.lastInsertRowid;
|
|
48
|
+
}
|
|
49
|
+
getById(id) {
|
|
50
|
+
return this.stmts.getById.get(id);
|
|
51
|
+
}
|
|
52
|
+
getByFingerprint(fingerprint) {
|
|
53
|
+
return this.stmts.getByFingerprint.get(fingerprint);
|
|
54
|
+
}
|
|
55
|
+
listAll(limit = 50) {
|
|
56
|
+
return this.stmts.listAll.all(limit);
|
|
57
|
+
}
|
|
58
|
+
listByPlatform(platform, limit = 50) {
|
|
59
|
+
return this.stmts.listByPlatform.all(platform, limit);
|
|
60
|
+
}
|
|
61
|
+
listByCampaign(campaignId, limit = 50) {
|
|
62
|
+
return this.stmts.listByCampaign.all(campaignId, limit);
|
|
63
|
+
}
|
|
64
|
+
listPublished(limit = 50) {
|
|
65
|
+
return this.stmts.listPublished.all(limit);
|
|
66
|
+
}
|
|
67
|
+
recentPublished(sinceDate) {
|
|
68
|
+
return this.stmts.recentPublished.all(sinceDate);
|
|
69
|
+
}
|
|
70
|
+
update(id, data) {
|
|
71
|
+
const fields = Object.keys(data).filter((key) => key !== 'id' && key !== 'created_at' && data[key] !== undefined);
|
|
72
|
+
if (fields.length === 0)
|
|
73
|
+
return false;
|
|
74
|
+
const setClauses = fields.map((f) => `${f} = @${f}`).join(', ');
|
|
75
|
+
const stmt = this.db.prepare(`UPDATE posts SET ${setClauses}, updated_at = datetime('now') WHERE id = @id`);
|
|
76
|
+
const result = stmt.run({ ...data, id });
|
|
77
|
+
return result.changes > 0;
|
|
78
|
+
}
|
|
79
|
+
delete(id) {
|
|
80
|
+
const result = this.stmts.delete.run(id);
|
|
81
|
+
return result.changes > 0;
|
|
82
|
+
}
|
|
83
|
+
search(query, limit = 20) {
|
|
84
|
+
return this.stmts.search.all(query, limit);
|
|
85
|
+
}
|
|
86
|
+
countAll() {
|
|
87
|
+
const row = this.stmts.countAll.get();
|
|
88
|
+
return row.count;
|
|
89
|
+
}
|
|
90
|
+
countByPlatform() {
|
|
91
|
+
const rows = this.stmts.countByPlatform.all();
|
|
92
|
+
const result = {};
|
|
93
|
+
for (const row of rows)
|
|
94
|
+
result[row.platform] = row.count;
|
|
95
|
+
return result;
|
|
96
|
+
}
|
|
97
|
+
countByStatus() {
|
|
98
|
+
const rows = this.stmts.countByStatus.all();
|
|
99
|
+
const result = {};
|
|
100
|
+
for (const row of rows)
|
|
101
|
+
result[row.status] = row.count;
|
|
102
|
+
return result;
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
//# sourceMappingURL=post.repository.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"post.repository.js","sourceRoot":"","sources":["../../../src/db/repositories/post.repository.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import type Database from 'better-sqlite3';
|
|
2
|
+
import type { MarketingRule, RuleCreate } from '../../types/post.types.js';
|
|
3
|
+
export declare class RuleRepository {
|
|
4
|
+
private db;
|
|
5
|
+
private stmts;
|
|
6
|
+
constructor(db: Database.Database);
|
|
7
|
+
create(data: RuleCreate): number;
|
|
8
|
+
getById(id: number): MarketingRule | undefined;
|
|
9
|
+
listAll(): MarketingRule[];
|
|
10
|
+
listActive(): MarketingRule[];
|
|
11
|
+
update(id: number, data: Partial<MarketingRule>): boolean;
|
|
12
|
+
incrementTrigger(id: number, success: boolean): void;
|
|
13
|
+
countAll(): number;
|
|
14
|
+
countActive(): number;
|
|
15
|
+
delete(id: number): boolean;
|
|
16
|
+
}
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
export class RuleRepository {
|
|
2
|
+
db;
|
|
3
|
+
stmts;
|
|
4
|
+
constructor(db) {
|
|
5
|
+
this.db = db;
|
|
6
|
+
this.stmts = {
|
|
7
|
+
create: db.prepare(`
|
|
8
|
+
INSERT INTO marketing_rules (pattern, recommendation, confidence)
|
|
9
|
+
VALUES (@pattern, @recommendation, @confidence)
|
|
10
|
+
`),
|
|
11
|
+
getById: db.prepare('SELECT * FROM marketing_rules WHERE id = ?'),
|
|
12
|
+
listAll: db.prepare('SELECT * FROM marketing_rules ORDER BY confidence DESC'),
|
|
13
|
+
listActive: db.prepare(`SELECT * FROM marketing_rules WHERE active = 1 ORDER BY confidence DESC`),
|
|
14
|
+
countAll: db.prepare('SELECT COUNT(*) as count FROM marketing_rules'),
|
|
15
|
+
countActive: db.prepare(`SELECT COUNT(*) as count FROM marketing_rules WHERE active = 1`),
|
|
16
|
+
delete: db.prepare('DELETE FROM marketing_rules WHERE id = ?'),
|
|
17
|
+
};
|
|
18
|
+
}
|
|
19
|
+
create(data) {
|
|
20
|
+
const result = this.stmts.create.run({
|
|
21
|
+
pattern: data.pattern,
|
|
22
|
+
recommendation: data.recommendation,
|
|
23
|
+
confidence: data.confidence ?? 0.5,
|
|
24
|
+
});
|
|
25
|
+
return result.lastInsertRowid;
|
|
26
|
+
}
|
|
27
|
+
getById(id) {
|
|
28
|
+
return this.stmts.getById.get(id);
|
|
29
|
+
}
|
|
30
|
+
listAll() {
|
|
31
|
+
return this.stmts.listAll.all();
|
|
32
|
+
}
|
|
33
|
+
listActive() {
|
|
34
|
+
return this.stmts.listActive.all();
|
|
35
|
+
}
|
|
36
|
+
update(id, data) {
|
|
37
|
+
const fields = Object.keys(data).filter((key) => key !== 'id' && key !== 'created_at' && data[key] !== undefined);
|
|
38
|
+
if (fields.length === 0)
|
|
39
|
+
return false;
|
|
40
|
+
const setClauses = fields.map((f) => `${f} = @${f}`).join(', ');
|
|
41
|
+
const stmt = this.db.prepare(`UPDATE marketing_rules SET ${setClauses}, updated_at = datetime('now') WHERE id = @id`);
|
|
42
|
+
const result = stmt.run({ ...data, id });
|
|
43
|
+
return result.changes > 0;
|
|
44
|
+
}
|
|
45
|
+
incrementTrigger(id, success) {
|
|
46
|
+
const rule = this.getById(id);
|
|
47
|
+
if (!rule)
|
|
48
|
+
return;
|
|
49
|
+
const data = {
|
|
50
|
+
trigger_count: rule.trigger_count + 1,
|
|
51
|
+
id,
|
|
52
|
+
};
|
|
53
|
+
if (success)
|
|
54
|
+
data['success_count'] = rule.success_count + 1;
|
|
55
|
+
const stmt = this.db.prepare(`UPDATE marketing_rules SET trigger_count = @trigger_count${success ? ', success_count = @success_count' : ''}, updated_at = datetime('now') WHERE id = @id`);
|
|
56
|
+
stmt.run(data);
|
|
57
|
+
}
|
|
58
|
+
countAll() {
|
|
59
|
+
const row = this.stmts.countAll.get();
|
|
60
|
+
return row.count;
|
|
61
|
+
}
|
|
62
|
+
countActive() {
|
|
63
|
+
const row = this.stmts.countActive.get();
|
|
64
|
+
return row.count;
|
|
65
|
+
}
|
|
66
|
+
delete(id) {
|
|
67
|
+
const result = this.stmts.delete.run(id);
|
|
68
|
+
return result.changes > 0;
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
//# sourceMappingURL=rule.repository.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"rule.repository.js","sourceRoot":"","sources":["../../../src/db/repositories/rule.repository.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import type Database from 'better-sqlite3';
|
|
2
|
+
import type { Strategy, StrategyCreate } from '../../types/post.types.js';
|
|
3
|
+
export declare class StrategyRepository {
|
|
4
|
+
private db;
|
|
5
|
+
private stmts;
|
|
6
|
+
constructor(db: Database.Database);
|
|
7
|
+
create(data: StrategyCreate): number;
|
|
8
|
+
getById(id: number): Strategy | undefined;
|
|
9
|
+
listAll(limit?: number): Strategy[];
|
|
10
|
+
listByPost(postId: number): Strategy[];
|
|
11
|
+
search(query: string, limit?: number): Strategy[];
|
|
12
|
+
topByConfidence(minConfidence?: number, limit?: number): Strategy[];
|
|
13
|
+
update(id: number, data: Partial<Strategy>): boolean;
|
|
14
|
+
countAll(): number;
|
|
15
|
+
delete(id: number): boolean;
|
|
16
|
+
}
|