@sentry/wizard 3.39.0 → 3.41.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 (221) hide show
  1. package/CHANGELOG.md +12 -1
  2. package/README.md +21 -21
  3. package/bin.ts +5 -0
  4. package/codecov.yml +15 -0
  5. package/dist/bin.js +4 -0
  6. package/dist/bin.js.map +1 -1
  7. package/dist/e2e-tests/jest.config.d.ts +1 -0
  8. package/dist/e2e-tests/jest.config.js +1 -0
  9. package/dist/e2e-tests/jest.config.js.map +1 -1
  10. package/dist/e2e-tests/tests/flutter.test.d.ts +1 -0
  11. package/dist/e2e-tests/tests/flutter.test.js +190 -0
  12. package/dist/e2e-tests/tests/flutter.test.js.map +1 -0
  13. package/dist/e2e-tests/utils/index.d.ts +11 -0
  14. package/dist/e2e-tests/utils/index.js +36 -1
  15. package/dist/e2e-tests/utils/index.js.map +1 -1
  16. package/dist/lib/Constants.d.ts +1 -0
  17. package/dist/lib/Constants.js +5 -0
  18. package/dist/lib/Constants.js.map +1 -1
  19. package/dist/package.json +3 -2
  20. package/dist/src/apple/apple-wizard.js +2 -3
  21. package/dist/src/apple/apple-wizard.js.map +1 -1
  22. package/dist/src/apple/code-tools.d.ts +10 -0
  23. package/dist/src/apple/code-tools.js +16 -12
  24. package/dist/src/apple/code-tools.js.map +1 -1
  25. package/dist/src/apple/fastlane.d.ts +23 -0
  26. package/dist/src/apple/fastlane.js +11 -7
  27. package/dist/src/apple/fastlane.js.map +1 -1
  28. package/dist/src/apple/templates.d.ts +1 -1
  29. package/dist/src/apple/templates.js +0 -2
  30. package/dist/src/apple/templates.js.map +1 -1
  31. package/dist/src/apple/xcode-manager.d.ts +13 -9
  32. package/dist/src/apple/xcode-manager.js +146 -61
  33. package/dist/src/apple/xcode-manager.js.map +1 -1
  34. package/dist/src/flutter/code-tools.d.ts +17 -0
  35. package/dist/src/flutter/code-tools.js +263 -0
  36. package/dist/src/flutter/code-tools.js.map +1 -0
  37. package/dist/src/flutter/flutter-wizard.d.ts +2 -0
  38. package/dist/src/flutter/flutter-wizard.js +171 -0
  39. package/dist/src/flutter/flutter-wizard.js.map +1 -0
  40. package/dist/src/flutter/templates.d.ts +9 -0
  41. package/dist/src/flutter/templates.js +40 -0
  42. package/dist/src/flutter/templates.js.map +1 -0
  43. package/dist/src/nextjs/nextjs-wizard.js +5 -3
  44. package/dist/src/nextjs/nextjs-wizard.js.map +1 -1
  45. package/dist/src/nuxt/nuxt-wizard.js +6 -4
  46. package/dist/src/nuxt/nuxt-wizard.js.map +1 -1
  47. package/dist/src/nuxt/sdk-setup.d.ts +1 -1
  48. package/dist/src/nuxt/sdk-setup.js +2 -1
  49. package/dist/src/nuxt/sdk-setup.js.map +1 -1
  50. package/dist/src/react-native/react-native-wizard.js +5 -3
  51. package/dist/src/react-native/react-native-wizard.js.map +1 -1
  52. package/dist/src/remix/remix-wizard.js +5 -3
  53. package/dist/src/remix/remix-wizard.js.map +1 -1
  54. package/dist/src/run.d.ts +2 -1
  55. package/dist/src/run.js +40 -32
  56. package/dist/src/run.js.map +1 -1
  57. package/dist/src/sveltekit/sveltekit-wizard.js +5 -3
  58. package/dist/src/sveltekit/sveltekit-wizard.js.map +1 -1
  59. package/dist/src/utils/clack-utils.d.ts +4 -2
  60. package/dist/src/utils/clack-utils.js +18 -12
  61. package/dist/src/utils/clack-utils.js.map +1 -1
  62. package/dist/src/utils/package-manager.d.ts +1 -0
  63. package/dist/src/utils/package-manager.js +5 -0
  64. package/dist/src/utils/package-manager.js.map +1 -1
  65. package/dist/src/utils/types.d.ts +9 -0
  66. package/dist/src/utils/types.js.map +1 -1
  67. package/dist/test/apple/cocoapod.test.d.ts +1 -0
  68. package/dist/test/apple/cocoapod.test.js +409 -0
  69. package/dist/test/apple/cocoapod.test.js.map +1 -0
  70. package/dist/test/apple/code-tools.test.d.ts +1 -0
  71. package/dist/test/apple/code-tools.test.js +673 -0
  72. package/dist/test/apple/code-tools.test.js.map +1 -0
  73. package/dist/test/apple/fastfile.test.d.ts +1 -0
  74. package/dist/test/apple/fastfile.test.js +431 -0
  75. package/dist/test/apple/fastfile.test.js.map +1 -0
  76. package/dist/test/apple/templates.test.d.ts +1 -0
  77. package/dist/test/apple/templates.test.js +73 -0
  78. package/dist/test/apple/templates.test.js.map +1 -0
  79. package/dist/test/apple/xcode-manager.test.d.ts +1 -0
  80. package/dist/test/apple/xcode-manager.test.js +834 -0
  81. package/dist/test/apple/xcode-manager.test.js.map +1 -0
  82. package/dist/test/flutter/code-tools.test.d.ts +1 -0
  83. package/dist/test/flutter/code-tools.test.js +84 -0
  84. package/dist/test/flutter/code-tools.test.js.map +1 -0
  85. package/dist/test/flutter/templates.test.d.ts +1 -0
  86. package/dist/test/flutter/templates.test.js +41 -0
  87. package/dist/test/flutter/templates.test.js.map +1 -0
  88. package/dist/test/utils/clack-utils.test.js +89 -0
  89. package/dist/test/utils/clack-utils.test.js.map +1 -1
  90. package/e2e-tests/README.md +5 -1
  91. package/e2e-tests/jest.config.ts +1 -0
  92. package/e2e-tests/test-applications/apple/damaged-missing-configuration-list/Project.xcodeproj/project.pbxproj +52 -0
  93. package/e2e-tests/test-applications/apple/damaged-missing-configuration-list/Project.xcodeproj/xcshareddata/xcschemes/Project1.xcscheme +78 -0
  94. package/e2e-tests/test-applications/apple/no-targets/Project.xcodeproj/project.pbxproj +62 -0
  95. package/e2e-tests/test-applications/apple/no-targets/Project.xcodeproj/xcshareddata/xcschemes/Project1.xcscheme +78 -0
  96. package/e2e-tests/test-applications/apple/spm-swiftui-multi-targets/Project.xcodeproj/project.pbxproj +470 -0
  97. package/e2e-tests/test-applications/apple/spm-swiftui-multi-targets/Project.xcodeproj/xcshareddata/xcschemes/Project1.xcscheme +78 -0
  98. package/e2e-tests/test-applications/apple/spm-swiftui-multi-targets/Project1/ContentView.swift +7 -0
  99. package/e2e-tests/test-applications/apple/spm-swiftui-multi-targets/Project1/Project1App.swift +10 -0
  100. package/e2e-tests/test-applications/apple/spm-swiftui-multi-targets/Project2/ContentView.swift +7 -0
  101. package/e2e-tests/test-applications/apple/spm-swiftui-multi-targets/Project2/Project2App.swift +10 -0
  102. package/e2e-tests/test-applications/apple/spm-swiftui-single-target/Project.xcodeproj/project.pbxproj +382 -0
  103. package/e2e-tests/test-applications/apple/spm-swiftui-single-target/Project.xcodeproj/xcshareddata/xcschemes/Project.xcscheme +78 -0
  104. package/e2e-tests/test-applications/apple/spm-swiftui-single-target/Sources/ContentView.swift +7 -0
  105. package/e2e-tests/test-applications/apple/spm-swiftui-single-target/Sources/MainApp.swift +10 -0
  106. package/e2e-tests/test-applications/flutter-test-app/.metadata +45 -0
  107. package/e2e-tests/test-applications/flutter-test-app/README.md +16 -0
  108. package/e2e-tests/test-applications/flutter-test-app/analysis_options.yaml +28 -0
  109. package/e2e-tests/test-applications/flutter-test-app/android/app/build.gradle +44 -0
  110. package/e2e-tests/test-applications/flutter-test-app/android/app/src/debug/AndroidManifest.xml +7 -0
  111. package/e2e-tests/test-applications/flutter-test-app/android/app/src/main/AndroidManifest.xml +45 -0
  112. package/e2e-tests/test-applications/flutter-test-app/android/app/src/main/kotlin/com/example/flutter_magic/MainActivity.kt +5 -0
  113. package/e2e-tests/test-applications/flutter-test-app/android/app/src/main/res/drawable/launch_background.xml +12 -0
  114. package/e2e-tests/test-applications/flutter-test-app/android/app/src/main/res/drawable-v21/launch_background.xml +12 -0
  115. package/e2e-tests/test-applications/flutter-test-app/android/app/src/main/res/mipmap-hdpi/ic_launcher.png +0 -0
  116. package/e2e-tests/test-applications/flutter-test-app/android/app/src/main/res/mipmap-mdpi/ic_launcher.png +0 -0
  117. package/e2e-tests/test-applications/flutter-test-app/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png +0 -0
  118. package/e2e-tests/test-applications/flutter-test-app/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png +0 -0
  119. package/e2e-tests/test-applications/flutter-test-app/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png +0 -0
  120. package/e2e-tests/test-applications/flutter-test-app/android/app/src/main/res/values/styles.xml +18 -0
  121. package/e2e-tests/test-applications/flutter-test-app/android/app/src/main/res/values-night/styles.xml +18 -0
  122. package/e2e-tests/test-applications/flutter-test-app/android/app/src/profile/AndroidManifest.xml +7 -0
  123. package/e2e-tests/test-applications/flutter-test-app/android/build.gradle +18 -0
  124. package/e2e-tests/test-applications/flutter-test-app/android/gradle/wrapper/gradle-wrapper.properties +5 -0
  125. package/e2e-tests/test-applications/flutter-test-app/android/gradle.properties +3 -0
  126. package/e2e-tests/test-applications/flutter-test-app/android/settings.gradle +25 -0
  127. package/e2e-tests/test-applications/flutter-test-app/lib/main.dart +125 -0
  128. package/e2e-tests/test-applications/flutter-test-app/linux/CMakeLists.txt +145 -0
  129. package/e2e-tests/test-applications/flutter-test-app/linux/flutter/CMakeLists.txt +88 -0
  130. package/e2e-tests/test-applications/flutter-test-app/linux/flutter/generated_plugin_registrant.cc +11 -0
  131. package/e2e-tests/test-applications/flutter-test-app/linux/flutter/generated_plugin_registrant.h +15 -0
  132. package/e2e-tests/test-applications/flutter-test-app/linux/flutter/generated_plugins.cmake +23 -0
  133. package/e2e-tests/test-applications/flutter-test-app/linux/main.cc +6 -0
  134. package/e2e-tests/test-applications/flutter-test-app/linux/my_application.cc +124 -0
  135. package/e2e-tests/test-applications/flutter-test-app/linux/my_application.h +18 -0
  136. package/e2e-tests/test-applications/flutter-test-app/macos/Flutter/Flutter-Debug.xcconfig +2 -0
  137. package/e2e-tests/test-applications/flutter-test-app/macos/Flutter/Flutter-Release.xcconfig +2 -0
  138. package/e2e-tests/test-applications/flutter-test-app/macos/Flutter/GeneratedPluginRegistrant.swift +10 -0
  139. package/e2e-tests/test-applications/flutter-test-app/macos/Podfile +43 -0
  140. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/AppDelegate.swift +9 -0
  141. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json +68 -0
  142. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_1024.png +0 -0
  143. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_128.png +0 -0
  144. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_16.png +0 -0
  145. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_256.png +0 -0
  146. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_32.png +0 -0
  147. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_512.png +0 -0
  148. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_64.png +0 -0
  149. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Base.lproj/MainMenu.xib +343 -0
  150. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Configs/AppInfo.xcconfig +14 -0
  151. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Configs/Debug.xcconfig +2 -0
  152. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Configs/Release.xcconfig +2 -0
  153. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Configs/Warnings.xcconfig +13 -0
  154. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/DebugProfile.entitlements +12 -0
  155. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Info.plist +32 -0
  156. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/MainFlutterWindow.swift +15 -0
  157. package/e2e-tests/test-applications/flutter-test-app/macos/Runner/Release.entitlements +8 -0
  158. package/e2e-tests/test-applications/flutter-test-app/macos/Runner.xcodeproj/project.pbxproj +705 -0
  159. package/e2e-tests/test-applications/flutter-test-app/macos/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist +8 -0
  160. package/e2e-tests/test-applications/flutter-test-app/macos/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme +98 -0
  161. package/e2e-tests/test-applications/flutter-test-app/macos/Runner.xcworkspace/contents.xcworkspacedata +7 -0
  162. package/e2e-tests/test-applications/flutter-test-app/macos/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist +8 -0
  163. package/e2e-tests/test-applications/flutter-test-app/macos/RunnerTests/RunnerTests.swift +12 -0
  164. package/e2e-tests/test-applications/flutter-test-app/pubspec.lock +213 -0
  165. package/e2e-tests/test-applications/flutter-test-app/pubspec.yaml +89 -0
  166. package/e2e-tests/test-applications/flutter-test-app/test/widget_test.dart +30 -0
  167. package/e2e-tests/test-applications/flutter-test-app/web/favicon.png +0 -0
  168. package/e2e-tests/test-applications/flutter-test-app/web/icons/Icon-192.png +0 -0
  169. package/e2e-tests/test-applications/flutter-test-app/web/icons/Icon-512.png +0 -0
  170. package/e2e-tests/test-applications/flutter-test-app/web/icons/Icon-maskable-192.png +0 -0
  171. package/e2e-tests/test-applications/flutter-test-app/web/icons/Icon-maskable-512.png +0 -0
  172. package/e2e-tests/test-applications/flutter-test-app/web/index.html +38 -0
  173. package/e2e-tests/test-applications/flutter-test-app/web/manifest.json +35 -0
  174. package/e2e-tests/test-applications/flutter-test-app/windows/CMakeLists.txt +108 -0
  175. package/e2e-tests/test-applications/flutter-test-app/windows/flutter/CMakeLists.txt +109 -0
  176. package/e2e-tests/test-applications/flutter-test-app/windows/flutter/generated_plugin_registrant.cc +11 -0
  177. package/e2e-tests/test-applications/flutter-test-app/windows/flutter/generated_plugin_registrant.h +15 -0
  178. package/e2e-tests/test-applications/flutter-test-app/windows/flutter/generated_plugins.cmake +23 -0
  179. package/e2e-tests/test-applications/flutter-test-app/windows/runner/CMakeLists.txt +40 -0
  180. package/e2e-tests/test-applications/flutter-test-app/windows/runner/Runner.rc +121 -0
  181. package/e2e-tests/test-applications/flutter-test-app/windows/runner/flutter_window.cpp +71 -0
  182. package/e2e-tests/test-applications/flutter-test-app/windows/runner/flutter_window.h +33 -0
  183. package/e2e-tests/test-applications/flutter-test-app/windows/runner/main.cpp +43 -0
  184. package/e2e-tests/test-applications/flutter-test-app/windows/runner/resource.h +16 -0
  185. package/e2e-tests/test-applications/flutter-test-app/windows/runner/resources/app_icon.ico +0 -0
  186. package/e2e-tests/test-applications/flutter-test-app/windows/runner/runner.exe.manifest +14 -0
  187. package/e2e-tests/test-applications/flutter-test-app/windows/runner/utils.cpp +65 -0
  188. package/e2e-tests/test-applications/flutter-test-app/windows/runner/utils.h +19 -0
  189. package/e2e-tests/test-applications/flutter-test-app/windows/runner/win32_window.cpp +288 -0
  190. package/e2e-tests/test-applications/flutter-test-app/windows/runner/win32_window.h +102 -0
  191. package/e2e-tests/tests/flutter.test.ts +127 -0
  192. package/e2e-tests/utils/index.ts +33 -0
  193. package/lib/Constants.ts +5 -0
  194. package/package.json +3 -2
  195. package/src/apple/apple-wizard.ts +2 -3
  196. package/src/apple/code-tools.ts +21 -6
  197. package/src/apple/fastlane.ts +18 -2
  198. package/src/apple/templates.ts +2 -2
  199. package/src/apple/xcode-manager.ts +186 -99
  200. package/src/flutter/code-tools.ts +284 -0
  201. package/src/flutter/flutter-wizard.ts +164 -0
  202. package/src/flutter/templates.ts +90 -0
  203. package/src/nextjs/nextjs-wizard.ts +5 -2
  204. package/src/nuxt/nuxt-wizard.ts +6 -3
  205. package/src/nuxt/sdk-setup.ts +2 -0
  206. package/src/react-native/react-native-wizard.ts +5 -2
  207. package/src/remix/remix-wizard.ts +5 -2
  208. package/src/run.ts +9 -0
  209. package/src/sveltekit/sveltekit-wizard.ts +5 -2
  210. package/src/utils/clack-utils.ts +16 -4
  211. package/src/utils/package-manager.ts +6 -0
  212. package/src/utils/types.ts +10 -0
  213. package/test/apple/cocoapod.test.ts +306 -0
  214. package/test/apple/code-tools.test.ts +1042 -0
  215. package/test/apple/fastfile.test.ts +550 -0
  216. package/test/apple/templates.test.ts +191 -0
  217. package/test/apple/xcode-manager.test.ts +1066 -0
  218. package/test/flutter/code-tools.test.ts +212 -0
  219. package/test/flutter/templates.test.ts +100 -0
  220. package/test/utils/clack-utils.test.ts +92 -0
  221. package/types/xcode.d.ts +526 -0
