@hubspot/cli 7.9.0 → 7.10.0-beta.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.
@@ -1,12 +1,18 @@
1
1
  import { UploadProjectTools } from '../UploadProjectTools.js';
2
+ import { getAllHsProfiles } from '@hubspot/project-parsing-lib';
3
+ import { getProjectConfig } from '../../../../lib/projects/config.js';
2
4
  import { runCommandInDir } from '../../../utils/project.js';
3
5
  import { mcpFeedbackRequest } from '../../../utils/feedbackTracking.js';
4
6
  vi.mock('@modelcontextprotocol/sdk/server/mcp.js');
7
+ vi.mock('@hubspot/project-parsing-lib');
8
+ vi.mock('../../../../lib/projects/config.js');
5
9
  vi.mock('../../../utils/project');
6
10
  vi.mock('../../../utils/toolUsageTracking');
7
11
  vi.mock('../../../utils/feedbackTracking');
8
12
  const mockMcpFeedbackRequest = mcpFeedbackRequest;
9
13
  const mockRunCommandInDir = runCommandInDir;
14
+ const mockGetProjectConfig = getProjectConfig;
15
+ const mockGetAllHsProfiles = getAllHsProfiles;
10
16
  describe('mcp-server/tools/project/UploadProjectTools', () => {
11
17
  let mockMcpServer;
12
18
  let tool;
@@ -20,6 +26,15 @@ describe('mcp-server/tools/project/UploadProjectTools', () => {
20
26
  mockRegisteredTool = {};
21
27
  mockMcpServer.registerTool.mockReturnValue(mockRegisteredTool);
22
28
  mockMcpFeedbackRequest.mockResolvedValue('');
29
+ mockGetProjectConfig.mockResolvedValue({
30
+ projectConfig: {
31
+ srcDir: 'src',
32
+ name: 'test-project',
33
+ platformVersion: '2025.2',
34
+ },
35
+ projectDir: '/test/project',
36
+ });
37
+ mockGetAllHsProfiles.mockResolvedValue([]);
23
38
  tool = new UploadProjectTools(mockMcpServer);
24
39
  });
25
40
  describe('register', () => {
@@ -37,6 +52,7 @@ describe('mcp-server/tools/project/UploadProjectTools', () => {
37
52
  const input = {
38
53
  absoluteCurrentWorkingDirectory: '/test/dir',
39
54
  absoluteProjectPath: '/test/project',
55
+ uploadMessage: 'Test upload message',
40
56
  };
41
57
  it('should upload project successfully', async () => {
42
58
  mockRunCommandInDir.mockResolvedValue({
@@ -44,7 +60,9 @@ describe('mcp-server/tools/project/UploadProjectTools', () => {
44
60
  stderr: '',
45
61
  });
46
62
  const result = await tool.handler(input);
47
- expect(mockRunCommandInDir).toHaveBeenCalledWith('/test/project', 'hs project upload --force-create');
63
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/test/project', expect.stringContaining('hs project upload'));
64
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/test/project', expect.stringContaining('--force-create'));
65
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/test/project', expect.stringContaining('--message "Test upload message"'));
48
66
  expect(result).toEqual({
49
67
  content: [
50
68
  { type: 'text', text: 'Project uploaded successfully' },
@@ -68,13 +86,44 @@ describe('mcp-server/tools/project/UploadProjectTools', () => {
68
86
  mockRunCommandInDir.mockRejectedValue(error);
69
87
  await expect(tool.handler(input)).rejects.toThrow('Upload failed');
70
88
  });
71
- it('should use force-create flag', async () => {
89
+ it('should use force-create and message flags', async () => {
72
90
  mockRunCommandInDir.mockResolvedValue({
73
91
  stdout: 'Project created and uploaded',
74
92
  stderr: '',
75
93
  });
76
94
  await tool.handler(input);
77
- expect(mockRunCommandInDir).toHaveBeenCalledWith('/test/project', 'hs project upload --force-create');
95
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/test/project', expect.stringContaining('hs project upload'));
96
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/test/project', expect.stringContaining('--force-create'));
97
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/test/project', expect.stringContaining('--message "Test upload message"'));
98
+ });
99
+ it('should use profiles', async () => {
100
+ mockRunCommandInDir.mockResolvedValue({
101
+ stdout: 'Project created and uploaded',
102
+ stderr: '',
103
+ });
104
+ await tool.handler({
105
+ ...input,
106
+ profile: 'dev',
107
+ });
108
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/test/project', expect.stringContaining('hs project upload'));
109
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/test/project', expect.stringContaining('--force-create'));
110
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/test/project', expect.stringContaining('--message "Test upload message"'));
111
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/test/project', expect.stringContaining('--profile "dev"'));
112
+ });
113
+ it('should prompt for profile if not specified and the project requires them', async () => {
114
+ mockGetAllHsProfiles.mockResolvedValue(['prod', 'dev']);
115
+ mockRunCommandInDir.mockResolvedValue({
116
+ stdout: 'Project created and uploaded',
117
+ stderr: '',
118
+ });
119
+ const result = await tool.handler(input);
120
+ expect(mockRunCommandInDir).not.toHaveBeenCalled();
121
+ expect(result.content).toEqual([
122
+ {
123
+ type: 'text',
124
+ text: 'Ask the user which profile they would like to use for the upload.',
125
+ },
126
+ ]);
78
127
  });
79
128
  it('should handle empty stdout and stderr', async () => {
80
129
  mockRunCommandInDir.mockResolvedValue({
@@ -95,9 +144,12 @@ describe('mcp-server/tools/project/UploadProjectTools', () => {
95
144
  const differentInput = {
96
145
  absoluteCurrentWorkingDirectory: '/test/dir',
97
146
  absoluteProjectPath: '/different/path/to/project',
147
+ uploadMessage: 'Different test upload message',
98
148
  };
99
149
  await tool.handler(differentInput);
100
- expect(mockRunCommandInDir).toHaveBeenCalledWith('/different/path/to/project', 'hs project upload --force-create');
150
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/different/path/to/project', expect.stringContaining('hs project upload'));
151
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/different/path/to/project', expect.stringContaining('--force-create'));
152
+ expect(mockRunCommandInDir).toHaveBeenCalledWith('/different/path/to/project', expect.stringContaining('--message "Different test upload message"'));
101
153
  });
102
154
  it('should handle very long output', async () => {
103
155
  const longOutput = 'A'.repeat(10000);
package/package.json CHANGED
@@ -1,12 +1,12 @@
1
1
  {
2
2
  "name": "@hubspot/cli",
3
- "version": "7.9.0",
3
+ "version": "7.10.0-beta.0",
4
4
  "description": "The official CLI for developing on HubSpot",
5
5
  "license": "Apache-2.0",
6
6
  "repository": "https://github.com/HubSpot/hubspot-cli",
7
7
  "type": "module",
8
8
  "dependencies": {
9
- "@hubspot/local-dev-lib": "3.21.1",
9
+ "@hubspot/local-dev-lib": "3.21.2",
10
10
  "@hubspot/project-parsing-lib": "0.10.1",
11
11
  "@hubspot/serverless-dev-runtime": "7.0.6",
12
12
  "@hubspot/theme-preview-dev-server": "0.0.10",