@timmeck/marketing-brain 0.2.1 → 0.3.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 (123) hide show
  1. package/dist/cli/colors.d.ts +11 -24
  2. package/dist/cli/colors.js +3 -46
  3. package/dist/cli/colors.js.map +1 -1
  4. package/dist/cli/commands/peers.d.ts +2 -0
  5. package/dist/cli/commands/peers.js +38 -0
  6. package/dist/cli/commands/peers.js.map +1 -0
  7. package/dist/db/connection.d.ts +1 -2
  8. package/dist/db/connection.js +1 -18
  9. package/dist/db/connection.js.map +1 -1
  10. package/dist/hooks/post-tool-use.d.ts +2 -0
  11. package/dist/hooks/post-tool-use.js +182 -0
  12. package/dist/hooks/post-tool-use.js.map +1 -0
  13. package/dist/index.js +2 -0
  14. package/dist/index.js.map +1 -1
  15. package/dist/ipc/client.d.ts +1 -13
  16. package/dist/ipc/client.js +1 -92
  17. package/dist/ipc/client.js.map +1 -1
  18. package/dist/ipc/protocol.d.ts +1 -8
  19. package/dist/ipc/protocol.js +1 -28
  20. package/dist/ipc/protocol.js.map +1 -1
  21. package/dist/ipc/router.js +8 -0
  22. package/dist/ipc/router.js.map +1 -1
  23. package/dist/ipc/server.d.ts +1 -14
  24. package/dist/ipc/server.js +1 -129
  25. package/dist/ipc/server.js.map +1 -1
  26. package/dist/marketing-core.d.ts +1 -0
  27. package/dist/marketing-core.js +6 -1
  28. package/dist/marketing-core.js.map +1 -1
  29. package/dist/mcp/server.js +5 -60
  30. package/dist/mcp/server.js.map +1 -1
  31. package/dist/types/ipc.types.d.ts +1 -11
  32. package/dist/utils/events.d.ts +4 -8
  33. package/dist/utils/events.js +2 -14
  34. package/dist/utils/events.js.map +1 -1
  35. package/dist/utils/hash.d.ts +1 -1
  36. package/dist/utils/hash.js +1 -4
  37. package/dist/utils/hash.js.map +1 -1
  38. package/dist/utils/logger.d.ts +3 -2
  39. package/dist/utils/logger.js +8 -35
  40. package/dist/utils/logger.js.map +1 -1
  41. package/dist/utils/paths.d.ts +2 -1
  42. package/dist/utils/paths.js +4 -13
  43. package/dist/utils/paths.js.map +1 -1
  44. package/package.json +2 -1
  45. package/.github/FUNDING.yml +0 -1
  46. package/.github/workflows/ci.yml +0 -27
  47. package/.mcp.json +0 -9
  48. package/src/api/server.ts +0 -86
  49. package/src/cli/colors.ts +0 -59
  50. package/src/cli/commands/campaign.ts +0 -66
  51. package/src/cli/commands/config.ts +0 -168
  52. package/src/cli/commands/dashboard.ts +0 -165
  53. package/src/cli/commands/doctor.ts +0 -110
  54. package/src/cli/commands/export.ts +0 -40
  55. package/src/cli/commands/import.ts +0 -84
  56. package/src/cli/commands/insights.ts +0 -44
  57. package/src/cli/commands/learn.ts +0 -24
  58. package/src/cli/commands/network.ts +0 -71
  59. package/src/cli/commands/post.ts +0 -47
  60. package/src/cli/commands/query.ts +0 -108
  61. package/src/cli/commands/rules.ts +0 -27
  62. package/src/cli/commands/start.ts +0 -100
  63. package/src/cli/commands/status.ts +0 -73
  64. package/src/cli/commands/stop.ts +0 -33
  65. package/src/cli/commands/suggest.ts +0 -64
  66. package/src/cli/ipc-helper.ts +0 -22
  67. package/src/cli/update-check.ts +0 -63
  68. package/src/config.ts +0 -110
  69. package/src/dashboard/renderer.ts +0 -136
  70. package/src/dashboard/server.ts +0 -140
  71. package/src/db/connection.ts +0 -22
  72. package/src/db/migrations/001_core_schema.ts +0 -63
  73. package/src/db/migrations/002_learning_schema.ts +0 -46
  74. package/src/db/migrations/003_synapse_schema.ts +0 -27
  75. package/src/db/migrations/004_insights_schema.ts +0 -38
  76. package/src/db/migrations/005_fts_indexes.ts +0 -77
  77. package/src/db/migrations/index.ts +0 -62
  78. package/src/db/repositories/audience.repository.ts +0 -53
  79. package/src/db/repositories/campaign.repository.ts +0 -72
  80. package/src/db/repositories/engagement.repository.ts +0 -108
  81. package/src/db/repositories/insight.repository.ts +0 -100
  82. package/src/db/repositories/post.repository.ts +0 -123
  83. package/src/db/repositories/rule.repository.ts +0 -87
  84. package/src/db/repositories/strategy.repository.ts +0 -82
  85. package/src/db/repositories/synapse.repository.ts +0 -148
  86. package/src/db/repositories/template.repository.ts +0 -76
  87. package/src/index.ts +0 -69
  88. package/src/ipc/__tests__/protocol.test.ts +0 -153
  89. package/src/ipc/client.ts +0 -110
  90. package/src/ipc/protocol.ts +0 -35
  91. package/src/ipc/router.ts +0 -126
  92. package/src/ipc/server.ts +0 -140
  93. package/src/learning/confidence-scorer.ts +0 -36
  94. package/src/learning/learning-engine.ts +0 -254
  95. package/src/marketing-core.ts +0 -285
  96. package/src/mcp/server.ts +0 -72
  97. package/src/mcp/tools.ts +0 -216
  98. package/src/research/research-engine.ts +0 -226
  99. package/src/services/analytics.service.ts +0 -73
  100. package/src/services/audience.service.ts +0 -40
  101. package/src/services/campaign.service.ts +0 -80
  102. package/src/services/insight.service.ts +0 -54
  103. package/src/services/post.service.ts +0 -116
  104. package/src/services/rule.service.ts +0 -90
  105. package/src/services/strategy.service.ts +0 -53
  106. package/src/services/synapse.service.ts +0 -32
  107. package/src/services/template.service.ts +0 -50
  108. package/src/synapses/activation.ts +0 -80
  109. package/src/synapses/decay.ts +0 -38
  110. package/src/synapses/hebbian.ts +0 -68
  111. package/src/synapses/pathfinder.ts +0 -81
  112. package/src/synapses/synapse-manager.ts +0 -115
  113. package/src/types/config.types.ts +0 -79
  114. package/src/types/ipc.types.ts +0 -8
  115. package/src/types/post.types.ts +0 -156
  116. package/src/types/synapse.types.ts +0 -43
  117. package/src/utils/__tests__/hash.test.ts +0 -39
  118. package/src/utils/__tests__/paths.test.ts +0 -70
  119. package/src/utils/events.ts +0 -44
  120. package/src/utils/hash.ts +0 -5
  121. package/src/utils/logger.ts +0 -48
  122. package/src/utils/paths.ts +0 -19
  123. 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
- }