@hubspot/cli 7.10.0 → 7.10.1-experimental.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 (199) hide show
  1. package/bin/cli.js +5 -4
  2. package/commands/__tests__/getStarted.test.js +10 -0
  3. package/commands/__tests__/project.test.js +3 -0
  4. package/commands/account/__tests__/rename.test.js +10 -3
  5. package/commands/account/auth.js +10 -14
  6. package/commands/account/clean.js +11 -19
  7. package/commands/account/createOverride.js +15 -11
  8. package/commands/account/info.js +8 -5
  9. package/commands/account/list.js +15 -19
  10. package/commands/account/remove.js +23 -22
  11. package/commands/account/removeOverride.js +6 -6
  12. package/commands/account/rename.js +2 -2
  13. package/commands/account/use.js +19 -8
  14. package/commands/app/__tests__/migrate.test.js +8 -4
  15. package/commands/app/migrate.js +2 -2
  16. package/commands/auth.js +18 -14
  17. package/commands/config/migrate.js +5 -5
  18. package/commands/customObject/createSchema.js +2 -3
  19. package/commands/customObject/updateSchema.js +2 -3
  20. package/commands/getStarted.js +2 -3
  21. package/commands/hubdb/__tests__/list.test.js +1 -0
  22. package/commands/hubdb/list.js +2 -2
  23. package/commands/init.js +36 -32
  24. package/commands/project/__tests__/deploy.test.js +10 -5
  25. package/commands/project/__tests__/devUnifiedFlow.test.js +6 -4
  26. package/commands/project/__tests__/lint.test.js +709 -0
  27. package/commands/project/__tests__/logs.test.js +4 -0
  28. package/commands/project/__tests__/validate.test.js +286 -28
  29. package/commands/project/cloneApp.js +2 -2
  30. package/commands/project/deploy.js +16 -8
  31. package/commands/project/dev/deprecatedFlow.js +4 -5
  32. package/commands/project/dev/index.js +19 -7
  33. package/commands/project/dev/unifiedFlow.js +4 -5
  34. package/commands/project/lint.d.ts +6 -0
  35. package/commands/project/lint.js +178 -0
  36. package/commands/project/logs.js +2 -3
  37. package/commands/project/migrate.js +4 -13
  38. package/commands/project/profile/add.js +6 -7
  39. package/commands/project/profile/delete.js +2 -2
  40. package/commands/project/upload.js +10 -4
  41. package/commands/project/validate.js +73 -13
  42. package/commands/project.js +2 -0
  43. package/commands/sandbox/__tests__/create.test.js +14 -5
  44. package/commands/sandbox/create.js +4 -5
  45. package/commands/sandbox/delete.js +23 -20
  46. package/commands/testAccount/create.js +2 -2
  47. package/commands/testAccount/delete.js +9 -8
  48. package/lang/en.d.ts +48 -11
  49. package/lang/en.js +58 -15
  50. package/lib/__tests__/buildAccount.test.js +22 -30
  51. package/lib/__tests__/commonOpts.test.js +9 -13
  52. package/lib/__tests__/developerTestAccounts.test.js +29 -17
  53. package/lib/__tests__/importData.test.js +20 -10
  54. package/lib/__tests__/oauth.test.js +19 -8
  55. package/lib/__tests__/projectProfiles.test.js +273 -32
  56. package/lib/__tests__/sandboxSync.test.js +33 -11
  57. package/lib/__tests__/sandboxes.test.js +30 -19
  58. package/lib/__tests__/usageTracking.test.js +10 -10
  59. package/lib/__tests__/validation.test.js +32 -32
  60. package/lib/accountTypes.d.ts +9 -9
  61. package/lib/accountTypes.js +2 -4
  62. package/lib/app/__tests__/migrate.test.js +15 -0
  63. package/lib/app/__tests__/migrate_legacy.test.js +9 -0
  64. package/lib/app/migrate_legacy.d.ts +2 -2
  65. package/lib/buildAccount.d.ts +4 -4
  66. package/lib/buildAccount.js +7 -14
  67. package/lib/commonOpts.js +3 -3
  68. package/lib/configMigrate.d.ts +2 -2
  69. package/lib/configMigrate.js +42 -18
  70. package/lib/configOptions.js +3 -2
  71. package/lib/developerTestAccounts.d.ts +3 -3
  72. package/lib/developerTestAccounts.js +4 -7
  73. package/lib/doctor/DiagnosticInfoBuilder.d.ts +1 -1
  74. package/lib/doctor/DiagnosticInfoBuilder.js +9 -6
  75. package/lib/doctor/Doctor.js +4 -3
  76. package/lib/doctor/__tests__/Diagnosis.test.js +4 -3
  77. package/lib/doctor/__tests__/DiagnosticInfoBuilder.test.js +17 -9
  78. package/lib/doctor/__tests__/Doctor.test.js +14 -0
  79. package/lib/importData.js +8 -7
  80. package/lib/links.js +5 -5
  81. package/lib/middleware/{__test__ → __tests__}/commandTargetingUtils.test.js +3 -3
  82. package/lib/middleware/{__test__ → __tests__}/configMiddleware.test.js +23 -22
  83. package/lib/middleware/{__test__ → __tests__}/gitMiddleware.test.js +9 -7
  84. package/lib/middleware/autoUpdateMiddleware.js +34 -23
  85. package/lib/middleware/commandTargetingUtils.js +3 -2
  86. package/lib/middleware/configMiddleware.d.ts +6 -1
  87. package/lib/middleware/configMiddleware.js +36 -15
  88. package/lib/middleware/fireAlarmMiddleware.js +4 -15
  89. package/lib/middleware/gitMiddleware.js +8 -4
  90. package/lib/oauth.d.ts +2 -2
  91. package/lib/oauth.js +8 -10
  92. package/lib/projectProfiles.d.ts +4 -3
  93. package/lib/projectProfiles.js +78 -32
  94. package/lib/projects/__tests__/AppDevModeInterface.test.js +17 -6
  95. package/lib/projects/__tests__/DevServerManager.test.js +1 -0
  96. package/lib/projects/__tests__/LocalDevProcess.test.js +1 -0
  97. package/lib/projects/__tests__/deploy.test.js +1 -0
  98. package/lib/projects/__tests__/uieLinting.test.js +640 -0
  99. package/lib/projects/create/__tests__/v2.test.js +11 -0
  100. package/lib/projects/localDev/AppDevModeInterface.js +2 -2
  101. package/lib/projects/localDev/DevServerManager_DEPRECATED.js +2 -2
  102. package/lib/projects/localDev/LocalDevLogger.js +4 -4
  103. package/lib/projects/localDev/LocalDevManager_DEPRECATED.js +3 -3
  104. package/lib/projects/localDev/helpers/account.d.ts +10 -10
  105. package/lib/projects/localDev/helpers/account.js +6 -11
  106. package/lib/projects/uieLinting.d.ts +33 -0
  107. package/lib/projects/uieLinting.js +222 -0
  108. package/lib/projects/urls.js +5 -6
  109. package/lib/prompts/__tests__/downloadProjectPrompt.test.js +7 -5
  110. package/lib/prompts/accountNamePrompt.js +3 -3
  111. package/lib/prompts/accountsPrompt.d.ts +1 -1
  112. package/lib/prompts/accountsPrompt.js +6 -7
  113. package/lib/prompts/confirmImportDataPrompt.js +2 -2
  114. package/lib/prompts/downloadProjectPrompt.d.ts +1 -0
  115. package/lib/prompts/downloadProjectPrompt.js +5 -2
  116. package/lib/prompts/importDataTestAccountSelectPrompt.js +4 -5
  117. package/lib/prompts/personalAccessKeyPrompt.js +2 -2
  118. package/lib/prompts/projectDevTargetAccountPrompt.d.ts +3 -3
  119. package/lib/prompts/projectDevTargetAccountPrompt.js +5 -7
  120. package/lib/prompts/sandboxesPrompt.js +7 -8
  121. package/lib/prompts/setAsDefaultAccountPrompt.js +7 -6
  122. package/lib/sandboxSync.d.ts +2 -2
  123. package/lib/sandboxSync.js +3 -9
  124. package/lib/sandboxes.d.ts +4 -4
  125. package/lib/sandboxes.js +6 -11
  126. package/lib/serverlessLogs.js +2 -2
  127. package/lib/theme/__tests__/migrate.test.js +15 -0
  128. package/lib/ui/index.js +6 -3
  129. package/lib/usageTracking.js +15 -8
  130. package/lib/validation.js +13 -11
  131. package/mcp-server/tools/cms/HsCreateFunctionTool.js +4 -2
  132. package/mcp-server/tools/cms/HsCreateModuleTool.js +4 -2
  133. package/mcp-server/tools/cms/HsCreateTemplateTool.js +4 -2
  134. package/mcp-server/tools/cms/HsFunctionLogsTool.js +4 -2
  135. package/mcp-server/tools/cms/HsListFunctionsTool.js +3 -1
  136. package/mcp-server/tools/cms/HsListTool.js +3 -1
  137. package/mcp-server/tools/cms/__tests__/HsCreateFunctionTool.test.js +1 -0
  138. package/mcp-server/tools/index.js +4 -0
  139. package/mcp-server/tools/project/AddFeatureToProjectTool.js +4 -2
  140. package/mcp-server/tools/project/CreateProjectTool.js +4 -2
  141. package/mcp-server/tools/project/CreateTestAccountTool.js +17 -7
  142. package/mcp-server/tools/project/DeployProjectTool.js +3 -1
  143. package/mcp-server/tools/project/DocFetchTool.js +6 -4
  144. package/mcp-server/tools/project/DocsSearchTool.d.ts +1 -1
  145. package/mcp-server/tools/project/DocsSearchTool.js +10 -8
  146. package/mcp-server/tools/project/GetApiUsagePatternsByAppIdTool.d.ts +1 -1
  147. package/mcp-server/tools/project/GetApiUsagePatternsByAppIdTool.js +9 -7
  148. package/mcp-server/tools/project/GetApplicationInfoTool.js +8 -6
  149. package/mcp-server/tools/project/GetBuildLogsTool.d.ts +26 -0
  150. package/mcp-server/tools/project/GetBuildLogsTool.js +125 -0
  151. package/mcp-server/tools/project/GetBuildStatusTool.d.ts +26 -0
  152. package/mcp-server/tools/project/GetBuildStatusTool.js +166 -0
  153. package/mcp-server/tools/project/GetConfigValuesTool.d.ts +1 -1
  154. package/mcp-server/tools/project/GetConfigValuesTool.js +9 -7
  155. package/mcp-server/tools/project/GuidedWalkthroughTool.d.ts +1 -1
  156. package/mcp-server/tools/project/GuidedWalkthroughTool.js +5 -3
  157. package/mcp-server/tools/project/UploadProjectTools.js +3 -1
  158. package/mcp-server/tools/project/ValidateProjectTool.js +4 -2
  159. package/mcp-server/tools/project/__tests__/CreateTestAccountTool.test.js +12 -2
  160. package/mcp-server/tools/project/__tests__/DocFetchTool.test.js +5 -1
  161. package/mcp-server/tools/project/__tests__/DocsSearchTool.test.js +23 -11
  162. package/mcp-server/tools/project/__tests__/GetApiUsagePatternsByAppIdTool.test.js +7 -5
  163. package/mcp-server/tools/project/__tests__/GetApplicationInfoTool.test.js +7 -5
  164. package/mcp-server/tools/project/__tests__/GetBuildLogsTool.test.d.ts +1 -0
  165. package/mcp-server/tools/project/__tests__/GetBuildLogsTool.test.js +305 -0
  166. package/mcp-server/tools/project/__tests__/GetBuildStatusTool.test.d.ts +1 -0
  167. package/mcp-server/tools/project/__tests__/GetBuildStatusTool.test.js +240 -0
  168. package/mcp-server/tools/project/__tests__/GetConfigValuesTool.test.js +8 -6
  169. package/mcp-server/utils/__tests__/content.test.js +21 -20
  170. package/mcp-server/utils/__tests__/feedbackTracking.test.js +34 -28
  171. package/mcp-server/utils/config.d.ts +1 -0
  172. package/mcp-server/utils/config.js +10 -0
  173. package/mcp-server/utils/content.d.ts +1 -1
  174. package/mcp-server/utils/content.js +2 -2
  175. package/mcp-server/utils/feedbackTracking.d.ts +1 -1
  176. package/mcp-server/utils/feedbackTracking.js +3 -3
  177. package/mcp-server/utils/toolUsageTracking.js +4 -3
  178. package/package.json +9 -9
  179. package/ui/components/BoxWithTitle.d.ts +2 -1
  180. package/ui/components/BoxWithTitle.js +2 -2
  181. package/ui/components/StatusMessageBoxes.d.ts +5 -4
  182. package/ui/components/StatusMessageBoxes.js +8 -8
  183. package/lib/middleware/__test__/notificationsMiddleware.test.js +0 -8
  184. package/lib/middleware/notificationsMiddleware.d.ts +0 -1
  185. package/lib/middleware/notificationsMiddleware.js +0 -28
  186. package/lib/ui/boxen.d.ts +0 -5
  187. package/lib/ui/boxen.js +0 -26
  188. package/mcp-server/utils/__tests__/cliConfig.test.js +0 -110
  189. package/mcp-server/utils/cliConfig.d.ts +0 -1
  190. package/mcp-server/utils/cliConfig.js +0 -12
  191. /package/{lib/middleware/__test__/commandTargetingUtils.test.d.ts → commands/project/__tests__/lint.test.d.ts} +0 -0
  192. /package/lib/middleware/{__test__/configMiddleware.test.d.ts → __tests__/commandTargetingUtils.test.d.ts} +0 -0
  193. /package/lib/middleware/{__test__/gitMiddleware.test.d.ts → __tests__/configMiddleware.test.d.ts} +0 -0
  194. /package/lib/middleware/{__test__/notificationsMiddleware.test.d.ts → __tests__/gitMiddleware.test.d.ts} +0 -0
  195. /package/lib/middleware/{__test__ → __tests__}/requestMiddleware.test.d.ts +0 -0
  196. /package/lib/middleware/{__test__ → __tests__}/requestMiddleware.test.js +0 -0
  197. /package/lib/middleware/{__test__ → __tests__}/yargsChecksMiddleware.test.d.ts +0 -0
  198. /package/lib/middleware/{__test__ → __tests__}/yargsChecksMiddleware.test.js +0 -0
  199. /package/{mcp-server/utils/__tests__/cliConfig.test.d.ts → lib/projects/__tests__/uieLinting.test.d.ts} +0 -0
