@timmeck/marketing-brain 0.2.1 → 0.4.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 (160) hide show
  1. package/Dockerfile +22 -0
  2. package/README.md +20 -3
  3. package/dashboard.html +6 -0
  4. package/dist/cli/colors.d.ts +11 -24
  5. package/dist/cli/colors.js +3 -46
  6. package/dist/cli/colors.js.map +1 -1
  7. package/dist/cli/commands/dashboard.js +15 -0
  8. package/dist/cli/commands/dashboard.js.map +1 -1
  9. package/dist/cli/commands/peers.d.ts +2 -0
  10. package/dist/cli/commands/peers.js +38 -0
  11. package/dist/cli/commands/peers.js.map +1 -0
  12. package/dist/config.js +2 -29
  13. package/dist/config.js.map +1 -1
  14. package/dist/db/connection.d.ts +1 -2
  15. package/dist/db/connection.js +1 -18
  16. package/dist/db/connection.js.map +1 -1
  17. package/dist/hooks/post-tool-use.d.ts +2 -0
  18. package/dist/hooks/post-tool-use.js +182 -0
  19. package/dist/hooks/post-tool-use.js.map +1 -0
  20. package/dist/index.js +2 -0
  21. package/dist/index.js.map +1 -1
  22. package/dist/ipc/client.d.ts +1 -13
  23. package/dist/ipc/client.js +1 -92
  24. package/dist/ipc/client.js.map +1 -1
  25. package/dist/ipc/protocol.d.ts +1 -8
  26. package/dist/ipc/protocol.js +1 -28
  27. package/dist/ipc/protocol.js.map +1 -1
  28. package/dist/ipc/router.d.ts +2 -0
  29. package/dist/ipc/router.js +30 -0
  30. package/dist/ipc/router.js.map +1 -1
  31. package/dist/ipc/server.d.ts +1 -14
  32. package/dist/ipc/server.js +1 -129
  33. package/dist/ipc/server.js.map +1 -1
  34. package/dist/learning/confidence-scorer.d.ts +1 -5
  35. package/dist/learning/confidence-scorer.js +2 -13
  36. package/dist/learning/confidence-scorer.js.map +1 -1
  37. package/dist/learning/learning-engine.d.ts +2 -5
  38. package/dist/learning/learning-engine.js +3 -20
  39. package/dist/learning/learning-engine.js.map +1 -1
  40. package/dist/marketing-core.d.ts +2 -0
  41. package/dist/marketing-core.js +22 -2
  42. package/dist/marketing-core.js.map +1 -1
  43. package/dist/mcp/server.js +5 -60
  44. package/dist/mcp/server.js.map +1 -1
  45. package/dist/mcp/tools.js +29 -0
  46. package/dist/mcp/tools.js.map +1 -1
  47. package/dist/research/research-engine.d.ts +2 -5
  48. package/dist/research/research-engine.js +3 -24
  49. package/dist/research/research-engine.js.map +1 -1
  50. package/dist/services/analytics.service.d.ts +1 -1
  51. package/dist/services/synapse.service.d.ts +10 -10
  52. package/dist/synapses/activation.d.ts +2 -13
  53. package/dist/synapses/activation.js +2 -49
  54. package/dist/synapses/activation.js.map +1 -1
  55. package/dist/synapses/decay.d.ts +2 -11
  56. package/dist/synapses/decay.js +2 -26
  57. package/dist/synapses/decay.js.map +1 -1
  58. package/dist/synapses/hebbian.d.ts +2 -13
  59. package/dist/synapses/hebbian.js +2 -34
  60. package/dist/synapses/hebbian.js.map +1 -1
  61. package/dist/synapses/pathfinder.d.ts +2 -14
  62. package/dist/synapses/pathfinder.js +2 -49
  63. package/dist/synapses/pathfinder.js.map +1 -1
  64. package/dist/synapses/synapse-manager.d.ts +7 -23
  65. package/dist/synapses/synapse-manager.js +6 -63
  66. package/dist/synapses/synapse-manager.js.map +1 -1
  67. package/dist/types/ipc.types.d.ts +1 -11
  68. package/dist/utils/events.d.ts +4 -8
  69. package/dist/utils/events.js +2 -14
  70. package/dist/utils/events.js.map +1 -1
  71. package/dist/utils/hash.d.ts +1 -1
  72. package/dist/utils/hash.js +1 -4
  73. package/dist/utils/hash.js.map +1 -1
  74. package/dist/utils/logger.d.ts +3 -2
  75. package/dist/utils/logger.js +8 -35
  76. package/dist/utils/logger.js.map +1 -1
  77. package/dist/utils/paths.d.ts +2 -1
  78. package/dist/utils/paths.js +4 -13
  79. package/dist/utils/paths.js.map +1 -1
  80. package/eslint.config.js +14 -0
  81. package/package.json +9 -2
  82. package/.github/FUNDING.yml +0 -1
  83. package/.github/workflows/ci.yml +0 -27
  84. package/.mcp.json +0 -9
  85. package/src/api/server.ts +0 -86
  86. package/src/cli/colors.ts +0 -59
  87. package/src/cli/commands/campaign.ts +0 -66
  88. package/src/cli/commands/config.ts +0 -168
  89. package/src/cli/commands/dashboard.ts +0 -165
  90. package/src/cli/commands/doctor.ts +0 -110
  91. package/src/cli/commands/export.ts +0 -40
  92. package/src/cli/commands/import.ts +0 -84
  93. package/src/cli/commands/insights.ts +0 -44
  94. package/src/cli/commands/learn.ts +0 -24
  95. package/src/cli/commands/network.ts +0 -71
  96. package/src/cli/commands/post.ts +0 -47
  97. package/src/cli/commands/query.ts +0 -108
  98. package/src/cli/commands/rules.ts +0 -27
  99. package/src/cli/commands/start.ts +0 -100
  100. package/src/cli/commands/status.ts +0 -73
  101. package/src/cli/commands/stop.ts +0 -33
  102. package/src/cli/commands/suggest.ts +0 -64
  103. package/src/cli/ipc-helper.ts +0 -22
  104. package/src/cli/update-check.ts +0 -63
  105. package/src/config.ts +0 -110
  106. package/src/dashboard/renderer.ts +0 -136
  107. package/src/dashboard/server.ts +0 -140
  108. package/src/db/connection.ts +0 -22
  109. package/src/db/migrations/001_core_schema.ts +0 -63
  110. package/src/db/migrations/002_learning_schema.ts +0 -46
  111. package/src/db/migrations/003_synapse_schema.ts +0 -27
  112. package/src/db/migrations/004_insights_schema.ts +0 -38
  113. package/src/db/migrations/005_fts_indexes.ts +0 -77
  114. package/src/db/migrations/index.ts +0 -62
  115. package/src/db/repositories/audience.repository.ts +0 -53
  116. package/src/db/repositories/campaign.repository.ts +0 -72
  117. package/src/db/repositories/engagement.repository.ts +0 -108
  118. package/src/db/repositories/insight.repository.ts +0 -100
  119. package/src/db/repositories/post.repository.ts +0 -123
  120. package/src/db/repositories/rule.repository.ts +0 -87
  121. package/src/db/repositories/strategy.repository.ts +0 -82
  122. package/src/db/repositories/synapse.repository.ts +0 -148
  123. package/src/db/repositories/template.repository.ts +0 -76
  124. package/src/index.ts +0 -69
  125. package/src/ipc/__tests__/protocol.test.ts +0 -153
  126. package/src/ipc/client.ts +0 -110
  127. package/src/ipc/protocol.ts +0 -35
  128. package/src/ipc/router.ts +0 -126
  129. package/src/ipc/server.ts +0 -140
  130. package/src/learning/confidence-scorer.ts +0 -36
  131. package/src/learning/learning-engine.ts +0 -254
  132. package/src/marketing-core.ts +0 -285
  133. package/src/mcp/server.ts +0 -72
  134. package/src/mcp/tools.ts +0 -216
  135. package/src/research/research-engine.ts +0 -226
  136. package/src/services/analytics.service.ts +0 -73
  137. package/src/services/audience.service.ts +0 -40
  138. package/src/services/campaign.service.ts +0 -80
  139. package/src/services/insight.service.ts +0 -54
  140. package/src/services/post.service.ts +0 -116
  141. package/src/services/rule.service.ts +0 -90
  142. package/src/services/strategy.service.ts +0 -53
  143. package/src/services/synapse.service.ts +0 -32
  144. package/src/services/template.service.ts +0 -50
  145. package/src/synapses/activation.ts +0 -80
  146. package/src/synapses/decay.ts +0 -38
  147. package/src/synapses/hebbian.ts +0 -68
  148. package/src/synapses/pathfinder.ts +0 -81
  149. package/src/synapses/synapse-manager.ts +0 -115
  150. package/src/types/config.types.ts +0 -79
  151. package/src/types/ipc.types.ts +0 -8
  152. package/src/types/post.types.ts +0 -156
  153. package/src/types/synapse.types.ts +0 -43
  154. package/src/utils/__tests__/hash.test.ts +0 -39
  155. package/src/utils/__tests__/paths.test.ts +0 -70
  156. package/src/utils/events.ts +0 -44
  157. package/src/utils/hash.ts +0 -5
  158. package/src/utils/logger.ts +0 -48
  159. package/src/utils/paths.ts +0 -19
  160. package/tsconfig.json +0 -18