@@ -0,0 +1,284 @@
1
+ import * as fs from 'fs';
2
+ import * as path from 'path';
3
+ import * as Sentry from '@sentry/node';
4
+ // @ts-ignore - clack is ESM and TS complains about that. It works though
5
+ import * as clack from '@clack/prompts';
6
+ import chalk from 'chalk';
7
+ import {
8
+ sentryImport,
9
+ pubspecOptions,
10
+ sentryProperties,
11
+ initSnippet,
12
+ } from './templates';
13
+ import { featureSelectionPrompt } from '../utils/clack-utils';
14
+
15
+ /**
16
+ * Recursively finds a file per name in subfolders.
17
+ * @param dir - The directory to start searching.
18
+ * @param name - The name of the file including path extension.
19
+ * @returns The path to the main.dart file or null if not found.
20
+ */
21
+ export function findFile(dir: string, name: string): string | null {
22
+ const files: string[] = fs.readdirSync(dir);
23
+
24
+ for (const file of files) {
25
+ const fullPath: string = path.join(dir, file);
26
+ const stats: fs.Stats = fs.statSync(fullPath);
27
+
28
+ if (stats.isDirectory()) {
29
+ const result: string | null = findFile(fullPath, name);
30
+ if (result) {
31
+ return result;
32
+ }
33
+ } else if (file === name) {
34
+ return fullPath;
35
+ }
36
+ }
37
+
38
+ return null;
39
+ }
40
+
41
+ export function patchPubspec(
42
+ pubspecFile: string | null,
43
+ sentryDartFlutterVersion: string,
44
+ sentryDartPluginVersion: string,
45
+ project: string,
46
+ org: string,
47
+ ): boolean {
48
+ try {
49
+ if (!pubspecFile) {
50
+ throw new Error('pubspec.yaml is not provided or invalid.');
51
+ }
52
+
53
+ let pubspecContent = fs.readFileSync(pubspecFile, 'utf8');
54
+
55
+ if (!pubspecContent.includes('sentry_flutter:')) {
56
+ const dependenciesIndex = getDependenciesLocation(pubspecContent);
57
+
58
+ pubspecContent =
59
+ pubspecContent.slice(0, dependenciesIndex) +
60
+ ` sentry_flutter: ${sentryDartFlutterVersion}\n` +
61
+ pubspecContent.slice(dependenciesIndex);
62
+
63
+ clack.log.success(
64
+ chalk.greenBright(
65
+ `${chalk.bold('sentry_flutter')} added to pubspec.yaml`,
66
+ ),
67
+ );
68
+ } else {
69
+ clack.log.success(
70
+ chalk.greenBright(
71
+ `${chalk.bold('sentry_flutter')} is already included in pubspec.yaml`,
72
+ ),
73
+ );
74
+ }
75
+
76
+ if (!pubspecContent.includes('sentry_dart_plugin:')) {
77
+ const devDependenciesIndex = getDevDependenciesLocation(pubspecContent);
78
+ pubspecContent =
79
+ pubspecContent.slice(0, devDependenciesIndex) +
80
+ ` sentry_dart_plugin: ${sentryDartPluginVersion}\n` +
81
+ pubspecContent.slice(devDependenciesIndex);
82
+
83
+ clack.log.success(
84
+ chalk.greenBright(
85
+ `${chalk.bold('sentry_dart_plugin')} added to pubspec.yaml`,
86
+ ),
87
+ );
88
+ } else {
89
+ clack.log.success(
90
+ chalk.greenBright(
91
+ `${chalk.bold(
92
+ 'sentry_dart_plugin',
93
+ )} is already included in pubspec.yaml`,
94
+ ),
95
+ );
96
+ }
97
+
98
+ if (!pubspecContent.includes('sentry:')) {
99
+ pubspecContent += '\n';
100
+ pubspecContent += pubspecOptions(project, org);
101
+
102
+ clack.log.success(
103
+ chalk.greenBright(
104
+ `${chalk.bold('sentry plugin configuration')} added to pubspec.yaml`,
105
+ ),
106
+ );
107
+ } else {
108
+ clack.log.success(
109
+ chalk.greenBright(
110
+ `${chalk.bold(
111
+ 'sentry plugin configuration',
112
+ )} is already included in pubspec.yaml`,
113
+ ),
114
+ );
115
+ }
116
+
117
+ fs.writeFileSync(pubspecFile, pubspecContent, 'utf8');
118
+
119
+ return true;
120
+ } catch (error) {
121
+ clack.log.warn(`Failed to read/write ${chalk.cyan('pubspec.yaml')} file.`);
122
+ Sentry.captureException(error);
123
+ return false;
124
+ }
125
+ }
126
+
127
+ export function addProperties(pubspecFile: string | null, authToken: string) {
128
+ try {
129
+ if (!pubspecFile) {
130
+ throw new Error('pubspec.yaml is not provided or invalid.');
131
+ }
132
+
133
+ const pubspecDir = path.dirname(pubspecFile);
134
+ const sentryPropertiesFileName = 'sentry.properties';
135
+ const sentryPropertiesFile = path.join(
136
+ pubspecDir,
137
+ sentryPropertiesFileName,
138
+ );
139
+ const sentryPropertiesContent = sentryProperties(authToken);
140
+
141
+ fs.writeFileSync(sentryPropertiesFile, sentryPropertiesContent, 'utf8');
142
+
143
+ const gitignoreFile = path.join(pubspecDir, '.gitignore');
144
+ if (fs.existsSync(gitignoreFile)) {
145
+ fs.appendFileSync(gitignoreFile, `\n${sentryPropertiesFileName}\n`);
146
+ } else {
147
+ fs.writeFileSync(gitignoreFile, `${sentryPropertiesFileName}\n`, 'utf8');
148
+ }
149
+ return true;
150
+ } catch (error) {
151
+ clack.log.warn(`Failed to read/write ${chalk.cyan('pubspec.yaml')} file.`);
152
+ Sentry.captureException(error);
153
+ return false;
154
+ }
155
+ }
156
+
157
+ export async function patchMain(
158
+ mainFile: string | null,
159
+ dsn: string,
160
+ canEnableProfiling: boolean,
161
+ ): Promise<boolean> {
162
+ try {
163
+ if (!mainFile) {
164
+ throw new Error('pubspec.yaml is not provided or invalid.');
165
+ }
166
+
167
+ let mainContent = fs.readFileSync(mainFile, 'utf8');
168
+ if (
169
+ /import\s+['"]package[:]sentry_flutter\/sentry_flutter\.dart['"];?/i.test(
170
+ mainContent,
171
+ )
172
+ ) {
173
+ // sentry is already configured
174
+ clack.log.success(
175
+ chalk.greenBright(
176
+ `${chalk.bold('main.dart')} already has Sentry configured.`,
177
+ ),
178
+ );
179
+ return true;
180
+ }
181
+
182
+ const features = [
183
+ {
184
+ id: 'tracing',
185
+ prompt: `Do you want to enable ${chalk.bold(
186
+ 'Tracing',
187
+ )} to track the performance of your application?`,
188
+ enabledHint: 'recommended',
189
+ },
190
+ ];
191
+ if (canEnableProfiling) {
192
+ features.push({
193
+ id: 'profiling',
194
+ prompt: `Do you want to enable ${chalk.bold(
195
+ 'Profiling',
196
+ )} to analyze CPU usage and optimize performance-critical code on iOS & macOS?`,
197
+ enabledHint: 'recommended, tracing must be enabled',
198
+ });
199
+ }
200
+
201
+ const selectedFeatures = await featureSelectionPrompt(features);
202
+ const normalizedSelectedFeatures = {
203
+ tracing: selectedFeatures.tracing ?? false,
204
+ profiling: selectedFeatures.profiling ?? false,
205
+ };
206
+ mainContent = patchMainContent(
207
+ dsn,
208
+ mainContent,
209
+ normalizedSelectedFeatures,
210
+ );
211
+
212
+ fs.writeFileSync(mainFile, mainContent, 'utf8');
213
+
214
+ clack.log.success(
215
+ chalk.greenBright(
216
+ `Patched ${chalk.bold(
217
+ 'main.dart',
218
+ )} with the Sentry setup and test error snippet.`,
219
+ ),
220
+ );
221
+
222
+ return true;
223
+ } catch (error) {
224
+ clack.log.warn(`Failed to read/write ${chalk.cyan('main.dart')} file.`);
225
+ Sentry.captureException(error);
226
+ return false;
227
+ }
228
+ }
229
+
230
+ export function patchMainContent(
231
+ dsn: string,
232
+ mainContent: string,
233
+ selectedFeatures: {
234
+ tracing: boolean;
235
+ profiling: boolean;
236
+ },
237
+ ): string {
238
+ const importIndex = getLastImportLineLocation(mainContent);
239
+ mainContent =
240
+ mainContent.slice(0, importIndex) +
241
+ sentryImport +
242
+ mainContent.slice(importIndex);
243
+
244
+ // Find and replace `runApp(...)`
245
+ mainContent = mainContent.replace(
246
+ /runApp\(([\s\S]*?)\);/g, // Match the `runApp(...)` invocation
247
+ (_, runAppArgs) => initSnippet(dsn, selectedFeatures, runAppArgs as string),
248
+ );
249
+
250
+ // Make the `main` function async if it's not already
251
+ mainContent = mainContent.replace(
252
+ /void\s+main\(\)\s*\{/g,
253
+ 'Future<void> main() async {',
254
+ );
255
+
256
+ return mainContent;
257
+ }
258
+
259
+ export function getLastImportLineLocation(sourceCode: string): number {
260
+ const importRegex = /import\s+['"].*['"].*;/gim;
261
+ return getLastReqExpLocation(sourceCode, importRegex);
262
+ }
263
+
264
+ export function getDependenciesLocation(sourceCode: string): number {
265
+ const dependencyRegex = /^dependencies:\s*$/gim;
266
+ return getLastReqExpLocation(sourceCode, dependencyRegex);
267
+ }
268
+
269
+ export function getDevDependenciesLocation(sourceCode: string): number {
270
+ const dependencyRegex = /^dev_dependencies:\s*$/gim;
271
+ return getLastReqExpLocation(sourceCode, dependencyRegex);
272
+ }
273
+
274
+ // Helper
275
+
276
+ function getLastReqExpLocation(sourceCode: string, regExp: RegExp): number {
277
+ let match = regExp.exec(sourceCode);
278
+ let importIndex = 0;
279
+ while (match) {
280
+ importIndex = match.index + match[0].length + 1;
281
+ match = regExp.exec(sourceCode);
282
+ }
283
+ return importIndex;
284
+ }
@@ -0,0 +1,164 @@
1
+ import { WizardOptions } from '../utils/types';
2
+ import * as Sentry from '@sentry/node';
3
+ import * as codetools from './code-tools';
4
+ import * as fs from 'fs';
5
+ import * as path from 'path';
6
+ import { showCopyPasteInstructions } from '../utils/clack-utils';
7
+ import { pubspecSnippetColored, initSnippetColored } from './templates';
8
+ import { fetchSdkVersion } from '../utils/release-registry';
9
+ // @ts-ignore - clack is ESM and TS complains about that. It works though
10
+ import * as clack from '@clack/prompts';
11
+ import chalk from 'chalk';
12
+
13
+ import {
14
+ confirmContinueIfNoOrDirtyGitRepo,
15
+ getOrAskForProjectData,
16
+ printWelcome,
17
+ } from '../utils/clack-utils';
18
+
19
+ import { traceStep, withTelemetry } from '../telemetry';
20
+ import { findFile } from './code-tools';
21
+
22
+ export async function runFlutterWizard(options: WizardOptions): Promise<void> {
23
+ return withTelemetry(
24
+ {
25
+ enabled: options.telemetryEnabled,
26
+ integration: 'flutter',
27
+ wizardOptions: options,
28
+ },
29
+ () => runFlutterWizardWithTelemetry(options),
30
+ );
31
+ }
32
+
33
+ async function runFlutterWizardWithTelemetry(
34
+ options: WizardOptions,
35
+ ): Promise<void> {
36
+ printWelcome({
37
+ wizardName: 'Sentry Flutter Wizard',
38
+ promoCode: options.promoCode,
39
+ });
40
+
41
+ await confirmContinueIfNoOrDirtyGitRepo();
42
+
43
+ const { selectedProject, selfHosted, sentryUrl, authToken } =
44
+ await getOrAskForProjectData(options, 'flutter');
45
+
46
+ const projectDir = process.cwd();
47
+ const pubspecFile = path.join(projectDir, 'pubspec.yaml');
48
+ if (!fs.existsSync(pubspecFile)) {
49
+ clack.log.error(
50
+ `Could not find ${chalk.cyan(
51
+ 'pubspec.yaml',
52
+ )}. Make sure you run the wizard in the projects root folder.`,
53
+ );
54
+ return;
55
+ }
56
+
57
+ // ======== STEP 1. Add sentry_flutter and sentry_dart_plugin to pubspec.yaml ============
58
+
59
+ clack.log.step(
60
+ `Adding ${chalk.bold('Sentry')} to your apps ${chalk.cyan(
61
+ 'pubspec.yaml',
62
+ )} file.`,
63
+ );
64
+
65
+ const flutterVersion = await fetchSdkVersion('sentry.dart.flutter');
66
+ const flutterVersionOrAny = flutterVersion ? `^${flutterVersion}` : 'any';
67
+
68
+ const pluginVersion = await fetchSdkVersion('sentry.dart.plugin');
69
+ const pluginVersionOrAny = pluginVersion ? `^${pluginVersion}` : 'any';
70
+
71
+ const pubspecPatched = traceStep('Patch pubspec.yaml', () =>
72
+ codetools.patchPubspec(
73
+ pubspecFile,
74
+ flutterVersionOrAny,
75
+ pluginVersionOrAny,
76
+ selectedProject.slug,
77
+ selectedProject.organization.slug,
78
+ ),
79
+ );
80
+ if (!pubspecPatched) {
81
+ clack.log.warn(
82
+ `Could not patch ${chalk.cyan(
83
+ 'pubspec.yaml',
84
+ )}. Add the dependencies to it.`,
85
+ );
86
+ await showCopyPasteInstructions(
87
+ 'pubspec.yaml',
88
+ pubspecSnippetColored(
89
+ flutterVersionOrAny,
90
+ pluginVersionOrAny,
91
+ selectedProject.slug,
92
+ selectedProject.organization.slug,
93
+ ),
94
+ 'This ensures the Sentry SDK and plugin can be imported.',
95
+ );
96
+ }
97
+ Sentry.setTag('pubspec-patched', pubspecPatched);
98
+
99
+ // ======== STEP 2. Add sentry.properties with auth token ============
100
+
101
+ const propertiesAdded = traceStep('Add sentry.properties', () =>
102
+ codetools.addProperties(pubspecFile, authToken),
103
+ );
104
+ if (!propertiesAdded) {
105
+ clack.log.warn(
106
+ `We could not add ${chalk.cyan(
107
+ 'sentry.properties',
108
+ )} file in your project directory in order to provide an auth token for Sentry CLI. You'll have to add it manually, or you can set the SENTRY_AUTH_TOKEN environment variable instead. See https://docs.sentry.io/cli/configuration/#auth-token for more information.`,
109
+ );
110
+ } else {
111
+ clack.log.info(
112
+ `We created ${chalk.cyan(
113
+ 'sentry.properties',
114
+ )} file in your project directory in order to provide an auth token for Sentry CLI.\nIt was also added to your ".gitignore" file.\nAt your CI enviroment, you can set the SENTRY_AUTH_TOKEN environment variable instead. See https://docs.sentry.io/cli/configuration/#auth-token for more information.`,
115
+ );
116
+ }
117
+ Sentry.setTag('sentry-properties-added', pubspecPatched);
118
+
119
+ // ======== STEP 3. Patch main.dart with setup and a test error snippet ============
120
+
121
+ clack.log.step(
122
+ `Patching ${chalk.cyan('main.dart')} with setup and test error snippet.`,
123
+ );
124
+
125
+ const mainFile = findFile(`${projectDir}/lib`, 'main.dart');
126
+ const dsn = selectedProject.keys[0].dsn.public;
127
+ const canEnableProfiling =
128
+ fs.existsSync(`${projectDir}/ios`) || fs.existsSync(`${projectDir}/macos`);
129
+
130
+ const mainPatched = await traceStep('Patch main.dart', () =>
131
+ codetools.patchMain(mainFile, dsn, canEnableProfiling),
132
+ );
133
+ if (!mainPatched) {
134
+ clack.log.warn(
135
+ `Could not patch ${chalk.cyan(
136
+ 'main.dart',
137
+ )} file. Place the following code snippet within the apps main function.`,
138
+ );
139
+ await showCopyPasteInstructions(
140
+ 'main.dart',
141
+ initSnippetColored(dsn),
142
+ 'This ensures the Sentry SDK is ready to capture errors.',
143
+ );
144
+ }
145
+ Sentry.setTag('main-patched', mainPatched);
146
+
147
+ // ======== OUTRO ========
148
+
149
+ const issuesPageLink = selfHosted
150
+ ? `${sentryUrl}organizations/${selectedProject.organization.slug}/issues/?project=${selectedProject.id}`
151
+ : `https://${selectedProject.organization.slug}.sentry.io/issues/?project=${selectedProject.id}`;
152
+
153
+ clack.outro(`
154
+ ${chalk.greenBright('Successfully installed the Sentry Flutter SDK!')}
155
+
156
+ ${chalk.cyan(
157
+ `You can validate your setup by launching your application and checking Sentry issues page afterwards
158
+ ${issuesPageLink}`,
159
+ )}
160
+
161
+ Check out the SDK documentation for further configuration:
162
+ https://docs.sentry.io/platforms/flutter/
163
+ `);
164
+ }
@@ -0,0 +1,90 @@
1
+ import { makeCodeSnippet } from '../utils/clack-utils';
2
+
3
+ export const sentryImport = `import 'package:sentry_flutter/sentry_flutter.dart';\n`;
4
+
5
+ export function pubspecOptions(project: string, org: string): string {
6
+ return `sentry:
7
+ upload_debug_symbols: true
8
+ upload_source_maps: true
9
+ project: ${project}
10
+ org: ${org}
11
+ `;
12
+ }
13
+
14
+ export function sentryProperties(authToken: string): string {
15
+ return `auth_token=${authToken}`;
16
+ }
17
+
18
+ export function initSnippet(
19
+ dsn: string,
20
+ selectedFeaturesMap: {
21
+ tracing: boolean;
22
+ profiling: boolean;
23
+ },
24
+ runApp: string,
25
+ ): string {
26
+ let snippet = `await SentryFlutter.init(
27
+ (options) {
28
+ options.dsn = '${dsn}';`;
29
+
30
+ if (selectedFeaturesMap.tracing) {
31
+ snippet += `
32
+ // Set tracesSampleRate to 1.0 to capture 100% of transactions for tracing.
33
+ // We recommend adjusting this value in production.
34
+ options.tracesSampleRate = 1.0;`;
35
+ }
36
+
37
+ if (selectedFeaturesMap.profiling && selectedFeaturesMap.tracing) {
38
+ snippet += `
39
+ // The sampling rate for profiling is relative to tracesSampleRate
40
+ // Setting to 1.0 will profile 100% of sampled transactions:
41
+ options.profilesSampleRate = 1.0;`;
42
+ }
43
+
44
+ snippet += `
45
+ },
46
+ appRunner: () => runApp(SentryWidget(child: ${runApp})),
47
+ );
48
+ // TODO: Remove this line after sending the first sample event to sentry.
49
+ await Sentry.captureException(Exception('This is a sample exception.'));`;
50
+
51
+ return snippet;
52
+ }
53
+
54
+ export function pubspecSnippetColored(
55
+ sentryVersion: string,
56
+ pluginVersion: string,
57
+ project: string,
58
+ org: string,
59
+ ): string {
60
+ const snippet = `dependencies:
61
+ sentry_flutter: ${sentryVersion}
62
+
63
+ dev_dependencies:
64
+ sentry_dart_plugin: ${pluginVersion}
65
+
66
+ ${pubspecOptions(project, org)}`;
67
+
68
+ return makeCodeSnippet(true, (_unchanged, plus, _minus) => {
69
+ return plus(snippet);
70
+ });
71
+ }
72
+
73
+ export function initSnippetColored(dsn: string): string {
74
+ const snippet = `import 'package:sentry_flutter/sentry_flutter.dart';
75
+
76
+ Future<void>main() async {
77
+ await SentryFlutter.init(
78
+ (options) {
79
+ options.dsn = '${dsn}';
80
+ // Set tracesSampleRate to 1.0 to capture 100% of transactions for tracing.
81
+ // We recommend adjusting this value in production.
82
+ options.tracesSampleRate = 1.0;
83
+ },
84
+ appRunner: () => runApp(SentryWidget(child: YourApp())),
85
+ )
86
+ }`;
87
+ return makeCodeSnippet(true, (_unchanged, plus, _minus) => {
88
+ return plus(snippet);
89
+ });
90
+ }
@@ -65,10 +65,12 @@ export function runNextjsWizard(options: WizardOptions) {
65
65
  export async function runNextjsWizardWithTelemetry(
66
66
  options: WizardOptions,
67
67
  ): Promise<void> {
68
+ const { promoCode, telemetryEnabled, forceInstall } = options;
69
+
68
70
  printWelcome({
69
71
  wizardName: 'Sentry Next.js Wizard',
70
- promoCode: options.promoCode,
71
- telemetryEnabled: options.telemetryEnabled,
72
+ promoCode,
73
+ telemetryEnabled,
72
74
  });
73
75
 
74
76
  const typeScriptDetected = isUsingTypeScript();
@@ -96,6 +98,7 @@ export async function runNextjsWizardWithTelemetry(
96
98
  packageName: '@sentry/nextjs@^8',
97
99
  packageNameDisplayLabel: '@sentry/nextjs',
98
100
  alreadyInstalled: !!packageJson?.dependencies?.['@sentry/nextjs'],
101
+ forceInstall,
99
102
  });
100
103
 
101
104
  await traceStep('configure-sdk', async () => {
@@ -50,10 +50,12 @@ export function runNuxtWizard(options: WizardOptions) {
50
50
  export async function runNuxtWizardWithTelemetry(
51
51
  options: WizardOptions,
52
52
  ): Promise<void> {
53
+ const { promoCode, telemetryEnabled, forceInstall } = options;
54
+
53
55
  printWelcome({
54
56
  wizardName: 'Sentry Nuxt Wizard',
55
- promoCode: options.promoCode,
56
- telemetryEnabled: options.telemetryEnabled,
57
+ promoCode,
58
+ telemetryEnabled,
57
59
  });
58
60
 
59
61
  await confirmContinueIfNoOrDirtyGitRepo();
@@ -95,7 +97,7 @@ export async function runNuxtWizardWithTelemetry(
95
97
 
96
98
  const packageManager = await getPackageManager();
97
99
 
98
- await addNuxtOverrides(packageJson, packageManager, minVer);
100
+ await addNuxtOverrides(packageJson, packageManager, minVer, forceInstall);
99
101
 
100
102
  const sdkAlreadyInstalled = hasPackageInstalled('@sentry/nuxt', packageJson);
101
103
  Sentry.setTag('sdk-already-installed', sdkAlreadyInstalled);
@@ -104,6 +106,7 @@ export async function runNuxtWizardWithTelemetry(
104
106
  packageName: '@sentry/nuxt',
105
107
  alreadyInstalled: sdkAlreadyInstalled,
106
108
  packageManager,
109
+ forceInstall,
107
110
  });
108
111
 
109
112
  await addDotEnvSentryBuildPluginFile(authToken);
@@ -250,6 +250,7 @@ export async function addNuxtOverrides(
250
250
  packageJson: PackageDotJson,
251
251
  packageManager: PackageManager,
252
252
  nuxtMinVer: SemVer | null,
253
+ forceInstall?: boolean,
253
254
  ) {
254
255
  const isPNPM = PNPM.detect();
255
256
 
@@ -306,6 +307,7 @@ export async function addNuxtOverrides(
306
307
  packageName: 'import-in-the-middle',
307
308
  alreadyInstalled: iitmAlreadyInstalled,
308
309
  packageManager,
310
+ forceInstall,
309
311
  });
310
312
  }
311
313
  }
@@ -111,10 +111,12 @@ export async function runReactNativeWizardWithTelemetry(
111
111
  return runReactNativeUninstall(options);
112
112
  }
113
113
 
114
+ const { promoCode, telemetryEnabled, forceInstall } = options;
115
+
114
116
  printWelcome({
115
117
  wizardName: 'Sentry React Native Wizard',
116
- promoCode: options.promoCode,
117
- telemetryEnabled: options.telemetryEnabled,
118
+ promoCode,
119
+ telemetryEnabled,
118
120
  });
119
121
 
120
122
  await confirmContinueIfNoOrDirtyGitRepo();
@@ -147,6 +149,7 @@ Or setup using ${chalk.cyan(
147
149
  await installPackage({
148
150
  packageName: RN_SDK_PACKAGE,
149
151
  alreadyInstalled: hasPackageInstalled(RN_SDK_PACKAGE, packageJson),
152
+ forceInstall,
150
153
  });
151
154
  const sdkVersion = getPackageVersion(
152
155
  RN_SDK_PACKAGE,
@@ -52,10 +52,12 @@ export async function runRemixWizard(options: WizardOptions): Promise<void> {
52
52
  async function runRemixWizardWithTelemetry(
53
53
  options: WizardOptions,
54
54
  ): Promise<void> {
55
+ const { promoCode, telemetryEnabled, forceInstall } = options;
56
+
55
57
  printWelcome({
56
58
  wizardName: 'Sentry Remix Wizard',
57
- promoCode: options.promoCode,
58
- telemetryEnabled: options.telemetryEnabled,
59
+ promoCode,
60
+ telemetryEnabled,
59
61
  });
60
62
 
61
63
  await confirmContinueIfNoOrDirtyGitRepo();
@@ -73,6 +75,7 @@ async function runRemixWizardWithTelemetry(
73
75
  packageName: '@sentry/remix@^8',
74
76
  packageNameDisplayLabel: '@sentry/remix',
75
77
  alreadyInstalled: hasPackageInstalled('@sentry/remix', packageJson),
78
+ forceInstall,
76
79
  });
77
80
 
78
81
  const dsn = selectedProject.keys[0].dsn.public;