@@ -3,11 +3,16 @@ import { CLIOptions } from '@hubspot/local-dev-lib/types/CLIOptions';
3
3
  export declare function handleDeprecatedEnvVariables(argv: Arguments<{
4
4
  useEnv?: boolean;
5
5
  }>): void;
6
+ export declare function handleCustomConfigLocationMiddleware(argv: Arguments<{
7
+ useEnv?: boolean;
8
+ config?: string;
9
+ }>): void;
6
10
  /**
7
11
  * Auto-injects the derivedAccountId flag into all commands
8
12
  */
9
13
  export declare function injectAccountIdMiddleware(argv: Arguments<{
10
14
  account?: string;
15
+ config?: string;
11
16
  }>): Promise<void>;
12
- export declare function loadAndValidateConfigMiddleware(argv: Arguments<CLIOptions>): Promise<void>;
17
+ export declare function validateConfigMiddleware(argv: Arguments<CLIOptions>): Promise<void>;
13
18
  export declare function validateAccountOptions(argv: Arguments): Promise<void>;
@@ -1,9 +1,11 @@
1
- import { loadConfig, getAccountId, configFileExists, getConfigPath, validateConfig, } from '@hubspot/local-dev-lib/config';
1
+ import path from 'path';
2
+ import { getConfigAccountIfExists, validateConfig, getConfigDefaultAccountIfExists, configFileExists, } from '@hubspot/local-dev-lib/config';
3
+ import { getCwd } from '@hubspot/local-dev-lib/path';
2
4
  import { validateAccount } from '../validation.js';
