@sentry/wizard 6.10.0 → 6.11.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 (144) hide show
  1. package/CHANGELOG.md +20 -0
  2. package/dist/ci-ensure-runtime-loaded.sh +82 -0
  3. package/dist/e2e-tests/tests/angular-17.test.js +72 -82
  4. package/dist/e2e-tests/tests/angular-17.test.js.map +1 -1
  5. package/dist/e2e-tests/tests/angular-19.test.js +71 -80
  6. package/dist/e2e-tests/tests/angular-19.test.js.map +1 -1
  7. package/dist/e2e-tests/tests/cloudflare-worker.test.d.ts +1 -0
  8. package/dist/e2e-tests/tests/cloudflare-worker.test.js +64 -0
  9. package/dist/e2e-tests/tests/cloudflare-worker.test.js.map +1 -0
  10. package/dist/e2e-tests/tests/cloudflare-wrangler-sourcemaps.test.js +2 -5
  11. package/dist/e2e-tests/tests/cloudflare-wrangler-sourcemaps.test.js.map +1 -1
  12. package/dist/e2e-tests/tests/expo.test.js +36 -61
  13. package/dist/e2e-tests/tests/expo.test.js.map +1 -1
  14. package/dist/e2e-tests/tests/flutter.test.js +63 -70
  15. package/dist/e2e-tests/tests/flutter.test.js.map +1 -1
  16. package/dist/e2e-tests/tests/help-message.test.js +2 -2
  17. package/dist/e2e-tests/tests/help-message.test.js.map +1 -1
  18. package/dist/e2e-tests/tests/nextjs-14.test.js +48 -76
  19. package/dist/e2e-tests/tests/nextjs-14.test.js.map +1 -1
  20. package/dist/e2e-tests/tests/nextjs-15.test.js +89 -99
  21. package/dist/e2e-tests/tests/nextjs-15.test.js.map +1 -1
  22. package/dist/e2e-tests/tests/nextjs-16.test.js +48 -45
  23. package/dist/e2e-tests/tests/nextjs-16.test.js.map +1 -1
  24. package/dist/e2e-tests/tests/nuxt-3.test.js +45 -58
  25. package/dist/e2e-tests/tests/nuxt-3.test.js.map +1 -1
  26. package/dist/e2e-tests/tests/nuxt-4.test.js +59 -73
  27. package/dist/e2e-tests/tests/nuxt-4.test.js.map +1 -1
  28. package/dist/e2e-tests/tests/pnpm-workspace.test.js +4 -7
  29. package/dist/e2e-tests/tests/pnpm-workspace.test.js.map +1 -1
  30. package/dist/e2e-tests/tests/react-native.test.js +44 -80
  31. package/dist/e2e-tests/tests/react-native.test.js.map +1 -1
  32. package/dist/e2e-tests/tests/react-router.test.js +163 -145
  33. package/dist/e2e-tests/tests/react-router.test.js.map +1 -1
  34. package/dist/e2e-tests/tests/remix.test.js +162 -132
  35. package/dist/e2e-tests/tests/remix.test.js.map +1 -1
  36. package/dist/e2e-tests/tests/sveltekit-hooks.test.js +48 -36
  37. package/dist/e2e-tests/tests/sveltekit-hooks.test.js.map +1 -1
  38. package/dist/e2e-tests/tests/sveltekit-tracing.test.js +3 -6
  39. package/dist/e2e-tests/tests/sveltekit-tracing.test.js.map +1 -1
  40. package/dist/e2e-tests/utils/index.d.ts +15 -43
  41. package/dist/e2e-tests/utils/index.js +95 -185
  42. package/dist/e2e-tests/utils/index.js.map +1 -1
  43. package/dist/get-e2e-test-matrix.mjs +11 -0
  44. package/dist/lib/Constants.d.ts +1 -0
  45. package/dist/lib/Constants.js +5 -0
  46. package/dist/lib/Constants.js.map +1 -1
  47. package/dist/src/android/android-wizard.js +2 -4
  48. package/dist/src/android/android-wizard.js.map +1 -1
  49. package/dist/src/angular/angular-wizard.js +4 -6
  50. package/dist/src/angular/angular-wizard.js.map +1 -1
  51. package/dist/src/angular/sdk-setup.js +1 -1
  52. package/dist/src/angular/sdk-setup.js.map +1 -1
  53. package/dist/src/apple/apple-wizard.js +2 -4
  54. package/dist/src/apple/apple-wizard.js.map +1 -1
  55. package/dist/src/cloudflare/cloudflare-wizard.d.ts +3 -0
  56. package/dist/src/cloudflare/cloudflare-wizard.js +99 -0
  57. package/dist/src/cloudflare/cloudflare-wizard.js.map +1 -0
  58. package/dist/src/cloudflare/sdk-setup.d.ts +7 -0
  59. package/dist/src/cloudflare/sdk-setup.js +47 -0
  60. package/dist/src/cloudflare/sdk-setup.js.map +1 -0
  61. package/dist/src/cloudflare/templates.d.ts +4 -0
  62. package/dist/src/cloudflare/templates.js +44 -0
  63. package/dist/src/cloudflare/templates.js.map +1 -0
  64. package/dist/src/cloudflare/wrangler/create-wrangler-config.d.ts +4 -0
  65. package/dist/src/cloudflare/wrangler/create-wrangler-config.js +27 -0
  66. package/dist/src/cloudflare/wrangler/create-wrangler-config.js.map +1 -0
  67. package/dist/src/cloudflare/wrangler/ensure-wrangler-config.d.ts +4 -0
  68. package/dist/src/cloudflare/wrangler/ensure-wrangler-config.js +25 -0
  69. package/dist/src/cloudflare/wrangler/ensure-wrangler-config.js.map +1 -0
  70. package/dist/src/cloudflare/wrangler/find-wrangler-config.d.ts +4 -0
  71. package/dist/src/cloudflare/wrangler/find-wrangler-config.js +23 -0
  72. package/dist/src/cloudflare/wrangler/find-wrangler-config.js.map +1 -0
  73. package/dist/src/cloudflare/wrangler/get-entry-point-from-wrangler-config.d.ts +6 -0
  74. package/dist/src/cloudflare/wrangler/get-entry-point-from-wrangler-config.js +52 -0
  75. package/dist/src/cloudflare/wrangler/get-entry-point-from-wrangler-config.js.map +1 -0
  76. package/dist/src/cloudflare/wrangler/update-wrangler-config.d.ts +17 -0
  77. package/dist/src/cloudflare/wrangler/update-wrangler-config.js +173 -0
  78. package/dist/src/cloudflare/wrangler/update-wrangler-config.js.map +1 -0
  79. package/dist/src/cloudflare/wrap-worker.d.ts +32 -0
  80. package/dist/src/cloudflare/wrap-worker.js +109 -0
  81. package/dist/src/cloudflare/wrap-worker.js.map +1 -0
  82. package/dist/src/flutter/flutter-wizard.js +3 -6
  83. package/dist/src/flutter/flutter-wizard.js.map +1 -1
  84. package/dist/src/nextjs/nextjs-wizard.js +0 -2
  85. package/dist/src/nextjs/nextjs-wizard.js.map +1 -1
  86. package/dist/src/nuxt/nuxt-wizard.js +3 -5
  87. package/dist/src/nuxt/nuxt-wizard.js.map +1 -1
  88. package/dist/src/react-native/react-native-wizard.js +2 -4
  89. package/dist/src/react-native/react-native-wizard.js.map +1 -1
  90. package/dist/src/react-router/react-router-wizard.js +3 -5
  91. package/dist/src/react-router/react-router-wizard.js.map +1 -1
  92. package/dist/src/react-router/sdk-setup.d.ts +1 -1
  93. package/dist/src/react-router/sdk-setup.js +3 -4
  94. package/dist/src/react-router/sdk-setup.js.map +1 -1
  95. package/dist/src/remix/remix-wizard.js +2 -4
  96. package/dist/src/remix/remix-wizard.js.map +1 -1
  97. package/dist/src/run.d.ts +1 -1
  98. package/dist/src/run.js +5 -0
  99. package/dist/src/run.js.map +1 -1
  100. package/dist/src/sveltekit/sveltekit-wizard.js +2 -4
  101. package/dist/src/sveltekit/sveltekit-wizard.js.map +1 -1
  102. package/dist/src/utils/abort-if-sportlight-not-supported.d.ts +5 -0
  103. package/dist/src/utils/abort-if-sportlight-not-supported.js +40 -0
  104. package/dist/src/utils/abort-if-sportlight-not-supported.js.map +1 -0
  105. package/dist/src/utils/ast-utils.d.ts +1 -1
  106. package/dist/src/utils/ast-utils.js.map +1 -1
  107. package/dist/src/utils/clack/index.d.ts +2 -2
  108. package/dist/src/utils/clack/index.js.map +1 -1
  109. package/dist/src/utils/clack/mcp-config.js +117 -59
  110. package/dist/src/utils/clack/mcp-config.js.map +1 -1
  111. package/dist/src/version.d.ts +1 -1
  112. package/dist/src/version.js +1 -1
  113. package/dist/src/version.js.map +1 -1
  114. package/dist/test/angular/angular-wizard.test.js +2 -4
  115. package/dist/test/angular/angular-wizard.test.js.map +1 -1
  116. package/dist/test/cloudflare/create-wrangler-config.test.d.ts +1 -0
  117. package/dist/test/cloudflare/create-wrangler-config.test.js +48 -0
  118. package/dist/test/cloudflare/create-wrangler-config.test.js.map +1 -0
  119. package/dist/test/cloudflare/ensure-wrangler-config.test.d.ts +1 -0
  120. package/dist/test/cloudflare/ensure-wrangler-config.test.js +61 -0
  121. package/dist/test/cloudflare/ensure-wrangler-config.test.js.map +1 -0
  122. package/dist/test/cloudflare/find-wrangler-config.test.d.ts +1 -0
  123. package/dist/test/cloudflare/find-wrangler-config.test.js +77 -0
  124. package/dist/test/cloudflare/find-wrangler-config.test.js.map +1 -0
  125. package/dist/test/cloudflare/get-entry-point-from-wrangler-config.test.d.ts +1 -0
  126. package/dist/test/cloudflare/get-entry-point-from-wrangler-config.test.js +81 -0
  127. package/dist/test/cloudflare/get-entry-point-from-wrangler-config.test.js.map +1 -0
  128. package/dist/test/cloudflare/sdk-setup.test.d.ts +1 -0
  129. package/dist/test/cloudflare/sdk-setup.test.js +152 -0
  130. package/dist/test/cloudflare/sdk-setup.test.js.map +1 -0
  131. package/dist/test/cloudflare/templates.test.d.ts +1 -0
  132. package/dist/test/cloudflare/templates.test.js +68 -0
  133. package/dist/test/cloudflare/templates.test.js.map +1 -0
  134. package/dist/test/cloudflare/update-wrangler-config.test.d.ts +1 -0
  135. package/dist/test/cloudflare/update-wrangler-config.test.js +216 -0
  136. package/dist/test/cloudflare/update-wrangler-config.test.js.map +1 -0
  137. package/dist/test/cloudflare/wrap-worker.test.d.ts +1 -0
  138. package/dist/test/cloudflare/wrap-worker.test.js +143 -0
  139. package/dist/test/cloudflare/wrap-worker.test.js.map +1 -0
  140. package/dist/test/react-router/sdk-setup.test.js +2 -2
  141. package/dist/test/react-router/sdk-setup.test.js.map +1 -1
  142. package/dist/test/utils/clack/mcp-config.test.js +176 -51
  143. package/dist/test/utils/clack/mcp-config.test.js.map +1 -1
  144. package/package.json +5 -4
