@posthog/wizard 1.2.1 → 1.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 (28) hide show
  1. package/README.md +28 -0
  2. package/dist/src/nextjs/docs.d.ts +4 -0
  3. package/dist/src/nextjs/docs.js +67 -38
  4. package/dist/src/nextjs/docs.js.map +1 -1
  5. package/dist/src/nextjs/nextjs-wizard.js +62 -7
  6. package/dist/src/nextjs/nextjs-wizard.js.map +1 -1
  7. package/dist/src/run.js +13 -3
  8. package/dist/src/run.js.map +1 -1
  9. package/dist/src/steps/add-mcp-server-to-clients/MCPClient.d.ts +19 -3
  10. package/dist/src/steps/add-mcp-server-to-clients/MCPClient.js +102 -1
  11. package/dist/src/steps/add-mcp-server-to-clients/MCPClient.js.map +1 -1
  12. package/dist/src/steps/add-mcp-server-to-clients/clients/__tests__/claude.test.js +34 -16
  13. package/dist/src/steps/add-mcp-server-to-clients/clients/__tests__/claude.test.js.map +1 -1
  14. package/dist/src/steps/add-mcp-server-to-clients/clients/claude.d.ts +42 -29
  15. package/dist/src/steps/add-mcp-server-to-clients/clients/claude.js +6 -61
  16. package/dist/src/steps/add-mcp-server-to-clients/clients/claude.js.map +1 -1
  17. package/dist/src/steps/add-mcp-server-to-clients/clients/cursor.d.ts +42 -29
  18. package/dist/src/steps/add-mcp-server-to-clients/clients/cursor.js +5 -62
  19. package/dist/src/steps/add-mcp-server-to-clients/clients/cursor.js.map +1 -1
  20. package/dist/src/steps/add-mcp-server-to-clients/defaults.d.ts +38 -22
  21. package/dist/src/steps/add-mcp-server-to-clients/defaults.js +7 -5
  22. package/dist/src/steps/add-mcp-server-to-clients/defaults.js.map +1 -1
  23. package/dist/src/steps/add-mcp-server-to-clients/index.js +2 -2
  24. package/dist/src/steps/add-mcp-server-to-clients/index.js.map +1 -1
  25. package/package.json +1 -1
  26. package/dist/src/steps/add-mcp-server-to-clients/clients/__tests__/cursor.test.d.ts +0 -1
  27. package/dist/src/steps/add-mcp-server-to-clients/clients/__tests__/cursor.test.js +0 -443
  28. package/dist/src/steps/add-mcp-server-to-clients/clients/__tests__/cursor.test.js.map +0 -1
@@ -33,6 +33,7 @@ var __importStar = (this && this.__importStar) || (function () {
33
33
  };
34
34
  })();
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
+ // We use the ClaudeMCPClient as a reference to test the DefaultMCPClient
36
37
  const fs = __importStar(require("fs"));
37
38
  const path = __importStar(require("path"));
38
39
  const os = __importStar(require("os"));
@@ -92,60 +93,60 @@ describe('ClaudeMCPClient', () => {
92
93
  });
93
94
  });