3
5
  import { EXIT_CODES } from '../enums/exitCodes.js';
4
6
  import { commands } from '../../lang/en.js';
5
7
  import { uiDeprecatedTag } from '../ui/index.js';
6
- import { isTargetedCommand, shouldLoadConfigForCommand, shouldRunAccountValidationForCommand, shouldRunConfigValidationForCommand, } from './commandTargetingUtils.js';
8
+ import { shouldLoadConfigForCommand, shouldRunAccountValidationForCommand, shouldRunConfigValidationForCommand, } from './commandTargetingUtils.js';
7
9
  import { parseStringToNumber } from '../parsing.js';
8
10
  import { uiLogger } from '../ui/logger.js';
9
11
  import { lib } from '../../lang/en.js';
@@ -17,6 +19,18 @@ export function handleDeprecatedEnvVariables(argv) {
17
19
  process.env.HUBSPOT_ACCOUNT_ID = process.env.HUBSPOT_PORTAL_ID;
18
20
  }
19
21
  }
22
+ export function handleCustomConfigLocationMiddleware(argv) {
23
+ const { useEnv, config } = argv;
24
+ if (useEnv) {
25
+ process.env.USE_ENVIRONMENT_HUBSPOT_CONFIG = 'true';
26
+ }
27
+ else if (config && typeof config === 'string') {
28
+ const absoluteConfigPath = path.isAbsolute(config)
29
+ ? config
30
+ : path.join(getCwd(), config);
31
+ process.env.HUBSPOT_CONFIG_PATH = absoluteConfigPath;
32
+ }
33
+ }
20
34
  /**
21
35
  * Auto-injects the derivedAccountId flag into all commands
22
36
  */