@@ -0,0 +1,99 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.buildOutroMessage = exports.runCloudflareWizard = void 0;
7
+ // @ts-expect-error - clack is ESM and TS complains about that. It works though
8
+ const prompts_1 = __importDefault(require("@clack/prompts"));
9
+ const chalk_1 = __importDefault(require("chalk"));
10
+ const telemetry_1 = require("../telemetry");
11
+ const clack_1 = require("../utils/clack");
12
+ const mcp_config_1 = require("../utils/clack/mcp-config");
13
+ const package_json_1 = require("../utils/package-json");
14
+ const sdk_setup_1 = require("./sdk-setup");
15
+ const abort_if_sportlight_not_supported_1 = require("../utils/abort-if-sportlight-not-supported");
16
+ const ensure_wrangler_config_1 = require("./wrangler/ensure-wrangler-config");
17
+ const update_wrangler_config_1 = require("./wrangler/update-wrangler-config");
18
+ const debug_1 = require("../utils/debug");
19
+ const get_entry_point_from_wrangler_config_1 = require("./wrangler/get-entry-point-from-wrangler-config");
20
+ async function runCloudflareWizard(options) {
21
+ return (0, telemetry_1.withTelemetry)({
22
+ enabled: options.telemetryEnabled,
23
+ integration: 'cloudflare',
24
+ wizardOptions: options,
25
+ }, () => runCloudflareWizardWithTelemetry(options));
26
+ }
27
+ exports.runCloudflareWizard = runCloudflareWizard;
28
+ async function runCloudflareWizardWithTelemetry(options) {
29
+ const { promoCode, telemetryEnabled, forceInstall } = options;
30
+ (0, clack_1.printWelcome)({
31
+ wizardName: 'Sentry Cloudflare Wizard',
32
+ promoCode,
33
+ telemetryEnabled,
34
+ });
35
+ await (0, clack_1.confirmContinueIfNoOrDirtyGitRepo)({
36
+ ignoreGitChanges: options.ignoreGitChanges,
37
+ cwd: undefined,
38
+ });
39
+ const packageJson = await (0, clack_1.getPackageDotJson)();
40
+ await (0, clack_1.ensurePackageIsInstalled)(packageJson, 'wrangler', 'Cloudflare');
41
+ (0, telemetry_1.traceStep)('Ensure Wrangler config', () => {
42
+ (0, ensure_wrangler_config_1.ensureWranglerConfig)();
43
+ });
44
+ const projectData = await (0, clack_1.getOrAskForProjectData)(options, 'node-cloudflare-workers');
45
+ if (projectData.spotlight) {
46
+ return (0, abort_if_sportlight_not_supported_1.abortIfSpotlightNotSupported)('Cloudflare');
47
+ }
48
+ const { selectedProject } = projectData;
49
+ await (0, clack_1.installPackage)({
50
+ packageName: '@sentry/cloudflare@^10',
51
+ packageNameDisplayLabel: '@sentry/cloudflare',
52
+ alreadyInstalled: (0, package_json_1.hasPackageInstalled)('@sentry/cloudflare', packageJson),
53
+ forceInstall,
54
+ });
55
+ const selectedFeatures = await (0, clack_1.featureSelectionPrompt)([
56
+ {
57
+ id: 'performance',
58
+ prompt: `Do you want to enable ${chalk_1.default.bold('Tracing')} to track the performance of your application?`,
59
+ enabledHint: 'recommended',
60
+ },
61
+ ]);
62
+ await (0, telemetry_1.traceStep)('Create Sentry initialization', async () => {
63
+ try {
64
+ await (0, sdk_setup_1.createSentryInitFile)(selectedProject.keys[0].dsn.public, selectedFeatures);
65
+ }
66
+ catch (e) {
67
+ prompts_1.default.log.warn('Could not automatically set up Sentry initialization. Please set it up manually using instructions from https://docs.sentry.io/platforms/javascript/guides/cloudflare/');
68
+ (0, debug_1.debug)(e);
69
+ }
70
+ });
71
+ const mainFile = (0, get_entry_point_from_wrangler_config_1.getEntryPointFromWranglerConfig)();
72
+ await (0, telemetry_1.traceStep)('Update Wrangler config with Sentry requirements', () => (0, update_wrangler_config_1.updateWranglerConfig)({
73
+ ...(mainFile ? {} : { main: get_entry_point_from_wrangler_config_1.defaultEntryPoint }),
74
+ compatibility_flags: ['nodejs_als'],
75
+ compatibility_date: new Date().toISOString().slice(0, 10),
76
+ version_metadata: {
77
+ binding: 'CF_VERSION_METADATA',
78
+ },
79
+ }));
80
+ await (0, clack_1.runPrettierIfInstalled)({ cwd: undefined });
81
+ // Offer optional project-scoped MCP config for Sentry with org and project scope
82
+ await (0, mcp_config_1.offerProjectScopedMcpConfig)(selectedProject.organization.slug, selectedProject.slug);
83
+ prompts_1.default.outro(buildOutroMessage());
84
+ }
85
+ function buildOutroMessage() {
86
+ return `
87
+ ${chalk_1.default.green('Sentry has been successfully configured for your Cloudflare project.')}
88
+
89
+ ${chalk_1.default.dim('Next steps:')}
90
+ ${chalk_1.default.dim('1. Wrap your worker with Sentry as instructed above')}
91
+ ${chalk_1.default.dim('2. Deploy your application with:')} ${chalk_1.default.cyan('wrangler deploy')}
92
+ ${chalk_1.default.dim('3. Trigger an error to test Sentry integration')}
93
+
94
+ ${chalk_1.default.cyan(`To learn more about using Sentry with Cloudflare, visit:
95
+ https://docs.sentry.io/platforms/javascript/guides/cloudflare/`)}
96
+ `;
97
+ }
98
+ exports.buildOutroMessage = buildOutroMessage;
99
+ //# sourceMappingURL=cloudflare-wizard.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cloudflare-wizard.js","sourceRoot":"","sources":["../../../src/cloudflare/cloudflare-wizard.ts"],"names":[],"mappings":"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","sourcesContent":["// @ts-expect-error - clack is ESM and TS complains about that. It works though\nimport clack from '@clack/prompts';\nimport chalk from 'chalk';\n\nimport { traceStep, withTelemetry } from '../telemetry';\nimport {\n confirmContinueIfNoOrDirtyGitRepo,\n ensurePackageIsInstalled,\n featureSelectionPrompt,\n getOrAskForProjectData,\n getPackageDotJson,\n installPackage,\n printWelcome,\n runPrettierIfInstalled,\n} from '../utils/clack';\nimport { offerProjectScopedMcpConfig } from '../utils/clack/mcp-config';\nimport { hasPackageInstalled } from '../utils/package-json';\nimport type { WizardOptions } from '../utils/types';\nimport { createSentryInitFile } from './sdk-setup';\nimport { abortIfSpotlightNotSupported } from '../utils/abort-if-sportlight-not-supported';\nimport { ensureWranglerConfig } from './wrangler/ensure-wrangler-config';\nimport { updateWranglerConfig } from './wrangler/update-wrangler-config';\nimport { debug } from '../utils/debug';\nimport {\n defaultEntryPoint,\n getEntryPointFromWranglerConfig,\n} from './wrangler/get-entry-point-from-wrangler-config';\n\nexport async function runCloudflareWizard(\n options: WizardOptions,\n): Promise<void> {\n return withTelemetry(\n {\n enabled: options.telemetryEnabled,\n integration: 'cloudflare',\n wizardOptions: options,\n },\n () => runCloudflareWizardWithTelemetry(options),\n );\n}\n\nasync function runCloudflareWizardWithTelemetry(\n options: WizardOptions,\n): Promise<void> {\n const { promoCode, telemetryEnabled, forceInstall } = options;\n\n printWelcome({\n wizardName: 'Sentry Cloudflare Wizard',\n promoCode,\n telemetryEnabled,\n });\n\n await confirmContinueIfNoOrDirtyGitRepo({\n ignoreGitChanges: options.ignoreGitChanges,\n cwd: undefined,\n });\n\n const packageJson = await getPackageDotJson();\n\n await ensurePackageIsInstalled(packageJson, 'wrangler', 'Cloudflare');\n\n traceStep('Ensure Wrangler config', () => {\n ensureWranglerConfig();\n });\n\n const projectData = await getOrAskForProjectData(\n options,\n 'node-cloudflare-workers',\n );\n\n if (projectData.spotlight) {\n return abortIfSpotlightNotSupported('Cloudflare');\n }\n\n const { selectedProject } = projectData;\n\n await installPackage({\n packageName: '@sentry/cloudflare@^10',\n packageNameDisplayLabel: '@sentry/cloudflare',\n alreadyInstalled: hasPackageInstalled('@sentry/cloudflare', packageJson),\n forceInstall,\n });\n\n const selectedFeatures = await featureSelectionPrompt([\n {\n id: 'performance',\n prompt: `Do you want to enable ${chalk.bold(\n 'Tracing',\n )} to track the performance of your application?`,\n enabledHint: 'recommended',\n },\n ] as const);\n\n await traceStep('Create Sentry initialization', async () => {\n try {\n await createSentryInitFile(\n selectedProject.keys[0].dsn.public,\n selectedFeatures,\n );\n } catch (e) {\n clack.log.warn(\n 'Could not automatically set up Sentry initialization. Please set it up manually using instructions from https://docs.sentry.io/platforms/javascript/guides/cloudflare/',\n );\n debug(e);\n }\n });\n\n const mainFile = getEntryPointFromWranglerConfig();\n\n await traceStep('Update Wrangler config with Sentry requirements', () =>\n updateWranglerConfig({\n ...(mainFile ? {} : { main: defaultEntryPoint }),\n compatibility_flags: ['nodejs_als'],\n compatibility_date: new Date().toISOString().slice(0, 10),\n version_metadata: {\n binding: 'CF_VERSION_METADATA',\n },\n }),\n );\n\n await runPrettierIfInstalled({ cwd: undefined });\n\n // Offer optional project-scoped MCP config for Sentry with org and project scope\n await offerProjectScopedMcpConfig(\n selectedProject.organization.slug,\n selectedProject.slug,\n );\n\n clack.outro(buildOutroMessage());\n}\n\nexport function buildOutroMessage(): string {\n return `\n ${chalk.green(\n 'Sentry has been successfully configured for your Cloudflare project.',\n )}\n\n ${chalk.dim('Next steps:')}\n ${chalk.dim('1. Wrap your worker with Sentry as instructed above')}\n ${chalk.dim('2. Deploy your application with:')} ${chalk.cyan(\n 'wrangler deploy',\n )}\n ${chalk.dim('3. Trigger an error to test Sentry integration')}\n\n ${chalk.cyan(\n `To learn more about using Sentry with Cloudflare, visit:\n https://docs.sentry.io/platforms/javascript/guides/cloudflare/`,\n )}\n `;\n}\n"]}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * Creates or updates the main worker file with Sentry initialization.
3
+ * Currently focused on Cloudflare Workers
4
+ */
5
+ export declare function createSentryInitFile(dsn: string, selectedFeatures: {
6
+ performance: boolean;
7
+ }): Promise<void>;
@@ -0,0 +1,47 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.createSentryInitFile = void 0;
7
+ // @ts-expect-error - clack is ESM and TS complains about that. It works though
8
+ const prompts_1 = __importDefault(require("@clack/prompts"));
9
+ const chalk_1 = __importDefault(require("chalk"));
10
+ const node_fs_1 = __importDefault(require("node:fs"));
11
+ const node_path_1 = __importDefault(require("node:path"));
12
+ const templates_1 = require("./templates");
13
+ const get_entry_point_from_wrangler_config_1 = require("./wrangler/get-entry-point-from-wrangler-config");
14
+ const wrap_worker_1 = require("./wrap-worker");
15
+ /**
16
+ * Creates or updates the main worker file with Sentry initialization.
17
+ * Currently focused on Cloudflare Workers
18
+ */
19
+ async function createSentryInitFile(dsn, selectedFeatures) {
20
+ const entryPointFromConfig = (0, get_entry_point_from_wrangler_config_1.getEntryPointFromWranglerConfig)();
21
+ if (!entryPointFromConfig) {
22
+ prompts_1.default.log.info('No entry point found in wrangler config, creating a new one.');
23
+ const cloudflareWorkerTemplate = (0, templates_1.getCloudflareWorkerTemplateWithHandler)();
24
+ await node_fs_1.default.promises.mkdir(node_path_1.default.join(process.cwd(), node_path_1.default.dirname(get_entry_point_from_wrangler_config_1.defaultEntryPoint)), {
25
+ recursive: true,
26
+ });
27
+ await node_fs_1.default.promises.writeFile(node_path_1.default.join(process.cwd(), get_entry_point_from_wrangler_config_1.defaultEntryPoint), cloudflareWorkerTemplate, { encoding: 'utf-8', flag: 'w' });
28
+ prompts_1.default.log.success(`Created ${chalk_1.default.cyan(get_entry_point_from_wrangler_config_1.defaultEntryPoint)}.`);
29
+ return;
30
+ }
31
+ const entryPointPath = node_path_1.default.join(process.cwd(), entryPointFromConfig);
32
+ if (node_fs_1.default.existsSync(entryPointPath)) {
33
+ prompts_1.default.log.info(`Found existing entry point: ${chalk_1.default.cyan(entryPointFromConfig)}`);
34
+ try {
35
+ await (0, wrap_worker_1.wrapWorkerWithSentry)(entryPointPath, dsn, selectedFeatures);
36
+ prompts_1.default.log.success(`Wrapped ${chalk_1.default.cyan(entryPointFromConfig)} with Sentry initialization.`);
37
+ }
38
+ catch (error) {
39
+ prompts_1.default.log.warn('Failed to wrap worker automatically.');
40
+ prompts_1.default.log.step('Please wrap your handler with Sentry initialization:');
41
+ prompts_1.default.note(chalk_1.default.cyan((0, templates_1.getCloudflareWorkerTemplate)(dsn, selectedFeatures)));
42
+ }
43
+ return;
44
+ }
45
+ }
46
+ exports.createSentryInitFile = createSentryInitFile;
47
+ //# sourceMappingURL=sdk-setup.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sdk-setup.js","sourceRoot":"","sources":["../../../src/cloudflare/sdk-setup.ts"],"names":[],"mappings":";;;;;;AAAA,+EAA+E;AAC/E,6DAAmC;AACnC,kDAA0B;AAC1B,sDAAyB;AACzB,0DAA6B;AAC7B,2CAGqB;AACrB,0GAGyD;AACzD,+CAAqD;AAErD;;;GAGG;AACI,KAAK,UAAU,oBAAoB,CACxC,GAAW,EACX,gBAEC;IAED,MAAM,oBAAoB,GAAG,IAAA,sEAA+B,GAAE,CAAC;IAE/D,IAAI,CAAC,oBAAoB,EAAE;QACzB,iBAAK,CAAC,GAAG,CAAC,IAAI,CACZ,8DAA8D,CAC/D,CAAC;QAEF,MAAM,wBAAwB,GAAG,IAAA,kDAAsC,GAAE,CAAC;QAE1E,MAAM,iBAAE,CAAC,QAAQ,CAAC,KAAK,CACrB,mBAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,mBAAI,CAAC,OAAO,CAAC,wDAAiB,CAAC,CAAC,EACzD;YACE,SAAS,EAAE,IAAI;SAChB,CACF,CAAC;QACF,MAAM,iBAAE,CAAC,QAAQ,CAAC,SAAS,CACzB,mBAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,wDAAiB,CAAC,EAC3C,wBAAwB,EACxB,EAAE,QAAQ,EAAE,OAAO,EAAE,IAAI,EAAE,GAAG,EAAE,CACjC,CAAC;QAEF,iBAAK,CAAC,GAAG,CAAC,OAAO,CAAC,WAAW,eAAK,CAAC,IAAI,CAAC,wDAAiB,CAAC,GAAG,CAAC,CAAC;QAE/D,OAAO;KACR;IAED,MAAM,cAAc,GAAG,mBAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,oBAAoB,CAAC,CAAC;IAEtE,IAAI,iBAAE,CAAC,UAAU,CAAC,cAAc,CAAC,EAAE;QACjC,iBAAK,CAAC,GAAG,CAAC,IAAI,CACZ,+BAA+B,eAAK,CAAC,IAAI,CAAC,oBAAoB,CAAC,EAAE,CAClE,CAAC;QAEF,IAAI;YACF,MAAM,IAAA,kCAAoB,EAAC,cAAc,EAAE,GAAG,EAAE,gBAAgB,CAAC,CAAC;YAClE,iBAAK,CAAC,GAAG,CAAC,OAAO,CACf,WAAW,eAAK,CAAC,IAAI,CACnB,oBAAoB,CACrB,8BAA8B,CAChC,CAAC;SACH;QAAC,OAAO,KAAK,EAAE;YACd,iBAAK,CAAC,GAAG,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;YACvD,iBAAK,CAAC,GAAG,CAAC,IAAI,CAAC,sDAAsD,CAAC,CAAC;YAEvE,iBAAK,CAAC,IAAI,CACR,eAAK,CAAC,IAAI,CAAC,IAAA,uCAA2B,EAAC,GAAG,EAAE,gBAAgB,CAAC,CAAC,CAC/D,CAAC;SACH;QACD,OAAO;KACR;AACH,CAAC;AAxDD,oDAwDC","sourcesContent":["// @ts-expect-error - clack is ESM and TS complains about that. It works though\nimport clack from '@clack/prompts';\nimport chalk from 'chalk';\nimport fs from 'node:fs';\nimport path from 'node:path';\nimport {\n getCloudflareWorkerTemplate,\n getCloudflareWorkerTemplateWithHandler,\n} from './templates';\nimport {\n defaultEntryPoint,\n getEntryPointFromWranglerConfig,\n} from './wrangler/get-entry-point-from-wrangler-config';\nimport { wrapWorkerWithSentry } from './wrap-worker';\n\n/**\n * Creates or updates the main worker file with Sentry initialization.\n * Currently focused on Cloudflare Workers\n */\nexport async function createSentryInitFile(\n dsn: string,\n selectedFeatures: {\n performance: boolean;\n },\n): Promise<void> {\n const entryPointFromConfig = getEntryPointFromWranglerConfig();\n\n if (!entryPointFromConfig) {\n clack.log.info(\n 'No entry point found in wrangler config, creating a new one.',\n );\n\n const cloudflareWorkerTemplate = getCloudflareWorkerTemplateWithHandler();\n\n await fs.promises.mkdir(\n path.join(process.cwd(), path.dirname(defaultEntryPoint)),\n {\n recursive: true,\n },\n );\n await fs.promises.writeFile(\n path.join(process.cwd(), defaultEntryPoint),\n cloudflareWorkerTemplate,\n { encoding: 'utf-8', flag: 'w' },\n );\n\n clack.log.success(`Created ${chalk.cyan(defaultEntryPoint)}.`);\n\n return;\n }\n\n const entryPointPath = path.join(process.cwd(), entryPointFromConfig);\n\n if (fs.existsSync(entryPointPath)) {\n clack.log.info(\n `Found existing entry point: ${chalk.cyan(entryPointFromConfig)}`,\n );\n\n try {\n await wrapWorkerWithSentry(entryPointPath, dsn, selectedFeatures);\n clack.log.success(\n `Wrapped ${chalk.cyan(\n entryPointFromConfig,\n )} with Sentry initialization.`,\n );\n } catch (error) {\n clack.log.warn('Failed to wrap worker automatically.');\n clack.log.step('Please wrap your handler with Sentry initialization:');\n\n clack.note(\n chalk.cyan(getCloudflareWorkerTemplate(dsn, selectedFeatures)),\n );\n }\n return;\n }\n}\n"]}
@@ -0,0 +1,4 @@
1
+ export declare function getCloudflareWorkerTemplate(dsn: string, selectedFeatures: {
2
+ performance: boolean;
3
+ }): string;
4
+ export declare function getCloudflareWorkerTemplateWithHandler(): string;
@@ -0,0 +1,44 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.getCloudflareWorkerTemplateWithHandler = exports.getCloudflareWorkerTemplate = void 0;
4
+ function getCloudflareWorkerTemplate(dsn, selectedFeatures) {
5
+ let performanceOptions = '';
6
+ if (selectedFeatures.performance) {
7
+ performanceOptions = `
8
+ // Define how likely traces are sampled. Adjust this value in production, or use tracesSampler for greater control.
9
+ tracesSampleRate: 1,`;
10
+ }
11
+ return `import * as Sentry from '@sentry/cloudflare';
12
+
13
+ export default Sentry.withSentry(
14
+ (env) => ({
15
+ dsn: '${dsn}',${performanceOptions}
16
+ }),
17
+ {
18
+ async fetch(request, env, ctx): Promise<Response> {
19
+ // Your worker logic here
20
+ return new Response('Hello World!');
21
+ },
22
+ } satisfies ExportedHandler<Env>,
23
+ );
24
+ `;
25
+ }
26
+ exports.getCloudflareWorkerTemplate = getCloudflareWorkerTemplate;
27
+ function getCloudflareWorkerTemplateWithHandler() {
28
+ return `export default {
29
+ async fetch(request, env, ctx): Promise<Response> {
30
+ const url = new URL(request.url);
31
+ switch (url.pathname) {
32
+ case '/message':
33
+ return new Response('Hello, World!');
34
+ case '/random':
35
+ return new Response(crypto.randomUUID());
36
+ default:
37
+ return new Response('Not Found', { status: 404 });
38
+ }
39
+ },
40
+ } satisfies ExportedHandler<Env>;
41
+ `;
42
+ }
43
+ exports.getCloudflareWorkerTemplateWithHandler = getCloudflareWorkerTemplateWithHandler;
44
+ //# sourceMappingURL=templates.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"templates.js","sourceRoot":"","sources":["../../../src/cloudflare/templates.ts"],"names":[],"mappings":";;;AAAA,SAAgB,2BAA2B,CACzC,GAAW,EACX,gBAEC;IAED,IAAI,kBAAkB,GAAG,EAAE,CAAC;IAC5B,IAAI,gBAAgB,CAAC,WAAW,EAAE;QAChC,kBAAkB,GAAG;;uBAEF,CAAC;KACrB;IAED,OAAO;;;;UAIC,GAAG,KAAK,kBAAkB;;;;;;;;;CASnC,CAAC;AACF,CAAC;AA3BD,kEA2BC;AAED,SAAgB,sCAAsC;IACpD,OAAO;;;;;;;;;;;;;CAaR,CAAC;AACF,CAAC;AAfD,wFAeC","sourcesContent":["export function getCloudflareWorkerTemplate(\n dsn: string,\n selectedFeatures: {\n performance: boolean;\n },\n): string {\n let performanceOptions = '';\n if (selectedFeatures.performance) {\n performanceOptions = `\n\t\t// Define how likely traces are sampled. Adjust this value in production, or use tracesSampler for greater control.\n\t\ttracesSampleRate: 1,`;\n }\n\n return `import * as Sentry from '@sentry/cloudflare';\n\nexport default Sentry.withSentry(\n\t(env) => ({\n\t\tdsn: '${dsn}',${performanceOptions}\n\t}),\n\t{\n\t\tasync fetch(request, env, ctx): Promise<Response> {\n\t\t\t// Your worker logic here\n\t\t\treturn new Response('Hello World!');\n\t\t},\n\t} satisfies ExportedHandler<Env>,\n);\n`;\n}\n\nexport function getCloudflareWorkerTemplateWithHandler(): string {\n return `export default {\n\tasync fetch(request, env, ctx): Promise<Response> {\n\t\tconst url = new URL(request.url);\n\t\tswitch (url.pathname) {\n\t\t\tcase '/message':\n\t\t\t\treturn new Response('Hello, World!');\n\t\t\tcase '/random':\n\t\t\t\treturn new Response(crypto.randomUUID());\n\t\t\tdefault:\n\t\t\t\treturn new Response('Not Found', { status: 404 });\n\t\t}\n\t},\n} satisfies ExportedHandler<Env>;\n`;\n}\n"]}
@@ -0,0 +1,4 @@
1
+ /**
2
+ * Creates a basic wrangler.jsonc config file for a Cloudflare Worker
3
+ */
4
+ export declare function createWranglerConfig(): void;
@@ -0,0 +1,27 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.createWranglerConfig = void 0;
7
+ // @ts-expect-error - clack is ESM and TS complains about that. It works though
8
+ const prompts_1 = __importDefault(require("@clack/prompts"));
9
+ const chalk_1 = __importDefault(require("chalk"));
10
+ const node_fs_1 = __importDefault(require("node:fs"));
11
+ const node_path_1 = __importDefault(require("node:path"));
12
+ /**
13
+ * Creates a basic wrangler.jsonc config file for a Cloudflare Worker
14
+ */
15
+ function createWranglerConfig() {
16
+ const configPath = node_path_1.default.join(process.cwd(), 'wrangler.jsonc');
17
+ const config = {
18
+ $schema: 'node_modules/wrangler/config-schema.json',
19
+ name: 'my-worker',
20
+ main: 'src/index.ts',
21
+ };
22
+ node_fs_1.default.writeFileSync(configPath, JSON.stringify(config, null, 2), 'utf-8');
23
+ prompts_1.default.log.success(`Created ${chalk_1.default.cyan('wrangler.jsonc')} configuration file.`);
24
+ prompts_1.default.log.info(`Please update the ${chalk_1.default.cyan('name')} and ${chalk_1.default.cyan('main')} fields in ${chalk_1.default.cyan('wrangler.jsonc')} to match your worker name and entry point.`);
25
+ }
26
+ exports.createWranglerConfig = createWranglerConfig;
27
+ //# sourceMappingURL=create-wrangler-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create-wrangler-config.js","sourceRoot":"","sources":["../../../../src/cloudflare/wrangler/create-wrangler-config.ts"],"names":[],"mappings":";;;;;;AAAA,+EAA+E;AAC/E,6DAAmC;AACnC,kDAA0B;AAC1B,sDAAyB;AACzB,0DAA6B;AAE7B;;GAEG;AACH,SAAgB,oBAAoB;IAClC,MAAM,UAAU,GAAG,mBAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,gBAAgB,CAAC,CAAC;IAE9D,MAAM,MAAM,GAAG;QACb,OAAO,EAAE,0CAA0C;QACnD,IAAI,EAAE,WAAW;QACjB,IAAI,EAAE,cAAc;KACrB,CAAC;IAEF,iBAAE,CAAC,aAAa,CAAC,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;IAEvE,iBAAK,CAAC,GAAG,CAAC,OAAO,CACf,WAAW,eAAK,CAAC,IAAI,CAAC,gBAAgB,CAAC,sBAAsB,CAC9D,CAAC;IACF,iBAAK,CAAC,GAAG,CAAC,IAAI,CACZ,qBAAqB,eAAK,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,eAAK,CAAC,IAAI,CACvD,MAAM,CACP,cAAc,eAAK,CAAC,IAAI,CACvB,gBAAgB,CACjB,6CAA6C,CAC/C,CAAC;AACJ,CAAC;AArBD,oDAqBC","sourcesContent":["// @ts-expect-error - clack is ESM and TS complains about that. It works though\nimport clack from '@clack/prompts';\nimport chalk from 'chalk';\nimport fs from 'node:fs';\nimport path from 'node:path';\n\n/**\n * Creates a basic wrangler.jsonc config file for a Cloudflare Worker\n */\nexport function createWranglerConfig(): void {\n const configPath = path.join(process.cwd(), 'wrangler.jsonc');\n\n const config = {\n $schema: 'node_modules/wrangler/config-schema.json',\n name: 'my-worker',\n main: 'src/index.ts',\n };\n\n fs.writeFileSync(configPath, JSON.stringify(config, null, 2), 'utf-8');\n\n clack.log.success(\n `Created ${chalk.cyan('wrangler.jsonc')} configuration file.`,\n );\n clack.log.info(\n `Please update the ${chalk.cyan('name')} and ${chalk.cyan(\n 'main',\n )} fields in ${chalk.cyan(\n 'wrangler.jsonc',\n )} to match your worker name and entry point.`,\n );\n}\n"]}
@@ -0,0 +1,4 @@
1
+ /**
2
+ * Ensures a wrangler config exists, creating one if necessary
3
+ */
4
+ export declare function ensureWranglerConfig(): void;
@@ -0,0 +1,25 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.ensureWranglerConfig = void 0;
7
+ // @ts-expect-error - clack is ESM and TS complains about that. It works though
8
+ const prompts_1 = __importDefault(require("@clack/prompts"));
9
+ const chalk_1 = __importDefault(require("chalk"));
10
+ const find_wrangler_config_1 = require("./find-wrangler-config");
11
+ const create_wrangler_config_1 = require("./create-wrangler-config");
12
+ /**
13
+ * Ensures a wrangler config exists, creating one if necessary
14
+ */
15
+ function ensureWranglerConfig() {
16
+ const existingConfig = (0, find_wrangler_config_1.findWranglerConfig)();
17
+ if (existingConfig) {
18
+ prompts_1.default.log.info(`Found existing Wrangler config: ${chalk_1.default.cyan(existingConfig)}`);
19
+ return;
20
+ }
21
+ prompts_1.default.log.step('No Wrangler configuration file found.');
22
+ (0, create_wrangler_config_1.createWranglerConfig)();
23
+ }
24
+ exports.ensureWranglerConfig = ensureWranglerConfig;
25
+ //# sourceMappingURL=ensure-wrangler-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ensure-wrangler-config.js","sourceRoot":"","sources":["../../../../src/cloudflare/wrangler/ensure-wrangler-config.ts"],"names":[],"mappings":";;;;;;AAAA,+EAA+E;AAC/E,6DAAmC;AACnC,kDAA0B;AAC1B,iEAA4D;AAC5D,qEAAgE;AAEhE;;GAEG;AACH,SAAgB,oBAAoB;IAClC,MAAM,cAAc,GAAG,IAAA,yCAAkB,GAAE,CAAC;IAE5C,IAAI,cAAc,EAAE;QAClB,iBAAK,CAAC,GAAG,CAAC,IAAI,CACZ,mCAAmC,eAAK,CAAC,IAAI,CAAC,cAAc,CAAC,EAAE,CAChE,CAAC;QACF,OAAO;KACR;IAED,iBAAK,CAAC,GAAG,CAAC,IAAI,CAAC,uCAAuC,CAAC,CAAC;IACxD,IAAA,6CAAoB,GAAE,CAAC;AACzB,CAAC;AAZD,oDAYC","sourcesContent":["// @ts-expect-error - clack is ESM and TS complains about that. It works though\nimport clack from '@clack/prompts';\nimport chalk from 'chalk';\nimport { findWranglerConfig } from './find-wrangler-config';\nimport { createWranglerConfig } from './create-wrangler-config';\n\n/**\n * Ensures a wrangler config exists, creating one if necessary\n */\nexport function ensureWranglerConfig(): void {\n const existingConfig = findWranglerConfig();\n\n if (existingConfig) {\n clack.log.info(\n `Found existing Wrangler config: ${chalk.cyan(existingConfig)}`,\n );\n return;\n }\n\n clack.log.step('No Wrangler configuration file found.');\n createWranglerConfig();\n}\n"]}
@@ -0,0 +1,4 @@
1
+ /**
2
+ * Checks if a wrangler config file exists in the project
3
+ */
4
+ export declare function findWranglerConfig(): string | undefined;
@@ -0,0 +1,23 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.findWranglerConfig = void 0;
7
+ const node_fs_1 = __importDefault(require("node:fs"));
8
+ const node_path_1 = __importDefault(require("node:path"));
9
+ /**
10
+ * Checks if a wrangler config file exists in the project
11
+ */
12
+ function findWranglerConfig() {
13
+ const possibleConfigs = ['wrangler.jsonc', 'wrangler.json', 'wrangler.toml'];
14
+ for (const configFile of possibleConfigs) {
15
+ const configPath = node_path_1.default.join(process.cwd(), configFile);
16
+ if (node_fs_1.default.existsSync(configPath)) {
17
+ return configFile;
18
+ }
19
+ }
20
+ return undefined;
21
+ }
22
+ exports.findWranglerConfig = findWranglerConfig;
23
+ //# sourceMappingURL=find-wrangler-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"find-wrangler-config.js","sourceRoot":"","sources":["../../../../src/cloudflare/wrangler/find-wrangler-config.ts"],"names":[],"mappings":";;;;;;AAAA,sDAAyB;AACzB,0DAA6B;AAE7B;;GAEG;AACH,SAAgB,kBAAkB;IAChC,MAAM,eAAe,GAAG,CAAC,gBAAgB,EAAE,eAAe,EAAE,eAAe,CAAC,CAAC;IAE7E,KAAK,MAAM,UAAU,IAAI,eAAe,EAAE;QACxC,MAAM,UAAU,GAAG,mBAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,UAAU,CAAC,CAAC;QACxD,IAAI,iBAAE,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE;YAC7B,OAAO,UAAU,CAAC;SACnB;KACF;IAED,OAAO,SAAS,CAAC;AACnB,CAAC;AAXD,gDAWC","sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\n\n/**\n * Checks if a wrangler config file exists in the project\n */\nexport function findWranglerConfig(): string | undefined {\n const possibleConfigs = ['wrangler.jsonc', 'wrangler.json', 'wrangler.toml'];\n\n for (const configFile of possibleConfigs) {\n const configPath = path.join(process.cwd(), configFile);\n if (fs.existsSync(configPath)) {\n return configFile;\n }\n }\n\n return undefined;\n}\n"]}
@@ -0,0 +1,6 @@
1
+ export declare const defaultEntryPoint = "src/index.ts";
2
+ /**
3
+ * Reads the main entry point from the wrangler config file
4
+ * Returns undefined if no config exists or if main field is not specified
5
+ */
6
+ export declare function getEntryPointFromWranglerConfig(): string | undefined;
@@ -0,0 +1,52 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.getEntryPointFromWranglerConfig = exports.defaultEntryPoint = void 0;
7
+ const node_fs_1 = __importDefault(require("node:fs"));
8
+ const node_path_1 = __importDefault(require("node:path"));
9
+ const find_wrangler_config_1 = require("./find-wrangler-config");
10
+ const ast_utils_1 = require("../../utils/ast-utils");
11
+ exports.defaultEntryPoint = 'src/index.ts';
12
+ /**
13
+ * Reads the main entry point from the wrangler config file
14
+ * Returns undefined if no config exists or if main field is not specified
15
+ */
16
+ function getEntryPointFromWranglerConfig() {
17
+ const configFile = (0, find_wrangler_config_1.findWranglerConfig)();
18
+ if (!configFile) {
19
+ return undefined;
20
+ }
21
+ const configPath = node_path_1.default.join(process.cwd(), configFile);
22
+ const configContent = node_fs_1.default.readFileSync(configPath, 'utf-8');
23
+ const extname = node_path_1.default.extname(configFile);
24
+ switch (extname) {
25
+ case '.toml': {
26
+ const mainMatch = configContent.match(/^main\s*=\s*["'](.+)["']/m);
27
+ return mainMatch ? mainMatch[1] : undefined;
28
+ }
29
+ case '.json':
30
+ case '.jsonc':
31
+ try {
32
+ const { jsonObject } = (0, ast_utils_1.parseJsonC)(configContent);
33
+ if (!jsonObject) {
34
+ return undefined;
35
+ }
36
+ const mainProperty = (0, ast_utils_1.getObjectProperty)(jsonObject, 'main');
37
+ if ((mainProperty?.value.type === 'StringLiteral' ||
38
+ mainProperty?.value.type === 'Literal') &&
39
+ typeof mainProperty.value.value === 'string') {
40
+ return mainProperty.value.value;
41
+ }
42
+ return undefined;
43
+ }
44
+ catch {
45
+ return undefined;
46
+ }
47
+ default:
48
+ return undefined;
49
+ }
50
+ }
51
+ exports.getEntryPointFromWranglerConfig = getEntryPointFromWranglerConfig;
52
+ //# sourceMappingURL=get-entry-point-from-wrangler-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"get-entry-point-from-wrangler-config.js","sourceRoot":"","sources":["../../../../src/cloudflare/wrangler/get-entry-point-from-wrangler-config.ts"],"names":[],"mappings":";;;;;;AAAA,sDAAyB;AACzB,0DAA6B;AAC7B,iEAA4D;AAC5D,qDAAsE;AAEzD,QAAA,iBAAiB,GAAG,cAAc,CAAC;AAEhD;;;GAGG;AACH,SAAgB,+BAA+B;IAC7C,MAAM,UAAU,GAAG,IAAA,yCAAkB,GAAE,CAAC;IAExC,IAAI,CAAC,UAAU,EAAE;QACf,OAAO,SAAS,CAAC;KAClB;IAED,MAAM,UAAU,GAAG,mBAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,UAAU,CAAC,CAAC;IACxD,MAAM,aAAa,GAAG,iBAAE,CAAC,YAAY,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;IAC3D,MAAM,OAAO,GAAG,mBAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAEzC,QAAQ,OAAO,EAAE;QACf,KAAK,OAAO,CAAC,CAAC;YACZ,MAAM,SAAS,GAAG,aAAa,CAAC,KAAK,CAAC,2BAA2B,CAAC,CAAC;YAEnE,OAAO,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;SAC7C;QAED,KAAK,OAAO,CAAC;QACb,KAAK,QAAQ;YACX,IAAI;gBACF,MAAM,EAAE,UAAU,EAAE,GAAG,IAAA,sBAAU,EAAC,aAAa,CAAC,CAAC;gBAEjD,IAAI,CAAC,UAAU,EAAE;oBACf,OAAO,SAAS,CAAC;iBAClB;gBAED,MAAM,YAAY,GAAG,IAAA,6BAAiB,EAAC,UAAU,EAAE,MAAM,CAAC,CAAC;gBAE3D,IACE,CAAC,YAAY,EAAE,KAAK,CAAC,IAAI,KAAK,eAAe;oBAC3C,YAAY,EAAE,KAAK,CAAC,IAAI,KAAK,SAAS,CAAC;oBACzC,OAAO,YAAY,CAAC,KAAK,CAAC,KAAK,KAAK,QAAQ,EAC5C;oBACA,OAAO,YAAY,CAAC,KAAK,CAAC,KAAK,CAAC;iBACjC;gBAED,OAAO,SAAS,CAAC;aAClB;YAAC,MAAM;gBACN,OAAO,SAAS,CAAC;aAClB;QAEH;YACE,OAAO,SAAS,CAAC;KACpB;AACH,CAAC;AA7CD,0EA6CC","sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\nimport { findWranglerConfig } from './find-wrangler-config';\nimport { parseJsonC, getObjectProperty } from '../../utils/ast-utils';\n\nexport const defaultEntryPoint = 'src/index.ts';\n\n/**\n * Reads the main entry point from the wrangler config file\n * Returns undefined if no config exists or if main field is not specified\n */\nexport function getEntryPointFromWranglerConfig(): string | undefined {\n const configFile = findWranglerConfig();\n\n if (!configFile) {\n return undefined;\n }\n\n const configPath = path.join(process.cwd(), configFile);\n const configContent = fs.readFileSync(configPath, 'utf-8');\n const extname = path.extname(configFile);\n\n switch (extname) {\n case '.toml': {\n const mainMatch = configContent.match(/^main\\s*=\\s*[\"'](.+)[\"']/m);\n\n return mainMatch ? mainMatch[1] : undefined;\n }\n\n case '.json':\n case '.jsonc':\n try {\n const { jsonObject } = parseJsonC(configContent);\n\n if (!jsonObject) {\n return undefined;\n }\n\n const mainProperty = getObjectProperty(jsonObject, 'main');\n\n if (\n (mainProperty?.value.type === 'StringLiteral' ||\n mainProperty?.value.type === 'Literal') &&\n typeof mainProperty.value.value === 'string'\n ) {\n return mainProperty.value.value;\n }\n\n return undefined;\n } catch {\n return undefined;\n }\n\n default:\n return undefined;\n }\n}\n"]}
@@ -0,0 +1,17 @@
1
+ type WranglerConfigUpdates = {
2
+ compatibility_date?: string;
3
+ compatibility_flags?: string[];
4
+ version_metadata?: {
5
+ binding: string;
6
+ };
7
+ [key: string]: unknown;
8
+ };
9
+ /**
10
+ * Updates the wrangler config file with the provided configuration
11
+ * Handles .toml (instructions only), .json, and .jsonc formats
12
+ * For arrays: merges and deduplicates values
13
+ * For objects: deep merges
14
+ * For other types: overwrites
15
+ */
16
+ export declare function updateWranglerConfig(updates: WranglerConfigUpdates): Promise<boolean>;
17
+ export {};
@@ -0,0 +1,173 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || function (mod) {
19
+ if (mod && mod.__esModule) return mod;
20
+ var result = {};
21
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
+ __setModuleDefault(result, mod);
23
+ return result;
24
+ };
25
+ var __importDefault = (this && this.__importDefault) || function (mod) {
26
+ return (mod && mod.__esModule) ? mod : { "default": mod };
27
+ };
28
+ Object.defineProperty(exports, "__esModule", { value: true });
29
+ exports.updateWranglerConfig = void 0;
30
+ // @ts-expect-error - clack is ESM and TS complains about that. It works though
31
+ const prompts_1 = __importDefault(require("@clack/prompts"));
32
+ const chalk_1 = __importDefault(require("chalk"));
33
+ const node_fs_1 = __importDefault(require("node:fs"));
34
+ const node_path_1 = __importDefault(require("node:path"));
35
+ const find_wrangler_config_1 = require("./find-wrangler-config");
36
+ const clack_1 = require("../../utils/clack");
37
+ const ast_utils_1 = require("../../utils/ast-utils");
38
+ const recast = __importStar(require("recast"));
39
+ const b = recast.types.builders;
40
+ const getTomlConfigSnippet = () => {
41
+ return (0, clack_1.makeCodeSnippet)(true, (unchanged, plus) => plus(`
42
+ compatibility_flags = ["nodejs_als"]
43
+ compatibility_date = "${new Date().toISOString().slice(0, 10)}"
44
+
45
+ [version_metadata]
46
+ binding = "CF_VERSION_METADATA"`));
47
+ };
48
+ /**
49
+ * Updates the wrangler config file with the provided configuration
50
+ * Handles .toml (instructions only), .json, and .jsonc formats
51
+ * For arrays: merges and deduplicates values
52
+ * For objects: deep merges
53
+ * For other types: overwrites
54
+ */
55
+ async function updateWranglerConfig(updates) {
56
+ const configFile = (0, find_wrangler_config_1.findWranglerConfig)();
57
+ if (!configFile) {
58
+ prompts_1.default.log.warn('No wrangler config file found.');
59
+ return false;
60
+ }
61
+ const configPath = node_path_1.default.join(process.cwd(), configFile);
62
+ try {
63
+ const configContent = node_fs_1.default.readFileSync(configPath, 'utf-8');
64
+ const extname = node_path_1.default.extname(configFile);
65
+ switch (extname) {
66
+ case '.jsonc':
67
+ case '.json':
68
+ updateJsoncConfig(configPath, configContent, updates);
69
+ prompts_1.default.log.success(`Updated ${chalk_1.default.cyan(configFile)} with Sentry configuration.`);
70
+ break;
71
+ case '.toml':
72
+ await (0, clack_1.showCopyPasteInstructions)({
73
+ filename: configFile,
74
+ codeSnippet: getTomlConfigSnippet(),
75
+ });
76
+ break;
77
+ }
78
+ return true;
79
+ }
80
+ catch (error) {
81
+ prompts_1.default.log.error(`Failed to update ${chalk_1.default.cyan(configFile)}: ${error instanceof Error ? error.message : String(error)}`);
82
+ return false;
83
+ }
84
+ }
85
+ exports.updateWranglerConfig = updateWranglerConfig;
86
+ /**
87
+ * Sets a string property in a JSON/JSONC config object.
88
+ * Overwrites any existing value.
89
+ *
90
+ * @param jsonObject The object expression to update
91
+ * @param propertyName The name of the string property
92
+ * @param value The string value to set
93
+ */
94
+ function setStringProperty(jsonObject, propertyName, value) {
95
+ (0, ast_utils_1.setOrUpdateObjectProperty)(jsonObject, propertyName, b.stringLiteral(value));
96
+ }
97
+ /**
98
+ * Merges an array property in a JSON/JSONC config object.
99
+ * Extracts existing array values, merges with new values, and deduplicates.
100
+ *
101
+ * @param jsonObject The object expression to update
102
+ * @param propertyName The name of the array property
103
+ * @param newValues The new array values to merge in
104
+ */
105
+ function mergeArrayProperty(jsonObject, propertyName, newValues) {
106
+ const existingProperty = (0, ast_utils_1.getObjectProperty)(jsonObject, propertyName);
107
+ const existingValues = [];
108
+ // Extract existing values from the AST if they exist
109
+ if (existingProperty && existingProperty.value.type === 'ArrayExpression') {
110
+ existingProperty.value.elements.forEach((element) => {
111
+ if (element &&
112
+ (element.type === 'StringLiteral' || element.type === 'Literal') &&
113
+ typeof element.value === 'string') {
114
+ existingValues.push(element.value);
115
+ }
116
+ });
117
+ }
118
+ // Merge existing and new values, deduplicate
119
+ const allValues = [...existingValues, ...newValues];
120
+ const uniqueValues = Array.from(new Set(allValues));
121
+ (0, ast_utils_1.setOrUpdateObjectProperty)(jsonObject, propertyName, b.arrayExpression(uniqueValues.map((value) => b.stringLiteral(value))));
122
+ }
123
+ /**
124
+ * Merges properties into a nested object property in a JSON/JSONC config object.
125
+ * Gets or creates the nested ObjectExpression if it doesn't exist,
126
+ * then merges the provided properties into it, preserving existing properties.
127
+ *
128
+ * @param jsonObject The object expression to update
129
+ * @param propertyName The name of the nested object property
130
+ * @param updates The properties to merge into the nested object
131
+ */
132
+ function setObjectProperty(jsonObject, propertyName, updates) {
133
+ const existingProperty = (0, ast_utils_1.getObjectProperty)(jsonObject, propertyName);
134
+ let nestedObject;
135
+ if (existingProperty && existingProperty.value.type === 'ObjectExpression') {
136
+ nestedObject = existingProperty.value;
137
+ }
138
+ else {
139
+ nestedObject = b.objectExpression([]);
140
+ (0, ast_utils_1.setOrUpdateObjectProperty)(jsonObject, propertyName, nestedObject);
141
+ }
142
+ updateJsoncObject(nestedObject, updates);
143
+ }
144
+ function updateJsoncObject(jsonObject, updates) {
145
+ for (const [key, value] of Object.entries(updates)) {
146
+ if (value === null || value === undefined) {
147
+ continue;
148
+ }
149
+ if (typeof value === 'string') {
150
+ setStringProperty(jsonObject, key, value);
151
+ }
152
+ else if (Array.isArray(value)) {
153
+ mergeArrayProperty(jsonObject, key, value);
154
+ }
155
+ else if (typeof value === 'object') {
156
+ setObjectProperty(jsonObject, key, value);
157
+ }
158
+ }
159
+ }
160
+ /**
161
+ * Updates a JSON/JSONC config file using jsonc-parser
162
+ * Preserves comments and formatting while merging values
163
+ */
164
+ function updateJsoncConfig(configPath, content, updates) {
165
+ const { jsonObject, ast } = (0, ast_utils_1.parseJsonC)(content);
166
+ if (!jsonObject) {
167
+ throw new Error('Failed to parse JSON/JSONC config file');
168
+ }
169
+ updateJsoncObject(jsonObject, updates);
170
+ const code = (0, ast_utils_1.printJsonC)(ast);
171
+ node_fs_1.default.writeFileSync(configPath, code, 'utf-8');
172
+ }
173
+ //# sourceMappingURL=update-wrangler-config.js.map