@@ -1,156 +0,0 @@
1
- export type Platform = 'x' | 'reddit' | 'linkedin' | 'bluesky' | 'mastodon' | 'threads' | 'other';
2
- export type PostFormat = 'text' | 'image' | 'video' | 'carousel' | 'thread' | 'article' | 'poll';
3
- export type PostStatus = 'draft' | 'scheduled' | 'published' | 'archived';
4
-
5
- export interface Post {
6
- id: number;
7
- campaign_id: number | null;
8
- platform: Platform;
9
- content: string;
10
- format: PostFormat;
11
- hashtags: string | null;
12
- url: string | null;
13
- published_at: string | null;
14
- fingerprint: string;
15
- status: PostStatus;
16
- created_at: string;
17
- updated_at: string;
18
- }
19
-
20
- export interface PostCreate {
21
- campaign_id?: number | null;
22
- platform: Platform;
23
- content: string;
24
- format?: PostFormat;
25
- hashtags?: string;
26
- url?: string;
27
- published_at?: string;
28
- status?: PostStatus;
29
- }
30
-
31
- export interface Engagement {
32
- id: number;
33
- post_id: number;
34
- timestamp: string;
35
- likes: number;
36
- comments: number;
37
- shares: number;
38
- impressions: number;
39
- clicks: number;
40
- saves: number;
41
- reach: number;
42
- }
43
-
44
- export interface EngagementCreate {
45
- post_id: number;
46
- likes?: number;
47
- comments?: number;
48
- shares?: number;
49
- impressions?: number;
50
- clicks?: number;
51
- saves?: number;
52
- reach?: number;
53
- }
54
-
55
- export interface Campaign {
56
- id: number;
57
- name: string;
58
- brand: string | null;
59
- goal: string | null;
60
- platform: string | null;
61
- status: string;
62
- created_at: string;
63
- updated_at: string;
64
- }
65
-
66
- export interface CampaignCreate {
67
- name: string;
68
- brand?: string;
69
- goal?: string;
70
- platform?: string;
71
- }
72
-
73
- export interface Audience {
74
- id: number;
75
- name: string;
76
- platform: Platform | null;
77
- demographics: string | null;
78
- interests: string | null;
79
- created_at: string;
80
- }
81
-
82
- export interface Strategy {
83
- id: number;
84
- post_id: number | null;
85
- description: string;
86
- approach: string | null;
87
- outcome: string | null;
88
- confidence: number;
89
- created_at: string;
90
- }
91
-
92
- export interface StrategyCreate {
93
- post_id?: number;
94
- description: string;
95
- approach?: string;
96
- outcome?: string;
97
- }
98
-
99
- export interface ContentTemplate {
100
- id: number;
101
- name: string;
102
- structure: string;
103
- example: string | null;
104
- platform: Platform | null;
105
- avg_engagement: number;
106
- use_count: number;
107
- created_at: string;
108
- }
109
-
110
- export interface ContentTemplateCreate {
111
- name: string;
112
- structure: string;
113
- example?: string;
114
- platform?: Platform;
115
- }
116
-
117
- export interface MarketingRule {
118
- id: number;
119
- pattern: string;
120
- recommendation: string;
121
- confidence: number;
122
- trigger_count: number;
123
- success_count: number;
124
- active: number;
125
- created_at: string;
126
- updated_at: string;
127
- }
128
-
129
- export interface RuleCreate {
130
- pattern: string;
131
- recommendation: string;
132
- confidence?: number;
133
- }
134
-
135
- export interface Insight {
136
- id: number;
137
- type: string;
138
- title: string;
139
- description: string;
140
- confidence: number;
141
- priority: number;
142
- campaign_id: number | null;
143
- active: number;
144
- expires_at: string | null;
145
- created_at: string;
146
- }
147
-
148
- export interface InsightCreate {
149
- type: string;
150
- title: string;
151
- description: string;
152
- confidence?: number;
153
- priority?: number;
154
- campaign_id?: number | null;
155
- expires_at?: string;
156
- }
@@ -1,43 +0,0 @@
1
- export type NodeType =
2
- | 'post'
3
- | 'campaign'
4
- | 'strategy'
5
- | 'template'
6
- | 'rule'
7
- | 'audience'
8
- | 'insight';
9
-
10
- export type SynapseType =
11
- | 'belongs_to'
12
- | 'similar_to'
13
- | 'engages_with'
14
- | 'improves'
15
- | 'prevents'
16
- | 'recommends'
17
- | 'generated_from'
18
- | 'cross_promotes'
19
- | 'informs'
20
- | 'co_occurs';
21
-
22
- export interface SynapseRecord {
23
- id: number;
24
- source_type: NodeType;
25
- source_id: number;
26
- target_type: NodeType;
27
- target_id: number;
28
- synapse_type: SynapseType;
29
- weight: number;
30
- activation_count: number;
31
- last_activated_at: string;
32
- metadata: string | null;
33
- created_at: string;
34
- updated_at: string;
35
- }
36
-
37
- export interface NetworkStats {
38
- totalNodes: number;
39
- totalSynapses: number;
40
- avgWeight: number;
41
- nodesByType: Record<NodeType, number>;
42
- synapsesByType: Record<SynapseType, number>;
43
- }
@@ -1,39 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { sha256 } from '../hash.js';
3
-
4
- describe('sha256', () => {
5
- it('returns a 64-character hex string', () => {
6
- const result = sha256('hello');
7
- expect(result).toHaveLength(64);
8
- expect(result).toMatch(/^[0-9a-f]{64}$/);
9
- });
10
-
11
- it('produces correct hash for known input', () => {
12
- // SHA-256 of "hello" is well-known
13
- expect(sha256('hello')).toBe(
14
- '2cf24dba5fb0a30e26e83b2ac5b9e29e1b161e5c1fa7425e73043362938b9824'
15
- );
16
- });
17
-
18
- it('produces different hashes for different inputs', () => {
19
- expect(sha256('foo')).not.toBe(sha256('bar'));
20
- });
21
-
22
- it('produces the same hash for the same input', () => {
23
- expect(sha256('deterministic')).toBe(sha256('deterministic'));
24
- });
25
-
26
- it('handles empty string', () => {
27
- const result = sha256('');
28
- expect(result).toHaveLength(64);
29
- expect(result).toBe(
30
- 'e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855'
31
- );
32
- });
33
-
34
- it('handles unicode input', () => {
35
- const result = sha256('Hallo Welt! 🚀');
36
- expect(result).toHaveLength(64);
37
- expect(result).toMatch(/^[0-9a-f]{64}$/);
38
- });
39
- });
@@ -1,70 +0,0 @@
1
- import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
- import path from 'node:path';
3
- import os from 'node:os';
4
- import { normalizePath, getDataDir, getPipeName } from '../paths.js';
5
-
6
- describe('normalizePath', () => {
7
- it('converts backslashes to forward slashes', () => {
8
- expect(normalizePath('C:\\Users\\test\\file.txt')).toBe('C:/Users/test/file.txt');
9
- });
10
-
11
- it('leaves forward slashes unchanged', () => {
12
- expect(normalizePath('/home/user/file.txt')).toBe('/home/user/file.txt');
13
- });
14
-
15
- it('handles mixed separators', () => {
16
- expect(normalizePath('src\\utils/hash.ts')).toBe('src/utils/hash.ts');
17
- });
18
-
19
- it('handles empty string', () => {
20
- expect(normalizePath('')).toBe('');
21
- });
22
-
23
- it('handles path with no separators', () => {
24
- expect(normalizePath('file.txt')).toBe('file.txt');
25
- });
26
- });
27
-
28
- describe('getDataDir', () => {
29
- const originalEnv = process.env['MARKETING_BRAIN_DATA_DIR'];
30
-
31
- afterEach(() => {
32
- if (originalEnv !== undefined) {
33
- process.env['MARKETING_BRAIN_DATA_DIR'] = originalEnv;
34
- } else {
35
- delete process.env['MARKETING_BRAIN_DATA_DIR'];
36
- }
37
- });
38
-
39
- it('returns env-based directory when MARKETING_BRAIN_DATA_DIR is set', () => {
40
- process.env['MARKETING_BRAIN_DATA_DIR'] = '/custom/data';
41
- const result = getDataDir();
42
- expect(result).toBe(path.resolve('/custom/data'));
43
- });
44
-
45
- it('returns homedir-based directory when env is not set', () => {
46
- delete process.env['MARKETING_BRAIN_DATA_DIR'];
47
- const result = getDataDir();
48
- expect(result).toBe(path.join(os.homedir(), '.marketing-brain'));
49
- });
50
- });
51
-
52
- describe('getPipeName', () => {
53
- it('uses default name when no argument is given', () => {
54
- const result = getPipeName();
55
- if (process.platform === 'win32') {
56
- expect(result).toBe('\\\\.\\pipe\\marketing-brain');
57
- } else {
58
- expect(result).toBe(path.join(os.tmpdir(), 'marketing-brain.sock'));
59
- }
60
- });
61
-
62
- it('uses custom name when provided', () => {
63
- const result = getPipeName('my-app');
64
- if (process.platform === 'win32') {
65
- expect(result).toBe('\\\\.\\pipe\\my-app');
66
- } else {
67
- expect(result).toBe(path.join(os.tmpdir(), 'my-app.sock'));
68
- }
69
- });
70
- });
@@ -1,44 +0,0 @@
1
- import { EventEmitter } from 'node:events';
2
-
3
- export interface MarketingEvents {
4
- 'post:created': { postId: number; campaignId: number | null; platform: string };
5
- 'post:published': { postId: number; platform: string };
6
- 'engagement:updated': { postId: number; engagementId: number };
7
- 'strategy:reported': { strategyId: number; postId: number };
8
- 'rule:learned': { ruleId: number; pattern: string };
9
- 'rule:triggered': { ruleId: number; postId: number };
10
- 'template:created': { templateId: number; platform: string };
11
- 'campaign:created': { campaignId: number; name: string };
12
- 'insight:created': { insightId: number; type: string };
13
- 'synapse:created': { synapseId: number; sourceType: string; targetType: string };
14
- 'synapse:strengthened': { synapseId: number; newWeight: number };
15
- }
16
-
17
- export type MarketingEventName = keyof MarketingEvents;
18
-
19
- export class TypedEventBus extends EventEmitter {
20
- emit<K extends MarketingEventName>(event: K, data: MarketingEvents[K]): boolean {
21
- return super.emit(event, data);
22
- }
23
-
24
- on<K extends MarketingEventName>(event: K, listener: (data: MarketingEvents[K]) => void): this {
25
- return super.on(event, listener);
26
- }
27
-
28
- once<K extends MarketingEventName>(event: K, listener: (data: MarketingEvents[K]) => void): this {
29
- return super.once(event, listener);
30
- }
31
-
32
- off<K extends MarketingEventName>(event: K, listener: (data: MarketingEvents[K]) => void): this {
33
- return super.off(event, listener);
34
- }
35
- }
36
-
37
- let busInstance: TypedEventBus | null = null;
38
-
39
- export function getEventBus(): TypedEventBus {
40
- if (!busInstance) {
41
- busInstance = new TypedEventBus();
42
- }
43
- return busInstance;
44
- }
package/src/utils/hash.ts DELETED
@@ -1,5 +0,0 @@
1
- import { createHash } from 'node:crypto';
2
-
3
- export function sha256(input: string): string {
4
- return createHash('sha256').update(input).digest('hex');
5
- }
@@ -1,48 +0,0 @@
1
- import winston from 'winston';
2
- import path from 'node:path';
3
- import { getDataDir } from './paths.js';
4
-
5
- const { combine, timestamp, printf, colorize } = winston.format;
6
-
7
- const logFormat = printf(({ level, message, timestamp, ...meta }) => {
8
- const metaStr = Object.keys(meta).length ? ` ${JSON.stringify(meta)}` : '';
9
- return `${timestamp} [${level}]${metaStr} ${message}`;
10
- });
11
-
12
- let loggerInstance: winston.Logger | null = null;
13
-
14
- export function createLogger(opts?: { level?: string; file?: string; maxSize?: number; maxFiles?: number }): winston.Logger {
15
- if (loggerInstance) return loggerInstance;
16
-
17
- const level = opts?.level ?? process.env['MARKETING_BRAIN_LOG_LEVEL'] ?? 'info';
18
- const logFile = opts?.file ?? path.join(getDataDir(), 'marketing-brain.log');
19
- const maxSize = opts?.maxSize ?? 10 * 1024 * 1024;
20
- const maxFiles = opts?.maxFiles ?? 3;
21
-
22
- const transports: winston.transport[] = [
23
- new winston.transports.File({
24
- filename: logFile,
25
- maxsize: maxSize,
26
- maxFiles,
27
- format: combine(timestamp(), logFormat),
28
- }),
29
- ];
30
-
31
- if (process.env['NODE_ENV'] !== 'production') {
32
- transports.push(
33
- new winston.transports.Console({
34
- format: combine(colorize(), timestamp(), logFormat),
35
- })
36
- );
37
- }
38
-
39
- loggerInstance = winston.createLogger({ level, transports });
40
- return loggerInstance;
41
- }
42
-
43
- export function getLogger(): winston.Logger {
44
- if (!loggerInstance) {
45
- return createLogger();
46
- }
47
- return loggerInstance;
48
- }
@@ -1,19 +0,0 @@
1
- import path from 'node:path';
2
- import os from 'node:os';
3
-
4
- export function normalizePath(filePath: string): string {
5
- return filePath.replace(/\\/g, '/');
6
- }
7
-
8
- export function getDataDir(): string {
9
- const envDir = process.env['MARKETING_BRAIN_DATA_DIR'];
10
- if (envDir) return path.resolve(envDir);
11
- return path.join(os.homedir(), '.marketing-brain');
12
- }
13
-
14
- export function getPipeName(name: string = 'marketing-brain'): string {
15
- if (process.platform === 'win32') {
16
- return `\\\\.\\pipe\\${name}`;
17
- }
18
- return path.join(os.tmpdir(), `${name}.sock`);
19
- }
package/tsconfig.json DELETED
@@ -1,18 +0,0 @@
1
- {
2
- "compilerOptions": {
3
- "target": "ES2022",
4
- "module": "NodeNext",
5
- "moduleResolution": "NodeNext",
6
- "strict": true,
7
- "esModuleInterop": true,
8
- "skipLibCheck": true,
9
- "forceConsistentCasingInFileNames": true,
10
- "outDir": "dist",
11
- "rootDir": "src",
12
- "declaration": true,
13
- "sourceMap": true,
14
- "resolveJsonModule": true
15
- },
16
- "include": ["src/**/*"],
17
- "exclude": ["node_modules", "dist"]
18
- }