94
95
  describe('isClientSupported', () => {
95
- it('should return true for macOS', () => {
96
+ it('should return true for macOS', async () => {
96
97
  Object.defineProperty(process, 'platform', {
97
98
  value: 'darwin',
98
99
  writable: true,
99
100
  });
100
- expect(client.isClientSupported()).toBe(true);
101
+ await expect(client.isClientSupported()).resolves.toBe(true);
101
102
  });
102
- it('should return true for Windows', () => {
103
+ it('should return true for Windows', async () => {
103
104
  Object.defineProperty(process, 'platform', {
104
105
  value: 'win32',
105
106
  writable: true,
106
107
  });
107
- expect(client.isClientSupported()).toBe(true);
108
+ await expect(client.isClientSupported()).resolves.toBe(true);
108
109
  });
109
- it('should return false for Linux', () => {
110
+ it('should return false for Linux', async () => {
110
111
  Object.defineProperty(process, 'platform', {
111
112
  value: 'linux',
112
113
  writable: true,
113
114
  });
114
- expect(client.isClientSupported()).toBe(false);
115
+ await expect(client.isClientSupported()).resolves.toBe(false);
115
116
  });
116
- it('should return false for other platforms', () => {
117
+ it('should return false for other platforms', async () => {
117
118
  Object.defineProperty(process, 'platform', {
118
119
  value: 'freebsd',
119
120
  writable: true,
120
121
  });
121
- expect(client.isClientSupported()).toBe(false);
122
+ await expect(client.isClientSupported()).resolves.toBe(false);
122
123
  });
123
124
  });
124
125
  describe('getConfigPath', () => {
125
- it('should return correct path for macOS', () => {
126
+ it('should return correct path for macOS', async () => {
126
127
  Object.defineProperty(process, 'platform', {
127
128
  value: 'darwin',
128
129
  writable: true,
129
130
  });
130
- const configPath = client.getConfigPath();
131
+ const configPath = await client.getConfigPath();
131
132
  expect(configPath).toBe(path.join(mockHomeDir, 'Library', 'Application Support', 'Claude', 'claude_desktop_config.json'));
132
133
  });
133
- it('should return correct path for Windows', () => {
134
+ it('should return correct path for Windows', async () => {
134
135
  Object.defineProperty(process, 'platform', {
135
136
  value: 'win32',
136
137
  writable: true,
137
138
  });
138
139
  const mockAppData = 'C:\\Users\\Test\\AppData\\Roaming';
139
140
  process.env.APPDATA = mockAppData;
140
- const configPath = client.getConfigPath();
141
+ const configPath = await client.getConfigPath();
141
142
  expect(configPath).toBe(path.join(mockAppData, 'Claude', 'claude_desktop_config.json'));
142
143
  });
143
- it('should throw error for unsupported platform', () => {
144
+ it('should throw error for unsupported platform', async () => {
144
145
  Object.defineProperty(process, 'platform', {
145
146
  value: 'linux',
146
147
  writable: true,
147
148
  });
148
- expect(() => client.getConfigPath()).toThrow('Unsupported platform: linux');
149
+ await expect(client.getConfigPath()).rejects.toThrow('Unsupported platform: linux');
149
150
  });
150
151
  });
151
152
  describe('isServerInstalled', () => {
@@ -237,11 +238,28 @@ describe('ClaudeMCPClient', () => {
237
238
  },
238
239
  }, null, 2), 'utf8');
239
240
  });
240
- it('should create new config when existing config is invalid', async () => {
241
+ it('should not overwrite existing config when it is invalid', async () => {
241
242
  existsSyncMock.mockReturnValue(true);
242
- readFileMock.mockResolvedValue('invalid json');
243
+ readFileMock.mockResolvedValue(JSON.stringify({
244
+ invalidKey: {
245
+ existingServer: {
246
+ command: 'existing',
247
+ args: [],
248
+ env: {},
249
+ },
250
+ },
251
+ x: 'y',
252
+ }));
243
253
  await client.addServer(mockApiKey);
244
254
  expect(writeFileMock).toHaveBeenCalledWith(expect.any(String), JSON.stringify({
255
+ invalidKey: {
256
+ existingServer: {
257
+ command: 'existing',
258
+ args: [],
259
+ env: {},
260
+ },
261
+ },
262
+ x: 'y',
245
263
  mcpServers: {
246
264
  posthog: mockServerConfig,
247
265
  },
@@ -1 +1 @@
1
- {"version":3,"file":"claude.test.js","sourceRoot":"","sources":["../../../../../../src/steps/add-mcp-server-to-clients/clients/__tests__/claude.test.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport * as os from 'os';\nimport { ClaudeMCPClient } from '../claude';\nimport { getDefaultServerConfig } from '../../defaults';\n\njest.mock('fs', () => ({\n promises: {\n mkdir: jest.fn(),\n readFile: jest.fn(),\n writeFile: jest.fn(),\n },\n existsSync: jest.fn(),\n}));\n\njest.mock('os', () => ({\n homedir: jest.fn(),\n}));\n\njest.mock('../../defaults', () => ({\n DefaultMCPClientConfig: {\n parse: jest.fn(),\n },\n getDefaultServerConfig: jest.fn(),\n}));\n\ndescribe('ClaudeMCPClient', () => {\n let client: ClaudeMCPClient;\n const mockHomeDir = '/mock/home';\n const mockApiKey = 'test-api-key';\n const mockServerConfig = {\n command: 'npx',\n args: ['-y', 'mcp-remote@latest'],\n env: { POSTHOG_AUTH_HEADER: `Bearer ${mockApiKey}` },\n };\n\n const mkdirMock = fs.promises.mkdir as jest.Mock;\n const readFileMock = fs.promises.readFile as jest.Mock;\n const writeFileMock = fs.promises.writeFile as jest.Mock;\n const existsSyncMock = fs.existsSync as jest.Mock;\n const homedirMock = os.homedir as jest.Mock;\n const getDefaultServerConfigMock = getDefaultServerConfig as jest.Mock;\n\n const originalPlatform = process.platform;\n\n beforeEach(() => {\n client = new ClaudeMCPClient();\n jest.clearAllMocks();\n homedirMock.mockReturnValue(mockHomeDir);\n getDefaultServerConfigMock.mockReturnValue(mockServerConfig);\n\n // Mock the Zod schema parse method\n const { DefaultMCPClientConfig } = require('../../defaults');\n DefaultMCPClientConfig.parse.mockImplementation((data: any) => data);\n });\n\n afterEach(() => {\n Object.defineProperty(process, 'platform', {\n value: originalPlatform,\n writable: true,\n });\n });\n\n describe('constructor', () => {\n it('should set the correct name', () => {\n expect(client.name).toBe('Claude Desktop');\n });\n });\n\n describe('isClientSupported', () => {\n it('should return true for macOS', () => {\n Object.defineProperty(process, 'platform', {\n value: 'darwin',\n writable: true,\n });\n expect(client.isClientSupported()).toBe(true);\n });\n\n it('should return true for Windows', () => {\n Object.defineProperty(process, 'platform', {\n value: 'win32',\n writable: true,\n });\n expect(client.isClientSupported()).toBe(true);\n });\n\n it('should return false for Linux', () => {\n Object.defineProperty(process, 'platform', {\n value: 'linux',\n writable: true,\n });\n expect(client.isClientSupported()).toBe(false);\n });\n\n it('should return false for other platforms', () => {\n Object.defineProperty(process, 'platform', {\n value: 'freebsd',\n writable: true,\n });\n expect(client.isClientSupported()).toBe(false);\n });\n });\n\n describe('getConfigPath', () => {\n it('should return correct path for macOS', () => {\n Object.defineProperty(process, 'platform', {\n value: 'darwin',\n writable: true,\n });\n\n const configPath = (client as any).getConfigPath();\n expect(configPath).toBe(\n path.join(\n mockHomeDir,\n 'Library',\n 'Application Support',\n 'Claude',\n 'claude_desktop_config.json',\n ),\n );\n });\n\n it('should return correct path for Windows', () => {\n Object.defineProperty(process, 'platform', {\n value: 'win32',\n writable: true,\n });\n\n const mockAppData = 'C:\\\\Users\\\\Test\\\\AppData\\\\Roaming';\n process.env.APPDATA = mockAppData;\n\n const configPath = (client as any).getConfigPath();\n expect(configPath).toBe(\n path.join(mockAppData, 'Claude', 'claude_desktop_config.json'),\n );\n });\n\n it('should throw error for unsupported platform', () => {\n Object.defineProperty(process, 'platform', {\n value: 'linux',\n writable: true,\n });\n\n expect(() => (client as any).getConfigPath()).toThrow(\n 'Unsupported platform: linux',\n );\n });\n });\n\n describe('isServerInstalled', () => {\n beforeEach(() => {\n Object.defineProperty(process, 'platform', {\n value: 'darwin',\n writable: true,\n });\n });\n\n it('should return false when config file does not exist', async () => {\n existsSyncMock.mockReturnValue(false);\n\n const result = await client.isServerInstalled();\n expect(result).toBe(false);\n });\n\n it('should return false when config file exists but posthog server is not configured', async () => {\n existsSyncMock.mockReturnValue(true);\n const configData = {\n mcpServers: {\n otherServer: mockServerConfig,\n },\n };\n readFileMock.mockResolvedValue(JSON.stringify(configData));\n\n const result = await client.isServerInstalled();\n expect(result).toBe(false);\n });\n\n it('should return true when posthog server is configured', async () => {\n existsSyncMock.mockReturnValue(true);\n const configData = {\n mcpServers: {\n posthog: mockServerConfig,\n otherServer: mockServerConfig,\n },\n };\n readFileMock.mockResolvedValue(JSON.stringify(configData));\n\n const result = await client.isServerInstalled();\n expect(result).toBe(true);\n });\n\n it('should return false when config file is invalid JSON', async () => {\n existsSyncMock.mockReturnValue(true);\n readFileMock.mockResolvedValue('invalid json');\n\n const result = await client.isServerInstalled();\n expect(result).toBe(false);\n });\n\n it('should return false when readFile throws an error', async () => {\n existsSyncMock.mockReturnValue(true);\n readFileMock.mockRejectedValue(new Error('File read error'));\n\n const result = await client.isServerInstalled();\n expect(result).toBe(false);\n });\n });\n\n describe('addServer', () => {\n beforeEach(() => {\n Object.defineProperty(process, 'platform', {\n value: 'darwin',\n writable: true,\n });\n });\n\n it('should create config directory and add server when config file does not exist', async () => {\n existsSyncMock.mockReturnValue(false);\n\n await client.addServer(mockApiKey);\n\n const expectedConfigPath = path.join(\n mockHomeDir,\n 'Library',\n 'Application Support',\n 'Claude',\n 'claude_desktop_config.json',\n );\n const expectedConfigDir = path.dirname(expectedConfigPath);\n\n expect(mkdirMock).toHaveBeenCalledWith(expectedConfigDir, {\n recursive: true,\n });\n expect(writeFileMock).toHaveBeenCalledWith(\n expectedConfigPath,\n JSON.stringify(\n {\n mcpServers: {\n posthog: mockServerConfig,\n },\n },\n null,\n 2,\n ),\n 'utf8',\n );\n });\n\n it('should merge with existing config when config file exists', async () => {\n existsSyncMock.mockReturnValue(true);\n const existingConfig = {\n mcpServers: {\n existingServer: {\n command: 'existing',\n args: [],\n env: {},\n },\n },\n };\n readFileMock.mockResolvedValue(JSON.stringify(existingConfig));\n\n await client.addServer(mockApiKey);\n\n expect(writeFileMock).toHaveBeenCalledWith(\n expect.any(String),\n JSON.stringify(\n {\n mcpServers: {\n existingServer: existingConfig.mcpServers.existingServer,\n posthog: mockServerConfig,\n },\n },\n null,\n 2,\n ),\n 'utf8',\n );\n });\n\n it('should create new config when existing config is invalid', async () => {\n existsSyncMock.mockReturnValue(true);\n readFileMock.mockResolvedValue('invalid json');\n\n await client.addServer(mockApiKey);\n\n expect(writeFileMock).toHaveBeenCalledWith(\n expect.any(String),\n JSON.stringify(\n {\n mcpServers: {\n posthog: mockServerConfig,\n },\n },\n null,\n 2,\n ),\n 'utf8',\n );\n });\n\n it('should call getDefaultServerConfig with the provided API key', async () => {\n existsSyncMock.mockReturnValue(false);\n\n await client.addServer(mockApiKey);\n\n expect(getDefaultServerConfigMock).toHaveBeenCalledWith(mockApiKey);\n });\n });\n\n describe('removeServer', () => {\n beforeEach(() => {\n Object.defineProperty(process, 'platform', {\n value: 'darwin',\n writable: true,\n });\n });\n\n it('should do nothing when config file does not exist', async () => {\n existsSyncMock.mockReturnValue(false);\n\n await client.removeServer();\n\n expect(readFileMock).not.toHaveBeenCalled();\n expect(writeFileMock).not.toHaveBeenCalled();\n });\n\n it('should remove posthog server from config', async () => {\n existsSyncMock.mockReturnValue(true);\n const configWithPosthog = {\n mcpServers: {\n posthog: mockServerConfig,\n otherServer: {\n command: 'other',\n args: [],\n env: {},\n },\n },\n };\n readFileMock.mockResolvedValue(JSON.stringify(configWithPosthog));\n\n await client.removeServer();\n\n expect(writeFileMock).toHaveBeenCalledWith(\n expect.any(String),\n JSON.stringify(\n {\n mcpServers: {\n otherServer: configWithPosthog.mcpServers.otherServer,\n },\n },\n null,\n 2,\n ),\n 'utf8',\n );\n });\n\n it('should do nothing when posthog server is not in config', async () => {\n existsSyncMock.mockReturnValue(true);\n const configWithoutPosthog = {\n mcpServers: {\n otherServer: {\n command: 'other',\n args: [],\n env: {},\n },\n },\n };\n readFileMock.mockResolvedValue(JSON.stringify(configWithoutPosthog));\n\n await client.removeServer();\n\n expect(writeFileMock).not.toHaveBeenCalled();\n });\n\n it('should handle invalid JSON gracefully', async () => {\n existsSyncMock.mockReturnValue(true);\n readFileMock.mockResolvedValue('invalid json');\n\n await client.removeServer();\n\n expect(writeFileMock).not.toHaveBeenCalled();\n });\n\n it('should handle file read errors gracefully', async () => {\n existsSyncMock.mockReturnValue(true);\n readFileMock.mockRejectedValue(new Error('File read error'));\n\n await client.removeServer();\n\n expect(writeFileMock).not.toHaveBeenCalled();\n });\n });\n});\n"]}
1
+ {"version":3,"file":"claude.test.js","sourceRoot":"","sources":["../../../../../../src/steps/add-mcp-server-to-clients/clients/__tests__/claude.test.ts"],"names":[],"mappings":"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","sourcesContent":["// We use the ClaudeMCPClient as a reference to test the DefaultMCPClient\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as os from 'os';\nimport { ClaudeMCPClient } from '../claude';\nimport { getDefaultServerConfig } from '../../defaults';\n\njest.mock('fs', () => ({\n promises: {\n mkdir: jest.fn(),\n readFile: jest.fn(),\n writeFile: jest.fn(),\n },\n existsSync: jest.fn(),\n}));\n\njest.mock('os', () => ({\n homedir: jest.fn(),\n}));\n\njest.mock('../../defaults', () => ({\n DefaultMCPClientConfig: {\n parse: jest.fn(),\n },\n getDefaultServerConfig: jest.fn(),\n}));\n\ndescribe('ClaudeMCPClient', () => {\n let client: ClaudeMCPClient;\n const mockHomeDir = '/mock/home';\n const mockApiKey = 'test-api-key';\n const mockServerConfig = {\n command: 'npx',\n args: ['-y', 'mcp-remote@latest'],\n env: { POSTHOG_AUTH_HEADER: `Bearer ${mockApiKey}` },\n };\n\n const mkdirMock = fs.promises.mkdir as jest.Mock;\n const readFileMock = fs.promises.readFile as jest.Mock;\n const writeFileMock = fs.promises.writeFile as jest.Mock;\n const existsSyncMock = fs.existsSync as jest.Mock;\n const homedirMock = os.homedir as jest.Mock;\n const getDefaultServerConfigMock = getDefaultServerConfig as jest.Mock;\n\n const originalPlatform = process.platform;\n\n beforeEach(() => {\n client = new ClaudeMCPClient();\n jest.clearAllMocks();\n homedirMock.mockReturnValue(mockHomeDir);\n getDefaultServerConfigMock.mockReturnValue(mockServerConfig);\n\n // Mock the Zod schema parse method\n const { DefaultMCPClientConfig } = require('../../defaults');\n DefaultMCPClientConfig.parse.mockImplementation((data: any) => data);\n });\n\n afterEach(() => {\n Object.defineProperty(process, 'platform', {\n value: originalPlatform,\n writable: true,\n });\n });\n\n describe('constructor', () => {\n it('should set the correct name', () => {\n expect(client.name).toBe('Claude Desktop');\n });\n });\n\n describe('isClientSupported', () => {\n it('should return true for macOS', async () => {\n Object.defineProperty(process, 'platform', {\n value: 'darwin',\n writable: true,\n });\n await expect(client.isClientSupported()).resolves.toBe(true);\n });\n\n it('should return true for Windows', async () => {\n Object.defineProperty(process, 'platform', {\n value: 'win32',\n writable: true,\n });\n await expect(client.isClientSupported()).resolves.toBe(true);\n });\n\n it('should return false for Linux', async () => {\n Object.defineProperty(process, 'platform', {\n value: 'linux',\n writable: true,\n });\n await expect(client.isClientSupported()).resolves.toBe(false);\n });\n\n it('should return false for other platforms', async () => {\n Object.defineProperty(process, 'platform', {\n value: 'freebsd',\n writable: true,\n });\n await expect(client.isClientSupported()).resolves.toBe(false);\n });\n });\n\n describe('getConfigPath', () => {\n it('should return correct path for macOS', async () => {\n Object.defineProperty(process, 'platform', {\n value: 'darwin',\n writable: true,\n });\n\n const configPath = await client.getConfigPath();\n expect(configPath).toBe(\n path.join(\n mockHomeDir,\n 'Library',\n 'Application Support',\n 'Claude',\n 'claude_desktop_config.json',\n ),\n );\n });\n\n it('should return correct path for Windows', async () => {\n Object.defineProperty(process, 'platform', {\n value: 'win32',\n writable: true,\n });\n\n const mockAppData = 'C:\\\\Users\\\\Test\\\\AppData\\\\Roaming';\n process.env.APPDATA = mockAppData;\n\n const configPath = await client.getConfigPath();\n expect(configPath).toBe(\n path.join(mockAppData, 'Claude', 'claude_desktop_config.json'),\n );\n });\n\n it('should throw error for unsupported platform', async () => {\n Object.defineProperty(process, 'platform', {\n value: 'linux',\n writable: true,\n });\n\n await expect(client.getConfigPath()).rejects.toThrow(\n 'Unsupported platform: linux',\n );\n });\n });\n\n describe('isServerInstalled', () => {\n beforeEach(() => {\n Object.defineProperty(process, 'platform', {\n value: 'darwin',\n writable: true,\n });\n });\n\n it('should return false when config file does not exist', async () => {\n existsSyncMock.mockReturnValue(false);\n\n const result = await client.isServerInstalled();\n expect(result).toBe(false);\n });\n\n it('should return false when config file exists but posthog server is not configured', async () => {\n existsSyncMock.mockReturnValue(true);\n const configData = {\n mcpServers: {\n otherServer: mockServerConfig,\n },\n };\n readFileMock.mockResolvedValue(JSON.stringify(configData));\n\n const result = await client.isServerInstalled();\n expect(result).toBe(false);\n });\n\n it('should return true when posthog server is configured', async () => {\n existsSyncMock.mockReturnValue(true);\n const configData = {\n mcpServers: {\n posthog: mockServerConfig,\n otherServer: mockServerConfig,\n },\n };\n readFileMock.mockResolvedValue(JSON.stringify(configData));\n\n const result = await client.isServerInstalled();\n expect(result).toBe(true);\n });\n\n it('should return false when config file is invalid JSON', async () => {\n existsSyncMock.mockReturnValue(true);\n readFileMock.mockResolvedValue('invalid json');\n\n const result = await client.isServerInstalled();\n expect(result).toBe(false);\n });\n\n it('should return false when readFile throws an error', async () => {\n existsSyncMock.mockReturnValue(true);\n readFileMock.mockRejectedValue(new Error('File read error'));\n\n const result = await client.isServerInstalled();\n expect(result).toBe(false);\n });\n });\n\n describe('addServer', () => {\n beforeEach(() => {\n Object.defineProperty(process, 'platform', {\n value: 'darwin',\n writable: true,\n });\n });\n\n it('should create config directory and add server when config file does not exist', async () => {\n existsSyncMock.mockReturnValue(false);\n\n await client.addServer(mockApiKey);\n\n const expectedConfigPath = path.join(\n mockHomeDir,\n 'Library',\n 'Application Support',\n 'Claude',\n 'claude_desktop_config.json',\n );\n const expectedConfigDir = path.dirname(expectedConfigPath);\n\n expect(mkdirMock).toHaveBeenCalledWith(expectedConfigDir, {\n recursive: true,\n });\n\n expect(writeFileMock).toHaveBeenCalledWith(\n expectedConfigPath,\n JSON.stringify(\n {\n mcpServers: {\n posthog: mockServerConfig,\n },\n },\n null,\n 2,\n ),\n 'utf8',\n );\n });\n\n it('should merge with existing config when config file exists', async () => {\n existsSyncMock.mockReturnValue(true);\n const existingConfig = {\n mcpServers: {\n existingServer: {\n command: 'existing',\n args: [],\n env: {},\n },\n },\n };\n readFileMock.mockResolvedValue(JSON.stringify(existingConfig));\n\n await client.addServer(mockApiKey);\n\n expect(writeFileMock).toHaveBeenCalledWith(\n expect.any(String),\n JSON.stringify(\n {\n mcpServers: {\n existingServer: existingConfig.mcpServers.existingServer,\n posthog: mockServerConfig,\n },\n },\n null,\n 2,\n ),\n 'utf8',\n );\n });\n\n it('should not overwrite existing config when it is invalid', async () => {\n existsSyncMock.mockReturnValue(true);\n readFileMock.mockResolvedValue(\n JSON.stringify({\n invalidKey: {\n existingServer: {\n command: 'existing',\n args: [],\n env: {},\n },\n },\n x: 'y',\n }),\n );\n\n await client.addServer(mockApiKey);\n\n expect(writeFileMock).toHaveBeenCalledWith(\n expect.any(String),\n JSON.stringify(\n {\n invalidKey: {\n existingServer: {\n command: 'existing',\n args: [],\n env: {},\n },\n },\n x: 'y',\n mcpServers: {\n posthog: mockServerConfig,\n },\n },\n null,\n 2,\n ),\n 'utf8',\n );\n });\n\n it('should call getDefaultServerConfig with the provided API key', async () => {\n existsSyncMock.mockReturnValue(false);\n\n await client.addServer(mockApiKey);\n\n expect(getDefaultServerConfigMock).toHaveBeenCalledWith(mockApiKey);\n });\n });\n\n describe('removeServer', () => {\n beforeEach(() => {\n Object.defineProperty(process, 'platform', {\n value: 'darwin',\n writable: true,\n });\n });\n\n it('should do nothing when config file does not exist', async () => {\n existsSyncMock.mockReturnValue(false);\n\n await client.removeServer();\n\n expect(readFileMock).not.toHaveBeenCalled();\n expect(writeFileMock).not.toHaveBeenCalled();\n });\n\n it('should remove posthog server from config', async () => {\n existsSyncMock.mockReturnValue(true);\n const configWithPosthog = {\n mcpServers: {\n posthog: mockServerConfig,\n otherServer: {\n command: 'other',\n args: [],\n env: {},\n },\n },\n };\n readFileMock.mockResolvedValue(JSON.stringify(configWithPosthog));\n\n await client.removeServer();\n\n expect(writeFileMock).toHaveBeenCalledWith(\n expect.any(String),\n JSON.stringify(\n {\n mcpServers: {\n otherServer: configWithPosthog.mcpServers.otherServer,\n },\n },\n null,\n 2,\n ),\n 'utf8',\n );\n });\n\n it('should do nothing when posthog server is not in config', async () => {\n existsSyncMock.mockReturnValue(true);\n const configWithoutPosthog = {\n mcpServers: {\n otherServer: {\n command: 'other',\n args: [],\n env: {},\n },\n },\n };\n readFileMock.mockResolvedValue(JSON.stringify(configWithoutPosthog));\n\n await client.removeServer();\n\n expect(writeFileMock).not.toHaveBeenCalled();\n });\n\n it('should handle invalid JSON gracefully', async () => {\n existsSyncMock.mockReturnValue(true);\n readFileMock.mockResolvedValue('invalid json');\n\n await client.removeServer();\n\n expect(writeFileMock).not.toHaveBeenCalled();\n });\n\n it('should handle file read errors gracefully', async () => {\n existsSyncMock.mockReturnValue(true);\n readFileMock.mockRejectedValue(new Error('File read error'));\n\n await client.removeServer();\n\n expect(writeFileMock).not.toHaveBeenCalled();\n });\n });\n});\n"]}
@@ -1,40 +1,53 @@
1
- import { MCPClient } from '../MCPClient';
1
+ import { DefaultMCPClient } from '../MCPClient';
2
2
  import { DefaultMCPClientConfig } from '../defaults';
3
3
  import { z } from 'zod';
4
4
  export declare const ClaudeMCPConfig: z.ZodObject<{
5
5
  mcpServers: z.ZodRecord<z.ZodString, z.ZodObject<{
6
- command: z.ZodString;
7
- args: z.ZodArray<z.ZodString, "many">;
8
- env: z.ZodRecord<z.ZodString, z.ZodString>;
6
+ command: z.ZodOptional<z.ZodString>;
7
+ args: z.ZodOptional<z.ZodArray<z.ZodString, "many">>;
8
+ env: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodString>>;
9
9
  }, "strip", z.ZodTypeAny, {
10
- command: string;
11
- args: string[];
12
- env: Record<string, string>;
10
+ command?: string | undefined;
11
+ args?: string[] | undefined;
12
+ env?: Record<string, string> | undefined;
13
13
  }, {
14
- command: string;
15
- args: string[];
16
- env: Record<string, string>;
14
+ command?: string | undefined;
15
+ args?: string[] | undefined;
16
+ env?: Record<string, string> | undefined;
17
17
  }>>;
18
- }, "strip", z.ZodTypeAny, {
19
- mcpServers: Record<string, {
20
- command: string;
21
- args: string[];
22
- env: Record<string, string>;
23
- }>;
24
- }, {
25
- mcpServers: Record<string, {
26
- command: string;
27
- args: string[];
28
- env: Record<string, string>;
29
- }>;
30
- }>;
18
+ }, "passthrough", z.ZodTypeAny, z.objectOutputType<{
19
+ mcpServers: z.ZodRecord<z.ZodString, z.ZodObject<{
20
+ command: z.ZodOptional<z.ZodString>;
21
+ args: z.ZodOptional<z.ZodArray<z.ZodString, "many">>;
22
+ env: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodString>>;
23
+ }, "strip", z.ZodTypeAny, {
24
+ command?: string | undefined;
25
+ args?: string[] | undefined;
26
+ env?: Record<string, string> | undefined;
27
+ }, {
28
+ command?: string | undefined;
29
+ args?: string[] | undefined;
30
+ env?: Record<string, string> | undefined;
31
+ }>>;
32
+ }, z.ZodTypeAny, "passthrough">, z.objectInputType<{
33
+ mcpServers: z.ZodRecord<z.ZodString, z.ZodObject<{
34
+ command: z.ZodOptional<z.ZodString>;
35
+ args: z.ZodOptional<z.ZodArray<z.ZodString, "many">>;
36
+ env: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodString>>;
37
+ }, "strip", z.ZodTypeAny, {
38
+ command?: string | undefined;
39
+ args?: string[] | undefined;
40
+ env?: Record<string, string> | undefined;
41
+ }, {
42
+ command?: string | undefined;
43
+ args?: string[] | undefined;
44
+ env?: Record<string, string> | undefined;
45
+ }>>;
46
+ }, z.ZodTypeAny, "passthrough">>;
31
47
  export type ClaudeMCPConfig = z.infer<typeof DefaultMCPClientConfig>;
32
- export declare class ClaudeMCPClient extends MCPClient {
48
+ export declare class ClaudeMCPClient extends DefaultMCPClient {
33
49
  name: string;
34
50
  constructor();
35
- isClientSupported(): boolean;
36
- private getConfigPath;
37
- isServerInstalled(): Promise<boolean>;
38
- addServer(apiKey: string): Promise<void>;
39
- removeServer(): Promise<void>;
51
+ isClientSupported(): Promise<boolean>;
52
+ getConfigPath(): Promise<string>;
40
53
  }
@@ -35,85 +35,30 @@ var __importStar = (this && this.__importStar) || (function () {
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
36
  exports.ClaudeMCPClient = exports.ClaudeMCPConfig = void 0;
37
37
  const MCPClient_1 = require("../MCPClient");
38
- const fs = __importStar(require("fs"));
39
38
  const path = __importStar(require("path"));
40
39
  const os = __importStar(require("os"));
41
40
  const defaults_1 = require("../defaults");
42
- const lodash_1 = require("lodash");
43
41
  exports.ClaudeMCPConfig = defaults_1.DefaultMCPClientConfig;
44
- class ClaudeMCPClient extends MCPClient_1.MCPClient {
42
+ class ClaudeMCPClient extends MCPClient_1.DefaultMCPClient {
45
43
  name = 'Claude Desktop';
46
44
  constructor() {
47
45
  super();
48
46
  }
49
- isClientSupported() {
50
- return process.platform === 'darwin' || process.platform === 'win32';
47
+ async isClientSupported() {
48
+ return Promise.resolve(process.platform === 'darwin' || process.platform === 'win32');
51
49
  }
52
- getConfigPath() {
50
+ async getConfigPath() {
53
51
  const homeDir = os.homedir();
54
52
  const isWindows = process.platform === 'win32';
55
53
  const isMac = process.platform === 'darwin';
56
54
  if (isMac) {
57
- return path.join(homeDir, 'Library', 'Application Support', 'Claude', 'claude_desktop_config.json');
55
+ return Promise.resolve(path.join(homeDir, 'Library', 'Application Support', 'Claude', 'claude_desktop_config.json'));
58
56
  }
59
57
  if (isWindows) {
60
- return path.join(process.env.APPDATA || '', 'Claude', 'claude_desktop_config.json');
58
+ return Promise.resolve(path.join(process.env.APPDATA || '', 'Claude', 'claude_desktop_config.json'));
61
59
  }
62
60
  throw new Error(`Unsupported platform: ${process.platform}`);
63
61
  }
64
- async isServerInstalled() {
65
- try {
66
- const configPath = this.getConfigPath();
67
- if (!fs.existsSync(configPath)) {
68
- return false;
69
- }
70
- const configContent = await fs.promises.readFile(configPath, 'utf8');
71
- const config = exports.ClaudeMCPConfig.parse(JSON.parse(configContent));
72
- return 'posthog' in config.mcpServers;
73
- }
74
- catch {
75
- return false;
76
- }
77
- }
78
- async addServer(apiKey) {
79
- const configPath = this.getConfigPath();
80
- const configDir = path.dirname(configPath);
81
- await fs.promises.mkdir(configDir, { recursive: true });
82
- let existingConfig = { mcpServers: {} };
83
- if (fs.existsSync(configPath)) {
84
- try {
85
- const existingContent = await fs.promises.readFile(configPath, 'utf8');
86
- existingConfig = exports.ClaudeMCPConfig.parse(JSON.parse(existingContent));
87
- }
88
- catch {
89
- existingConfig = { mcpServers: {} };
90
- }
91
- }
92
- const newServerConfig = {
93
- mcpServers: {
94
- posthog: (0, defaults_1.getDefaultServerConfig)(apiKey),
95
- },
96
- };
97
- const mergedConfig = (0, lodash_1.merge)({}, existingConfig, newServerConfig);
98
- await fs.promises.writeFile(configPath, JSON.stringify(mergedConfig, null, 2), 'utf8');
99
- }
100
- async removeServer() {
101
- const configPath = this.getConfigPath();
102
- if (!fs.existsSync(configPath)) {
103
- return;
104
- }
105
- try {
106
- const configContent = await fs.promises.readFile(configPath, 'utf8');
107
- const config = exports.ClaudeMCPConfig.parse(JSON.parse(configContent));
108
- if (config.mcpServers && 'posthog' in config.mcpServers) {
109
- delete config.mcpServers.posthog;
110
- await fs.promises.writeFile(configPath, JSON.stringify(config, null, 2), 'utf8');
111
- }
112
- }
113
- catch {
114
- // If we can't read or parse the config, there's nothing to remove
115
- }
116
- }
117
62
  }
118
63
  exports.ClaudeMCPClient = ClaudeMCPClient;
119
64
  //# sourceMappingURL=claude.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"claude.js","sourceRoot":"","sources":["../../../../../src/steps/add-mcp-server-to-clients/clients/claude.ts"],"names":[],"mappings":"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","sourcesContent":["import { MCPClient } from '../MCPClient';\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as os from 'os';\nimport { DefaultMCPClientConfig, getDefaultServerConfig } from '../defaults';\nimport { z } from 'zod';\nimport { merge } from 'lodash';\n\nexport const ClaudeMCPConfig = DefaultMCPClientConfig;\n\nexport type ClaudeMCPConfig = z.infer<typeof DefaultMCPClientConfig>;\n\nexport class ClaudeMCPClient extends MCPClient {\n name = 'Claude Desktop';\n\n constructor() {\n super();\n }\n\n isClientSupported(): boolean {\n return process.platform === 'darwin' || process.platform === 'win32';\n }\n\n private getConfigPath(): string {\n const homeDir = os.homedir();\n const isWindows = process.platform === 'win32';\n const isMac = process.platform === 'darwin';\n\n if (isMac) {\n return path.join(\n homeDir,\n 'Library',\n 'Application Support',\n 'Claude',\n 'claude_desktop_config.json',\n );\n }\n\n if (isWindows) {\n return path.join(\n process.env.APPDATA || '',\n 'Claude',\n 'claude_desktop_config.json',\n );\n }\n\n throw new Error(`Unsupported platform: ${process.platform}`);\n }\n\n async isServerInstalled(): Promise<boolean> {\n try {\n const configPath = this.getConfigPath();\n\n if (!fs.existsSync(configPath)) {\n return false;\n }\n\n const configContent = await fs.promises.readFile(configPath, 'utf8');\n const config = ClaudeMCPConfig.parse(JSON.parse(configContent));\n\n return 'posthog' in config.mcpServers;\n } catch {\n return false;\n }\n }\n\n async addServer(apiKey: string): Promise<void> {\n const configPath = this.getConfigPath();\n const configDir = path.dirname(configPath);\n\n await fs.promises.mkdir(configDir, { recursive: true });\n\n let existingConfig: ClaudeMCPConfig = { mcpServers: {} };\n\n if (fs.existsSync(configPath)) {\n try {\n const existingContent = await fs.promises.readFile(configPath, 'utf8');\n existingConfig = ClaudeMCPConfig.parse(JSON.parse(existingContent));\n } catch {\n existingConfig = { mcpServers: {} };\n }\n }\n\n const newServerConfig = {\n mcpServers: {\n posthog: getDefaultServerConfig(apiKey),\n },\n };\n\n const mergedConfig = merge({}, existingConfig, newServerConfig);\n\n await fs.promises.writeFile(\n configPath,\n JSON.stringify(mergedConfig, null, 2),\n 'utf8',\n );\n }\n\n async removeServer(): Promise<void> {\n const configPath = this.getConfigPath();\n\n if (!fs.existsSync(configPath)) {\n return;\n }\n\n try {\n const configContent = await fs.promises.readFile(configPath, 'utf8');\n const config = ClaudeMCPConfig.parse(JSON.parse(configContent));\n\n if (config.mcpServers && 'posthog' in config.mcpServers) {\n delete config.mcpServers.posthog;\n\n await fs.promises.writeFile(\n configPath,\n JSON.stringify(config, null, 2),\n 'utf8',\n );\n }\n } catch {\n // If we can't read or parse the config, there's nothing to remove\n }\n }\n}\n"]}
1
+ {"version":3,"file":"claude.js","sourceRoot":"","sources":["../../../../../src/steps/add-mcp-server-to-clients/clients/claude.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,4CAAgD;AAChD,2CAA6B;AAC7B,uCAAyB;AACzB,0CAAqD;AAGxC,QAAA,eAAe,GAAG,iCAAsB,CAAC;AAItD,MAAa,eAAgB,SAAQ,4BAAgB;IACnD,IAAI,GAAG,gBAAgB,CAAC;IAExB;QACE,KAAK,EAAE,CAAC;IACV,CAAC;IAED,KAAK,CAAC,iBAAiB;QACrB,OAAO,OAAO,CAAC,OAAO,CACpB,OAAO,CAAC,QAAQ,KAAK,QAAQ,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,CAC9D,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,aAAa;QACjB,MAAM,OAAO,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC;QAC7B,MAAM,SAAS,GAAG,OAAO,CAAC,QAAQ,KAAK,OAAO,CAAC;QAC/C,MAAM,KAAK,GAAG,OAAO,CAAC,QAAQ,KAAK,QAAQ,CAAC;QAE5C,IAAI,KAAK,EAAE,CAAC;YACV,OAAO,OAAO,CAAC,OAAO,CACpB,IAAI,CAAC,IAAI,CACP,OAAO,EACP,SAAS,EACT,qBAAqB,EACrB,QAAQ,EACR,4BAA4B,CAC7B,CACF,CAAC;QACJ,CAAC;QAED,IAAI,SAAS,EAAE,CAAC;YACd,OAAO,OAAO,CAAC,OAAO,CACpB,IAAI,CAAC,IAAI,CACP,OAAO,CAAC,GAAG,CAAC,OAAO,IAAI,EAAE,EACzB,QAAQ,EACR,4BAA4B,CAC7B,CACF,CAAC;QACJ,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,yBAAyB,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC;IAC/D,CAAC;CACF;AA1CD,0CA0CC","sourcesContent":["import { DefaultMCPClient } from '../MCPClient';\nimport * as path from 'path';\nimport * as os from 'os';\nimport { DefaultMCPClientConfig } from '../defaults';\nimport { z } from 'zod';\n\nexport const ClaudeMCPConfig = DefaultMCPClientConfig;\n\nexport type ClaudeMCPConfig = z.infer<typeof DefaultMCPClientConfig>;\n\nexport class ClaudeMCPClient extends DefaultMCPClient {\n name = 'Claude Desktop';\n\n constructor() {\n super();\n }\n\n async isClientSupported(): Promise<boolean> {\n return Promise.resolve(\n process.platform === 'darwin' || process.platform === 'win32',\n );\n }\n\n async getConfigPath(): Promise<string> {\n const homeDir = os.homedir();\n const isWindows = process.platform === 'win32';\n const isMac = process.platform === 'darwin';\n\n if (isMac) {\n return Promise.resolve(\n path.join(\n homeDir,\n 'Library',\n 'Application Support',\n 'Claude',\n 'claude_desktop_config.json',\n ),\n );\n }\n\n if (isWindows) {\n return Promise.resolve(\n path.join(\n process.env.APPDATA || '',\n 'Claude',\n 'claude_desktop_config.json',\n ),\n );\n }\n\n throw new Error(`Unsupported platform: ${process.platform}`);\n }\n}\n"]}
@@ -1,40 +1,53 @@
1
- import { MCPClient } from '../MCPClient';
1
+ import { DefaultMCPClient } from '../MCPClient';
2
2
  import { DefaultMCPClientConfig } from '../defaults';
3
3
  import { z } from 'zod';
4
4
  export declare const CursorMCPConfig: z.ZodObject<{
5
5
  mcpServers: z.ZodRecord<z.ZodString, z.ZodObject<{
6
- command: z.ZodString;
7
- args: z.ZodArray<z.ZodString, "many">;
8
- env: z.ZodRecord<z.ZodString, z.ZodString>;
6
+ command: z.ZodOptional<z.ZodString>;
7
+ args: z.ZodOptional<z.ZodArray<z.ZodString, "many">>;
8
+ env: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodString>>;
9
9
  }, "strip", z.ZodTypeAny, {
10
- command: string;
11
- args: string[];
12
- env: Record<string, string>;
10
+ command?: string | undefined;
11
+ args?: string[] | undefined;
12
+ env?: Record<string, string> | undefined;
13
13
  }, {
14
- command: string;
15
- args: string[];
16
- env: Record<string, string>;
14
+ command?: string | undefined;
15
+ args?: string[] | undefined;
16
+ env?: Record<string, string> | undefined;
17
17
  }>>;
18
- }, "strip", z.ZodTypeAny, {
19
- mcpServers: Record<string, {
20
- command: string;
21
- args: string[];
22
- env: Record<string, string>;
23
- }>;
24
- }, {
25
- mcpServers: Record<string, {
26
- command: string;
27
- args: string[];
28
- env: Record<string, string>;
29
- }>;
30
- }>;
18
+ }, "passthrough", z.ZodTypeAny, z.objectOutputType<{
19
+ mcpServers: z.ZodRecord<z.ZodString, z.ZodObject<{
20
+ command: z.ZodOptional<z.ZodString>;
21
+ args: z.ZodOptional<z.ZodArray<z.ZodString, "many">>;
22
+ env: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodString>>;
23
+ }, "strip", z.ZodTypeAny, {
24
+ command?: string | undefined;
25
+ args?: string[] | undefined;
26
+ env?: Record<string, string> | undefined;
27
+ }, {
28
+ command?: string | undefined;
29
+ args?: string[] | undefined;
30
+ env?: Record<string, string> | undefined;
31
+ }>>;
32
+ }, z.ZodTypeAny, "passthrough">, z.objectInputType<{
33
+ mcpServers: z.ZodRecord<z.ZodString, z.ZodObject<{
34
+ command: z.ZodOptional<z.ZodString>;
35
+ args: z.ZodOptional<z.ZodArray<z.ZodString, "many">>;
36
+ env: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodString>>;
37
+ }, "strip", z.ZodTypeAny, {
38
+ command?: string | undefined;
39
+ args?: string[] | undefined;
40
+ env?: Record<string, string> | undefined;
41
+ }, {
42
+ command?: string | undefined;
43
+ args?: string[] | undefined;
44
+ env?: Record<string, string> | undefined;
45
+ }>>;
46
+ }, z.ZodTypeAny, "passthrough">>;
31
47
  export type CursorMCPConfig = z.infer<typeof DefaultMCPClientConfig>;
32
- export declare class CursorMCPClient extends MCPClient {
48
+ export declare class CursorMCPClient extends DefaultMCPClient {
33
49
  name: string;
34
50
  constructor();
35
- isClientSupported(): boolean;
36
- private getConfigPath;
37
- isServerInstalled(): Promise<boolean>;
38
- addServer(apiKey: string): Promise<void>;
39
- removeServer(): Promise<void>;
51
+ isClientSupported(): Promise<boolean>;
52
+ getConfigPath(): Promise<string>;
40
53
  }
@@ -35,77 +35,20 @@ var __importStar = (this && this.__importStar) || (function () {
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
36
  exports.CursorMCPClient = exports.CursorMCPConfig = void 0;
37
37
  const MCPClient_1 = require("../MCPClient");
38
- const fs = __importStar(require("fs"));
39
38
  const path = __importStar(require("path"));
40
39
  const os = __importStar(require("os"));
41
40
  const defaults_1 = require("../defaults");
42
- const lodash_1 = require("lodash");
43
41
  exports.CursorMCPConfig = defaults_1.DefaultMCPClientConfig;
44
- class CursorMCPClient extends MCPClient_1.MCPClient {
42
+ class CursorMCPClient extends MCPClient_1.DefaultMCPClient {
45
43
  name = 'Cursor';
46
44
  constructor() {
47
45
  super();
48
46
  }
49
- isClientSupported() {
50
- return process.platform === 'darwin' || process.platform === 'win32';
47
+ async isClientSupported() {
48
+ return Promise.resolve(process.platform === 'darwin' || process.platform === 'win32');
51
49
  }
52
- getConfigPath() {
53
- return path.join(os.homedir(), '.cursor', 'mcp.json');
54
- }
55
- async isServerInstalled() {
56
- try {
57
- const configPath = this.getConfigPath();
58
- if (!fs.existsSync(configPath)) {
59
- return false;
60
- }
61
- const configContent = await fs.promises.readFile(configPath, 'utf8');
62
- const config = exports.CursorMCPConfig.parse(JSON.parse(configContent));
63
- return 'posthog' in config.mcpServers;
64
- }
65
- catch {
66
- return false;
67
- }
68
- }
69
- async addServer(apiKey) {
70
- const configPath = this.getConfigPath();
71
- const configDir = path.dirname(configPath);
72
- await fs.promises.mkdir(configDir, { recursive: true });
73
- let existingConfig = { mcpServers: {} };
74
- if (fs.existsSync(configPath)) {
75
- try {
76
- const existingContent = await fs.promises.readFile(configPath, 'utf8');
77
- existingConfig = exports.CursorMCPConfig.parse(JSON.parse(existingContent));
78
- }
79
- catch {
80
- existingConfig = { mcpServers: {} };
81
- }
82
- }
83
- const newServerConfig = {
84
- mcpServers: {
85
- posthog: (0, defaults_1.getDefaultServerConfig)(apiKey),
86
- },
87
- };
88
- const mergedConfig = (0, lodash_1.merge)({}, existingConfig, newServerConfig);
89
- await fs.promises.writeFile(configPath, JSON.stringify(mergedConfig, null, 2), 'utf8');
90
- }
91
- async removeServer() {
92
- const configPath = this.getConfigPath();
93
- if (!fs.existsSync(configPath)) {
94
- return;
95
- }
96
- let config;
97
- try {
98
- const configContent = await fs.promises.readFile(configPath, 'utf8');
99
- config = exports.CursorMCPConfig.parse(JSON.parse(configContent));
100
- }
101
- catch {
102
- // If we can't read or parse the config, there's nothing to remove
103
- return;
104
- }
105
- if (config.mcpServers && 'posthog' in config.mcpServers) {
106
- delete config.mcpServers.posthog;
107
- await fs.promises.writeFile(configPath, JSON.stringify(config, null, 2), 'utf8');
108
- }
50
+ async getConfigPath() {
51
+ return Promise.resolve(path.join(os.homedir(), '.cursor', 'mcp.json'));
109
52
  }
110
53
  }
111
54
  exports.CursorMCPClient = CursorMCPClient;
@@ -1 +1 @@
1
- {"version":3,"file":"cursor.js","sourceRoot":"","sources":["../../../../../src/steps/add-mcp-server-to-clients/clients/cursor.ts"],"names":[],"mappings":"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","sourcesContent":["import { MCPClient } from '../MCPClient';\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as os from 'os';\nimport { DefaultMCPClientConfig, getDefaultServerConfig } from '../defaults';\nimport { z } from 'zod';\nimport { merge } from 'lodash';\n\nexport const CursorMCPConfig = DefaultMCPClientConfig;\n\nexport type CursorMCPConfig = z.infer<typeof DefaultMCPClientConfig>;\n\nexport class CursorMCPClient extends MCPClient {\n name = 'Cursor';\n\n constructor() {\n super();\n }\n\n isClientSupported(): boolean {\n return process.platform === 'darwin' || process.platform === 'win32';\n }\n\n private getConfigPath(): string {\n return path.join(os.homedir(), '.cursor', 'mcp.json');\n }\n\n async isServerInstalled(): Promise<boolean> {\n try {\n const configPath = this.getConfigPath();\n\n if (!fs.existsSync(configPath)) {\n return false;\n }\n\n const configContent = await fs.promises.readFile(configPath, 'utf8');\n const config = CursorMCPConfig.parse(JSON.parse(configContent));\n\n return 'posthog' in config.mcpServers;\n } catch {\n return false;\n }\n }\n\n async addServer(apiKey: string): Promise<void> {\n const configPath = this.getConfigPath();\n const configDir = path.dirname(configPath);\n\n await fs.promises.mkdir(configDir, { recursive: true });\n\n let existingConfig: CursorMCPConfig = { mcpServers: {} };\n\n if (fs.existsSync(configPath)) {\n try {\n const existingContent = await fs.promises.readFile(configPath, 'utf8');\n existingConfig = CursorMCPConfig.parse(JSON.parse(existingContent));\n } catch {\n existingConfig = { mcpServers: {} };\n }\n }\n\n const newServerConfig = {\n mcpServers: {\n posthog: getDefaultServerConfig(apiKey),\n },\n };\n\n const mergedConfig = merge({}, existingConfig, newServerConfig);\n\n await fs.promises.writeFile(\n configPath,\n JSON.stringify(mergedConfig, null, 2),\n 'utf8',\n );\n }\n\n async removeServer(): Promise<void> {\n const configPath = this.getConfigPath();\n\n if (!fs.existsSync(configPath)) {\n return;\n }\n\n let config: CursorMCPConfig;\n\n try {\n const configContent = await fs.promises.readFile(configPath, 'utf8');\n config = CursorMCPConfig.parse(JSON.parse(configContent));\n } catch {\n // If we can't read or parse the config, there's nothing to remove\n return;\n }\n\n if (config.mcpServers && 'posthog' in config.mcpServers) {\n delete config.mcpServers.posthog;\n\n await fs.promises.writeFile(\n configPath,\n JSON.stringify(config, null, 2),\n 'utf8',\n );\n }\n }\n}\n"]}
1
+ {"version":3,"file":"cursor.js","sourceRoot":"","sources":["../../../../../src/steps/add-mcp-server-to-clients/clients/cursor.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,4CAAgD;AAChD,2CAA6B;AAC7B,uCAAyB;AACzB,0CAAqD;AAGxC,QAAA,eAAe,GAAG,iCAAsB,CAAC;AAItD,MAAa,eAAgB,SAAQ,4BAAgB;IACnD,IAAI,GAAG,QAAQ,CAAC;IAEhB;QACE,KAAK,EAAE,CAAC;IACV,CAAC;IAED,KAAK,CAAC,iBAAiB;QACrB,OAAO,OAAO,CAAC,OAAO,CACpB,OAAO,CAAC,QAAQ,KAAK,QAAQ,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,CAC9D,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,aAAa;QACjB,OAAO,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,UAAU,CAAC,CAAC,CAAC;IACzE,CAAC;CACF;AAhBD,0CAgBC","sourcesContent":["import { DefaultMCPClient } from '../MCPClient';\nimport * as path from 'path';\nimport * as os from 'os';\nimport { DefaultMCPClientConfig } from '../defaults';\nimport { z } from 'zod';\n\nexport const CursorMCPConfig = DefaultMCPClientConfig;\n\nexport type CursorMCPConfig = z.infer<typeof DefaultMCPClientConfig>;\n\nexport class CursorMCPClient extends DefaultMCPClient {\n name = 'Cursor';\n\n constructor() {\n super();\n }\n\n async isClientSupported(): Promise<boolean> {\n return Promise.resolve(\n process.platform === 'darwin' || process.platform === 'win32',\n );\n }\n\n async getConfigPath(): Promise<string> {\n return Promise.resolve(path.join(os.homedir(), '.cursor', 'mcp.json'));\n }\n}\n"]}