@@ -33,10 +47,23 @@ export async function injectAccountIdMiddleware(argv) {
33
47
  }
34
48
  }
35
49
  else {
36
- argv.derivedAccountId = getAccountId(account);
50
+ // Wrap in try-catch to handle cases where config file doesn't exist yet (e.g., during hs init)
51
+ try {
52
+ let accountInConfig = account
53
+ ? getConfigAccountIfExists(account)
54
+ : undefined;
55
+ if (!accountInConfig) {
56
+ accountInConfig = getConfigDefaultAccountIfExists();
57
+ }
58
+ argv.derivedAccountId = accountInConfig?.accountId;
59
+ }
60
+ catch (err) {
61
+ // Config file doesn't exist yet, which is fine for commands like hs init
62
+ argv.derivedAccountId = undefined;
63
+ }
37
64
  }
38
65
  }
39
- export async function loadAndValidateConfigMiddleware(argv) {
66
+ export async function validateConfigMiddleware(argv) {
40
67
  // Skip this when no command is provided
41
68
  if (!argv._.length || argv.help) {
42
69
  return;
@@ -45,23 +72,17 @@ export async function loadAndValidateConfigMiddleware(argv) {
45
72
  if (!shouldLoadConfigForCommand(argv._)) {
46
73
  return;
47
74
  }
48
- // If the config file exists and the --config flag is used, exit with an error
49
- if (configFileExists(true) &&
50
- argv.config &&
51
- !isTargetedCommand(argv._, { config: { migrate: true } })) {
52
- uiLogger.error(commands.generalErrors.loadConfigMiddleware.configFileExists(getConfigPath()));
53
- process.exit(EXIT_CODES.ERROR);
54
- }
55
- const config = loadConfig(argv.config, argv);
56
75
  // We don't run validation for auth because users should be able to run it when
57
76
  // no accounts are configured, but we still want to exit if the config file is not found
58
- if (isTargetedCommand(argv._, { auth: true }) && !config) {
77
+ if (!process.env.USE_ENVIRONMENT_HUBSPOT_CONFIG && !configFileExists()) {
78
+ console.error('Config file not found, run hs account auth to configure your account');
59
79
  process.exit(EXIT_CODES.ERROR);
60
80
  }
61
81
  // Only validate the config if the command requires it
62
82
  if (shouldRunConfigValidationForCommand(argv._)) {
63
- const configIsValid = validateConfig();
64
- if (!configIsValid) {
83
+ const { isValid, errors } = validateConfig();
84
+ if (!isValid) {
85
+ uiLogger.error(commands.generalErrors.validateConfigMiddleware.configValidationFailed(errors));
65
86
  process.exit(EXIT_CODES.ERROR);
66
87
  }
67
88
  }
@@ -2,7 +2,6 @@ import chalk from 'chalk';
2
2
  import { fetchFireAlarms } from '@hubspot/local-dev-lib/api/fireAlarm';
3
3
  import { debugError } from '../errorHandlers/index.js';
4
4
  import { pkg } from '../jsonLoader.js';
5
- import { logInBox } from '../ui/boxen.js';
6
5
  import { renderInline } from '../../ui/index.js';
7
6
  import { getWarningBox } from '../../ui/components/StatusMessageBoxes.js';
8
7
  /*
@@ -100,20 +99,10 @@ async function logFireAlarms(accountId, command, version) {
100
99
  }
101
100
  return acc;
102
101
  }, '');
103
- if (!process.env.HUBSPOT_ENABLE_INK) {
104
- await logInBox({
105
- contents: notifications,
106
- options: {
107
- title: 'Notifications',
108
- },
109
- });
110
- }
111
- else {
112
- await renderInline(getWarningBox({
113
- title: 'Notifications',
114
- message: notifications,
115
- }));
116
- }
102
+ await renderInline(getWarningBox({
103
+ title: 'Notifications',
104
+ message: notifications,
105
+ }));
117
106
  }
118
107
  }
119
108
  export async function checkFireAlarms(argv) {
@@ -1,15 +1,19 @@
1
- import { getConfigPath, configFileExists } from '@hubspot/local-dev-lib/config';
1
+ import { getConfigFilePath, globalConfigFileExists, } from '@hubspot/local-dev-lib/config';
2
2
  import { checkAndWarnGitInclusion } from '../ui/git.js';
3
+ import { debugError } from '../errorHandlers/index.js';
3
4
  export function checkAndWarnGitInclusionMiddleware(argv) {
4
5
  // Skip this when no command is provided
5
6
  if (argv._.length) {
6
7
  // Skip if using global config
7
- if (configFileExists(true)) {
8
+ if (globalConfigFileExists()) {
8
9
  return;
9
10
  }
10
- const configPath = getConfigPath();
11
- if (configPath) {
11
+ try {
12
+ const configPath = getConfigFilePath();
12
13
  checkAndWarnGitInclusion(configPath);
13
14
  }
15
+ catch (error) {
16
+ debugError(error);
17
+ }
14
18
  }
15
19
  }
package/lib/oauth.d.ts CHANGED
@@ -1,2 +1,2 @@
1
- import { OAuth2ManagerAccountConfig } from '@hubspot/local-dev-lib/types/Accounts';
2
- export declare function authenticateWithOauth(accountConfig: OAuth2ManagerAccountConfig): Promise<void>;
1
+ import { OAuthConfigAccount } from '@hubspot/local-dev-lib/types/Accounts';
2
+ export declare function authenticateWithOauth(accountConfig: OAuthConfigAccount): Promise<void>;
package/lib/oauth.js CHANGED
@@ -1,8 +1,7 @@
1
1
  import express from 'express';
2
2
  import open from 'open';
3
3
  import { OAuth2Manager } from '@hubspot/local-dev-lib/models/OAuth2Manager';
4
- import { getAccountConfig } from '@hubspot/local-dev-lib/config';
5
- import { getAccountIdentifier } from '@hubspot/local-dev-lib/config/getAccountIdentifier';
4
+ import { getConfigAccountById } from '@hubspot/local-dev-lib/config';
6
5
  import { addOauthToAccountConfig } from '@hubspot/local-dev-lib/oauth';
7
6
  import { getHubSpotWebsiteOrigin } from '@hubspot/local-dev-lib/urls';
8
7
  import { uiLogger } from './ui/logger.js';
@@ -14,15 +13,15 @@ import { EXIT_CODES } from './enums/exitCodes.js';
14
13
  const PORT = 3000;
15
14
  const redirectUri = `http://localhost:${PORT}/oauth-callback`;
16
15
  function buildAuthUrl(oauthManager) {
17
- const { env: accountEnv, clientId, scopes: accountScopes, } = oauthManager.account;
16
+ const { env: accountEnv, auth } = oauthManager.account;
18
17
  const env = accountEnv || ENVIRONMENTS.PROD;
19
- const scopes = accountScopes || DEFAULT_OAUTH_SCOPES;
20
- if (!clientId) {
18
+ const scopes = auth.scopes.length > 0 ? auth.scopes : DEFAULT_OAUTH_SCOPES;
19
+ if (!auth.clientId) {
21
20
  uiLogger.error(lib.oauth.missingClientId);
22
21
  process.exit(EXIT_CODES.ERROR);
23
22
  }
24
23
  return (`${getHubSpotWebsiteOrigin(env)}/oauth/${oauthManager.account.accountId}/authorize` +
25
- `?client_id=${encodeURIComponent(clientId)}` + // app's client ID
24
+ `?client_id=${encodeURIComponent(auth.clientId)}` + // app's client ID
26
25
  `&scope=${encodeURIComponent(scopes.join(' '))}` + // scopes being requested by the app
27
26
  `&redirect_uri=${encodeURIComponent(redirectUri)}` // where to send the user after the consent page
28
27
  );
@@ -45,8 +44,8 @@ async function authorize(oauthManager) {
45
44
  if (req.query.code) {
46
45
  const authCodeProof = {
47
46
  grant_type: 'authorization_code',
48
- client_id: oauthManager.account.clientId,
49
- client_secret: oauthManager.account.clientSecret,
47
+ client_id: oauthManager.account.auth.clientId,
48
+ client_secret: oauthManager.account.auth.clientSecret,
50
49
  redirect_uri: redirectUri,
51
50
  code: req.query.code,
52
51
  };
@@ -82,8 +81,7 @@ async function authorize(oauthManager) {
82
81
  });
83
82
  }
84
83
  function setupOauth(accountConfig) {
85
- const accountId = getAccountIdentifier(accountConfig);
86
- const config = getAccountConfig(accountId);
84
+ const config = getConfigAccountById(accountConfig.accountId);
87
85
  return new OAuth2Manager({
88
86
  ...accountConfig,
89
87
  env: accountConfig.env || config?.env || ENVIRONMENTS.PROD,
@@ -2,6 +2,7 @@ import { HsProfileFile } from '@hubspot/project-parsing-lib/src/lib/types.js';
2
2
  import { ProjectConfig } from '../types/Projects.js';
3
3
  export declare function logProfileHeader(profileName: string): void;
4
4
  export declare function logProfileFooter(profile: HsProfileFile, includeVariables?: boolean): void;
5
- export declare function loadProfile(projectConfig: ProjectConfig | null, projectDir: string | null, profileName: string): HsProfileFile | undefined;
6
- export declare function exitIfUsingProfiles(projectConfig: ProjectConfig | null, projectDir: string | null): Promise<void>;
7
- export declare function loadAndValidateProfile(projectConfig: ProjectConfig | null, projectDir: string | null, argsProfile: string | undefined): Promise<number | undefined>;
5
+ export declare function loadProfile(projectConfig: ProjectConfig | null, projectDir: string | null, profileName: string): HsProfileFile | never;
6
+ export declare function enforceProfileUsage(projectConfig: ProjectConfig | null, projectDir: string | null): Promise<void>;
7
+ export declare function loadAndValidateProfile(projectConfig: ProjectConfig | null, projectDir: string | null, profileName: string | undefined, silent?: boolean): Promise<number | undefined>;
8
+ export declare function validateProjectForProfile(projectConfig: ProjectConfig, projectDir: string, profileName: string, derivedAccountId: number, indentSpinners?: boolean): Promise<(string | Error)[]>;
@@ -1,9 +1,12 @@
1
1
  import path from 'path';
2
2
  import { loadHsProfileFile, getHsProfileFilename, getAllHsProfiles, } from '@hubspot/project-parsing-lib';
3
- import { lib } from '../lang/en.js';
3
+ import { commands, lib } from '../lang/en.js';
4
4
  import { uiBetaTag, uiLine } from './ui/index.js';
5
5
  import { uiLogger } from './ui/logger.js';
6
- import { EXIT_CODES } from './enums/exitCodes.js';
6
+ import { validateProfileVariables } from '@hubspot/project-parsing-lib/src/index.js';
7
+ import { getConfigAccountById } from '@hubspot/local-dev-lib/config';
8
+ import SpinniesManager from './ui/SpinniesManager.js';
9
+ import { handleTranslate } from './projects/upload.js';
7
10
  export function logProfileHeader(profileName) {
8
11
  uiLine();
9
12
  uiBetaTag(lib.projectProfiles.logs.usingProfile(getHsProfileFilename(profileName)));
@@ -23,51 +26,94 @@ export function logProfileFooter(profile, includeVariables = false) {
23
26
  }
24
27
  export function loadProfile(projectConfig, projectDir, profileName) {
25
28
  if (!projectConfig || !projectDir) {
26
- uiLogger.error(lib.projectProfiles.loadProfile.errors.noProjectConfig);
27
- return;
29
+ throw new Error(lib.projectProfiles.loadProfile.errors.noProjectConfig);
28
30
  }
29
31
  const projectSourceDir = path.join(projectDir, projectConfig.srcDir);
30
32
  const profileFilename = getHsProfileFilename(profileName);
33
+ let profile;
31
34
  try {
32
- const profile = loadHsProfileFile(projectSourceDir, profileName);
33
- if (!profile) {
34
- uiLogger.error(lib.projectProfiles.loadProfile.errors.profileNotFound(profileFilename));
35
- return;
36
- }
37
- if (!profile.accountId) {
38
- uiLogger.error(lib.projectProfiles.loadProfile.errors.missingAccountId(profileFilename));
39
- return;
40
- }
41
- return profile;
35
+ profile = loadHsProfileFile(projectSourceDir, profileName);
42
36
  }
43
37
  catch (e) {
44
- uiLogger.error(lib.projectProfiles.loadProfile.errors.failedToLoadProfile(profileFilename));
45
- return;
38
+ throw new Error(lib.projectProfiles.loadProfile.errors.failedToLoadProfile(profileFilename));
39
+ }
40
+ if (!profile) {
41
+ throw new Error(lib.projectProfiles.loadProfile.errors.profileNotFound(profileFilename));
42
+ }
43
+ if (!profile.accountId) {
44
+ throw new Error(lib.projectProfiles.loadProfile.errors.missingAccountId(profileFilename));
46
45
  }
46
+ try {
47
+ getConfigAccountById(profile.accountId);
48
+ }
49
+ catch (error) {
50
+ throw new Error(lib.projectProfiles.loadProfile.errors.listedAccountNotFound(profile.accountId, profileFilename));
51
+ }
52
+ return profile;
47
53
  }
48
- export async function exitIfUsingProfiles(projectConfig, projectDir) {
54
+ export async function enforceProfileUsage(projectConfig, projectDir) {
49
55
  if (projectConfig && projectDir) {
50
56
  const existingProfiles = await getAllHsProfiles(path.join(projectDir, projectConfig.srcDir));
51
57
  if (existingProfiles.length > 0) {
52
- uiLogger.error(lib.projectProfiles.exitIfUsingProfiles.errors.noProfileSpecified);
53
- process.exit(EXIT_CODES.ERROR);
58
+ throw new Error(lib.projectProfiles.exitIfUsingProfiles.errors.noProfileSpecified);
54
59
  }
55
60
  }
56
61
  }
57
- export async function loadAndValidateProfile(projectConfig, projectDir, argsProfile) {
58
- if (argsProfile) {
59
- logProfileHeader(argsProfile);
60
- const profile = loadProfile(projectConfig, projectDir, argsProfile);
61
- if (!profile) {
62
- uiLine();
63
- process.exit(EXIT_CODES.ERROR);
64
- }
62
+ export async function loadAndValidateProfile(projectConfig, projectDir, profileName, silent = false) {
63
+ if (!profileName) {
64
+ await enforceProfileUsage(projectConfig, projectDir);
65
+ return;
66
+ }
67
+ if (!silent) {
68
+ logProfileHeader(profileName);
69
+ }
70
+ const profile = loadProfile(projectConfig, projectDir, profileName);
71
+ if (!silent) {
65
72
  logProfileFooter(profile, true);
66
- return profile.accountId;
67
73
  }
68
- else {
69
- // A profile must be specified if this project has profiles configured
70
- await exitIfUsingProfiles(projectConfig, projectDir);
74
+ if (profile.variables) {
75
+ const validationResult = validateProfileVariables(profile.variables, profileName);
76
+ if (!validationResult.success) {
77
+ throw new Error(lib.projectProfiles.loadProfile.errors.profileNotValid(getHsProfileFilename(profileName), validationResult.errors));
78
+ }
79
+ }
80
+ return profile.accountId;
81
+ // A profile must be specified if this project has profiles configured
82
+ }
83
+ export async function validateProjectForProfile(projectConfig, projectDir, profileName, derivedAccountId, indentSpinners = false) {
84
+ let targetAccountId = derivedAccountId;
85
+ const spinnerName = `validatingProfile-${profileName}`;
86
+ const profileFilename = getHsProfileFilename(profileName);
87
+ SpinniesManager.init();
88
+ SpinniesManager.add(spinnerName, {
89
+ text: commands.project.validate.spinners.validatingProfile(profileFilename),
90
+ indent: indentSpinners ? 4 : 0,
91
+ });
92
+ try {
93
+ const accountId = await loadAndValidateProfile(projectConfig, projectDir, profileName, true);
94
+ targetAccountId = accountId || derivedAccountId;
95
+ }
96
+ catch (error) {
97
+ SpinniesManager.fail(spinnerName, {
98
+ text: commands.project.validate.spinners.profileValidationFailed(profileFilename),
99
+ });
100
+ return [error instanceof Error ? error.message : `${error}`];
101
+ }
102
+ try {
103
+ await handleTranslate(projectDir, projectConfig, targetAccountId, false, profileName);
104
+ }
105
+ catch (error) {
106
+ SpinniesManager.fail(spinnerName, {
107
+ text: commands.project.validate.spinners.invalidWithProfile(profileFilename, projectConfig.name),
108
+ });
109
+ const errors = [
110
+ commands.project.validate.failure(projectConfig.name),
111
+ ];
112
+ errors.push(error instanceof Error ? error : `${error}`);
113
+ return errors;
71
114
  }
72
- return undefined;
115
+ SpinniesManager.succeed(spinnerName, {
116
+ text: commands.project.validate.spinners.profileValidationSucceeded(profileFilename),
117
+ });
118
+ return [];
73
119
  }
@@ -12,7 +12,7 @@ vi.mock('@hubspot/ui-extensions-dev-server', () => {
12
12
  });
13
13
  import { fetchAppInstallationData } from '@hubspot/local-dev-lib/api/localDevAuth';
14
14
  import { fetchAppMetadataByUid, fetchPublicAppProductionInstallCounts, installStaticAuthAppOnTestAccount, } from '@hubspot/local-dev-lib/api/appsDev';
15
- import { getAccountConfig } from '@hubspot/local-dev-lib/config';
15
+ import { getConfigAccountById } from '@hubspot/local-dev-lib/config';
16
16
  import AppDevModeInterface from '../localDev/AppDevModeInterface.js';
17
17
  import LocalDevState from '../localDev/LocalDevState.js';
18
18
  import LocalDevLogger from '../localDev/LocalDevLogger.js';
@@ -121,7 +121,7 @@ describe('AppDevModeInterface', () => {
121
121
  previouslyAuthorizedScopeGroups: [],
122
122
  },
123
123
  });
124
- getAccountConfig.mockReturnValue({
124
+ getConfigAccountById.mockReturnValue({
125
125
  parentAccountId: 12345,
126
126
  });
127
127
  isDeveloperTestAccount.mockReturnValue(true);
@@ -290,13 +290,14 @@ describe('AppDevModeInterface', () => {
290
290
  });
291
291
  it('should handle app reinstallation', async () => {
292
292
  // Set up conditions for non-automatic installation
293
- getAccountConfig.mockReturnValue(null);
294
293
  fetchAppInstallationData.mockResolvedValue({
295
294
  data: {
296
295
  isInstalledWithScopeGroups: false,
297
296
  previouslyAuthorizedScopeGroups: ['old-scope'],
298
297
  },
299
298
  });
299
+ // Make it not automatically installable by making it not a test account
300
+ isDeveloperTestAccount.mockReturnValue(false);
300
301
  await appDevModeInterface.setup();
301
302
  expect(installAppBrowserPrompt).toHaveBeenCalledWith('http://static-install-url', true);
302
303
  });
@@ -308,7 +309,7 @@ describe('AppDevModeInterface', () => {
308
309
  });
309
310
  it('should exit if user declines auto-install', async () => {
310
311
  // Set up conditions for automatic installation
311
- getAccountConfig.mockReturnValue({
312
+ getConfigAccountById.mockReturnValue({
312
313
  parentAccountId: 12345, // matches targetProjectAccountId
313
314
  });
314
315
  isDeveloperTestAccount.mockReturnValue(true);
@@ -375,6 +376,15 @@ describe('AppDevModeInterface', () => {
375
376
  data: { uniquePortalInstallCount: 5 },
376
377
  });
377
378
  getStaticAuthAppInstallUrl.mockReturnValue('http://static-install-url');
379
+ getConfigAccountById.mockReturnValue({
380
+ parentAccountId: 12345,
381
+ });
382
+ isDeveloperTestAccount.mockReturnValue(true);
383
+ isSandbox.mockReturnValue(false);
384
+ installAppAutoPrompt.mockResolvedValue(true);
385
+ confirmPrompt.mockResolvedValue(true);
386
+ installStaticAuthAppOnTestAccount.mockResolvedValue(undefined);
387
+ isServerRunningAtUrl.mockResolvedValue(true);
378
388
  installAppBrowserPrompt.mockImplementation(async () => {
379
389
  const addListenerCall = mockLocalDevState.addListener.mock.calls.find(call => call[0] === 'devServerMessage');
380
390
  if (addListenerCall) {
@@ -387,8 +397,8 @@ describe('AppDevModeInterface', () => {
387
397
  mockLocalDevState.getAppDataByUid = vi.fn().mockReturnValue(mockAppData);
388
398
  mockLocalDevState.setAppDataForUid = vi.fn();
389
399
  mockLocalDevState.addListener = vi.fn();
390
- // Target account config is missing
391
- getAccountConfig.mockReturnValue(null);
400
+ // Target account config is missing - make it not a test account so isAutomaticallyInstallable returns false
401
+ isDeveloperTestAccount.mockReturnValue(false);
392
402
  // App is not installed
393
403
  fetchAppInstallationData.mockResolvedValue({
394
404
  data: {
@@ -401,6 +411,7 @@ describe('AppDevModeInterface', () => {
401
411
  localDevState: mockLocalDevState,
402
412
  localDevLogger: mockLocalDevLogger,
403
413
  });
414
+ // Remove the spy to see if that's causing the timeout
404
415
  await newAppDevModeInterface.setup();
405
416
  expect(installAppBrowserPrompt).toHaveBeenCalled();
406
417
  });
@@ -28,6 +28,7 @@ vi.mock('@hubspot/local-dev-lib/config', () => ({
28
28
  getAccountId: vi.fn().mockReturnValue(123),
29
29
  hasLocalStateFlag: vi.fn().mockReturnValue(false),
30
30
  getConfigDefaultAccount: vi.fn().mockReturnValue({ accountId: 123 }),
31
+ globalConfigFileExists: vi.fn().mockReturnValue(true),
31
32
  }));
32
33
  vi.mock('@hubspot/local-dev-lib/urls', () => ({
33
34
  getHubSpotApiOrigin: vi.fn().mockReturnValue('https://api.hubspot.com'),
@@ -26,6 +26,7 @@ vi.mock('../deploy');
26
26
  vi.mock('../config');
27
27
  vi.mock('@hubspot/local-dev-lib/api/projects');
28
28
  vi.mock('@hubspot/local-dev-lib/errors/index');
29
+ vi.mock('@hubspot/local-dev-lib/config');
29
30
  vi.mock('../localDev/LocalDevLogger');
30
31
  vi.mock('../localDev/DevServerManager');
31
32
  // Tests for LocalDevProcess and LocalDevState
@@ -8,6 +8,7 @@ import { pollDeployStatus } from '../pollProjectBuildAndDeploy.js';
8
8
  // Mock external dependencies
9
9
  vi.mock('../../ui/logger.js');
10
10
  vi.mock('@hubspot/local-dev-lib/api/projects');
11
+ vi.mock('@hubspot/local-dev-lib/config');
11
12
  vi.mock('../pollProjectBuildAndDeploy.js');
12
13
  const mockUiLogger = vi.mocked(uiLogger);
13
14
  const mockDeployProject = vi.mocked(deployProject);