neoagent 2.3.1-beta.2 → 2.3.1-beta.21

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 (264) hide show
  1. package/.env.example +39 -0
  2. package/README.md +2 -0
  3. package/docs/capabilities.md +2 -2
  4. package/docs/configuration.md +13 -5
  5. package/docs/integrations.md +4 -1
  6. package/flutter_app/.metadata +42 -0
  7. package/flutter_app/README.md +21 -0
  8. package/flutter_app/analysis_options.yaml +32 -0
  9. package/flutter_app/android/app/build.gradle.kts +109 -0
  10. package/flutter_app/android/app/src/debug/AndroidManifest.xml +7 -0
  11. package/flutter_app/android/app/src/main/AndroidManifest.xml +147 -0
  12. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/MainActivity.kt +747 -0
  13. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/HealthConnectGateway.kt +280 -0
  14. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/HealthSyncNotifications.kt +113 -0
  15. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/HealthSyncPayload.kt +57 -0
  16. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/HealthSyncScheduler.kt +78 -0
  17. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/HealthSyncWorker.kt +253 -0
  18. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/PermissionsRationaleActivity.kt +46 -0
  19. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/recording/RecordingBootReceiver.kt +21 -0
  20. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/recording/RecordingForegroundService.kt +586 -0
  21. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/recording/RecordingStateStore.kt +78 -0
  22. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/recording/RecordingUploadClient.kt +104 -0
  23. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/AiHomeWidgetProvider.kt +457 -0
  24. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/AiWidgetStore.kt +194 -0
  25. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/VoiceLaunchWidgetProvider.kt +67 -0
  26. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/WidgetConfigActivity.kt +228 -0
  27. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/WidgetSyncScheduler.kt +72 -0
  28. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/WidgetSyncWorker.kt +186 -0
  29. package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/WidgetTaskRunWorker.kt +210 -0
  30. package/flutter_app/android/app/src/main/res/drawable/launch_background.xml +12 -0
  31. package/flutter_app/android/app/src/main/res/drawable/neoagent_ai_widget_bg.xml +11 -0
  32. package/flutter_app/android/app/src/main/res/drawable/neoagent_ai_widget_task_bg.xml +8 -0
  33. package/flutter_app/android/app/src/main/res/drawable-v21/launch_background.xml +12 -0
  34. package/flutter_app/android/app/src/main/res/layout/neoagent_ai_widget.xml +138 -0
  35. package/flutter_app/android/app/src/main/res/layout/neoagent_ai_widget_task_row.xml +52 -0
  36. package/flutter_app/android/app/src/main/res/layout/neoagent_voice_widget.xml +49 -0
  37. package/flutter_app/android/app/src/main/res/mipmap-hdpi/ic_launcher.png +0 -0
  38. package/flutter_app/android/app/src/main/res/mipmap-mdpi/ic_launcher.png +0 -0
  39. package/flutter_app/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png +0 -0
  40. package/flutter_app/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png +0 -0
  41. package/flutter_app/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png +0 -0
  42. package/flutter_app/android/app/src/main/res/values/strings.xml +12 -0
  43. package/flutter_app/android/app/src/main/res/values/styles.xml +18 -0
  44. package/flutter_app/android/app/src/main/res/values-night/styles.xml +18 -0
  45. package/flutter_app/android/app/src/main/res/xml/file_paths.xml +6 -0
  46. package/flutter_app/android/app/src/main/res/xml/neoagent_ai_widget_info.xml +12 -0
  47. package/flutter_app/android/app/src/main/res/xml/neoagent_voice_widget_info.xml +12 -0
  48. package/flutter_app/android/app/src/profile/AndroidManifest.xml +7 -0
  49. package/flutter_app/android/build.gradle.kts +24 -0
  50. package/flutter_app/android/ci-release.keystore +0 -0
  51. package/flutter_app/android/gradle/wrapper/gradle-wrapper.properties +5 -0
  52. package/flutter_app/android/gradle.properties +3 -0
  53. package/flutter_app/android/key.properties +4 -0
  54. package/flutter_app/android/settings.gradle.kts +26 -0
  55. package/flutter_app/assets/branding/app_icon_1024.png +0 -0
  56. package/flutter_app/assets/branding/app_icon_128.png +0 -0
  57. package/flutter_app/assets/branding/app_icon_192.png +0 -0
  58. package/flutter_app/assets/branding/app_icon_256.png +0 -0
  59. package/flutter_app/assets/branding/app_icon_32.png +0 -0
  60. package/flutter_app/assets/branding/app_icon_512.png +0 -0
  61. package/flutter_app/assets/branding/app_icon_64.png +0 -0
  62. package/flutter_app/assets/branding/tray_icon_template.png +0 -0
  63. package/flutter_app/lib/features/location/location_service.dart +119 -0
  64. package/flutter_app/lib/features/notifications/notification_interceptor.dart +97 -0
  65. package/flutter_app/lib/main.dart +23057 -0
  66. package/flutter_app/lib/main_app_shell.dart +1682 -0
  67. package/flutter_app/lib/main_integrations.dart +931 -0
  68. package/flutter_app/lib/main_launcher.dart +959 -0
  69. package/flutter_app/lib/main_launcher_entry.dart +5 -0
  70. package/flutter_app/lib/main_models.dart +3473 -0
  71. package/flutter_app/lib/main_shared.dart +2861 -0
  72. package/flutter_app/lib/main_theme.dart +204 -0
  73. package/flutter_app/lib/main_voice_assistant.dart +831 -0
  74. package/flutter_app/lib/src/android_apk_drop_zone.dart +32 -0
  75. package/flutter_app/lib/src/android_apk_drop_zone_stub.dart +16 -0
  76. package/flutter_app/lib/src/android_apk_drop_zone_web.dart +348 -0
  77. package/flutter_app/lib/src/android_app_installer.dart +22 -0
  78. package/flutter_app/lib/src/android_app_installer_io.dart +122 -0
  79. package/flutter_app/lib/src/android_app_installer_stub.dart +21 -0
  80. package/flutter_app/lib/src/android_launcher_bridge.dart +239 -0
  81. package/flutter_app/lib/src/app_launch_bridge.dart +29 -0
  82. package/flutter_app/lib/src/app_release_updater.dart +511 -0
  83. package/flutter_app/lib/src/backend_client.dart +1833 -0
  84. package/flutter_app/lib/src/desktop_companion.dart +2 -0
  85. package/flutter_app/lib/src/desktop_companion_actions.dart +586 -0
  86. package/flutter_app/lib/src/desktop_companion_io.dart +538 -0
  87. package/flutter_app/lib/src/desktop_companion_stub.dart +59 -0
  88. package/flutter_app/lib/src/desktop_native_bridge.dart +91 -0
  89. package/flutter_app/lib/src/desktop_screen_capture.dart +21 -0
  90. package/flutter_app/lib/src/desktop_screen_capture_io.dart +142 -0
  91. package/flutter_app/lib/src/desktop_screen_capture_stub.dart +12 -0
  92. package/flutter_app/lib/src/diagnostics_logger.dart +119 -0
  93. package/flutter_app/lib/src/health_bridge.dart +136 -0
  94. package/flutter_app/lib/src/live_voice_capture.dart +85 -0
  95. package/flutter_app/lib/src/messaging_access_summary.dart +46 -0
  96. package/flutter_app/lib/src/network/app_http_client.dart +53 -0
  97. package/flutter_app/lib/src/network/app_http_client_factory.dart +6 -0
  98. package/flutter_app/lib/src/network/app_http_client_io.dart +138 -0
  99. package/flutter_app/lib/src/network/app_http_client_stub.dart +3 -0
  100. package/flutter_app/lib/src/network/app_http_client_web.dart +94 -0
  101. package/flutter_app/lib/src/oauth_launcher.dart +33 -0
  102. package/flutter_app/lib/src/oauth_launcher_io.dart +77 -0
  103. package/flutter_app/lib/src/oauth_launcher_stub.dart +33 -0
  104. package/flutter_app/lib/src/oauth_launcher_web.dart +107 -0
  105. package/flutter_app/lib/src/recording_bridge.dart +232 -0
  106. package/flutter_app/lib/src/recording_bridge_io.dart +1019 -0
  107. package/flutter_app/lib/src/recording_bridge_stub.dart +120 -0
  108. package/flutter_app/lib/src/recording_bridge_web.dart +689 -0
  109. package/flutter_app/lib/src/recording_payloads.dart +86 -0
  110. package/flutter_app/lib/src/theme/palette.dart +81 -0
  111. package/flutter_app/lib/src/widget_bridge.dart +49 -0
  112. package/flutter_app/linux/CMakeLists.txt +128 -0
  113. package/flutter_app/linux/flutter/CMakeLists.txt +88 -0
  114. package/flutter_app/linux/flutter/generated_plugin_registrant.cc +43 -0
  115. package/flutter_app/linux/flutter/generated_plugin_registrant.h +15 -0
  116. package/flutter_app/linux/flutter/generated_plugins.cmake +31 -0
  117. package/flutter_app/linux/runner/CMakeLists.txt +26 -0
  118. package/flutter_app/linux/runner/main.cc +6 -0
  119. package/flutter_app/linux/runner/my_application.cc +144 -0
  120. package/flutter_app/linux/runner/my_application.h +18 -0
  121. package/flutter_app/linux/runner/resources/app_icon.png +0 -0
  122. package/flutter_app/macos/Flutter/Flutter-Debug.xcconfig +2 -0
  123. package/flutter_app/macos/Flutter/Flutter-Release.xcconfig +2 -0
  124. package/flutter_app/macos/Flutter/GeneratedPluginRegistrant.swift +40 -0
  125. package/flutter_app/macos/Podfile +42 -0
  126. package/flutter_app/macos/Podfile.lock +87 -0
  127. package/flutter_app/macos/Runner/AppDelegate.swift +576 -0
  128. package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json +68 -0
  129. package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_1024.png +0 -0
  130. package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_128.png +0 -0
  131. package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_16.png +0 -0
  132. package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_256.png +0 -0
  133. package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_32.png +0 -0
  134. package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_512.png +0 -0
  135. package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_64.png +0 -0
  136. package/flutter_app/macos/Runner/Base.lproj/MainMenu.xib +342 -0
  137. package/flutter_app/macos/Runner/Configs/AppInfo.xcconfig +14 -0
  138. package/flutter_app/macos/Runner/Configs/Debug.xcconfig +2 -0
  139. package/flutter_app/macos/Runner/Configs/Release.xcconfig +2 -0
  140. package/flutter_app/macos/Runner/Configs/Warnings.xcconfig +13 -0
  141. package/flutter_app/macos/Runner/DebugProfile.entitlements +16 -0
  142. package/flutter_app/macos/Runner/Info.plist +36 -0
  143. package/flutter_app/macos/Runner/MainFlutterWindow.swift +19 -0
  144. package/flutter_app/macos/Runner/Release.entitlements +12 -0
  145. package/flutter_app/macos/Runner.xcodeproj/project.pbxproj +801 -0
  146. package/flutter_app/macos/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist +8 -0
  147. package/flutter_app/macos/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme +99 -0
  148. package/flutter_app/macos/Runner.xcworkspace/contents.xcworkspacedata +10 -0
  149. package/flutter_app/macos/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist +8 -0
  150. package/flutter_app/macos/RunnerTests/RunnerTests.swift +12 -0
  151. package/flutter_app/patch_strings.py +12 -0
  152. package/flutter_app/pubspec.lock +1088 -0
  153. package/flutter_app/pubspec.yaml +53 -0
  154. package/flutter_app/test/messaging_access_summary_test.dart +22 -0
  155. package/flutter_app/test/recording_payloads_test.dart +53 -0
  156. package/flutter_app/third_party/desktop_audio_capture/LICENSE +21 -0
  157. package/flutter_app/third_party/desktop_audio_capture/README.md +262 -0
  158. package/flutter_app/third_party/desktop_audio_capture/lib/audio_capture.dart +65 -0
  159. package/flutter_app/third_party/desktop_audio_capture/lib/config/mic_audio_config.dart +153 -0
  160. package/flutter_app/third_party/desktop_audio_capture/lib/config/system_adudio_config.dart +110 -0
  161. package/flutter_app/third_party/desktop_audio_capture/lib/mic/mic_audio_capture.dart +461 -0
  162. package/flutter_app/third_party/desktop_audio_capture/lib/model/audio_status.dart +91 -0
  163. package/flutter_app/third_party/desktop_audio_capture/lib/model/decibel_data.dart +106 -0
  164. package/flutter_app/third_party/desktop_audio_capture/lib/model/input_device_type.dart +219 -0
  165. package/flutter_app/third_party/desktop_audio_capture/lib/system/system_audio_capture.dart +336 -0
  166. package/flutter_app/third_party/desktop_audio_capture/linux/CMakeLists.txt +101 -0
  167. package/flutter_app/third_party/desktop_audio_capture/linux/audio_capture_plugin.cc +692 -0
  168. package/flutter_app/third_party/desktop_audio_capture/linux/include/audio_capture/audio_capture_plugin.h +35 -0
  169. package/flutter_app/third_party/desktop_audio_capture/linux/include/audio_capture/mic_capture_plugin.h +36 -0
  170. package/flutter_app/third_party/desktop_audio_capture/linux/include/desktop_audio_capture/audio_capture_plugin.h +32 -0
  171. package/flutter_app/third_party/desktop_audio_capture/linux/include/desktop_audio_capture/mic_capture_plugin.h +32 -0
  172. package/flutter_app/third_party/desktop_audio_capture/linux/mic_capture_plugin.cc +878 -0
  173. package/flutter_app/third_party/desktop_audio_capture/macos/Classes/AudioCapturePlugin.swift +27 -0
  174. package/flutter_app/third_party/desktop_audio_capture/macos/Classes/MicCapturePlugin.swift +1172 -0
  175. package/flutter_app/third_party/desktop_audio_capture/macos/Classes/SystemCapturePlugin.swift +655 -0
  176. package/flutter_app/third_party/desktop_audio_capture/macos/Resources/PrivacyInfo.xcprivacy +12 -0
  177. package/flutter_app/third_party/desktop_audio_capture/macos/desktop_audio_capture.podspec +30 -0
  178. package/flutter_app/third_party/desktop_audio_capture/pubspec.yaml +87 -0
  179. package/flutter_app/third_party/desktop_audio_capture/windows/CMakeLists.txt +105 -0
  180. package/flutter_app/third_party/desktop_audio_capture/windows/audio_capture_plugin.cpp +80 -0
  181. package/flutter_app/third_party/desktop_audio_capture/windows/audio_capture_plugin.h +31 -0
  182. package/flutter_app/third_party/desktop_audio_capture/windows/audio_capture_plugin_c_api.cpp +12 -0
  183. package/flutter_app/third_party/desktop_audio_capture/windows/include/audio_capture/audio_capture_plugin_c_api.h +23 -0
  184. package/flutter_app/third_party/desktop_audio_capture/windows/include/desktop_audio_capture/audio_capture_plugin.h +25 -0
  185. package/flutter_app/third_party/desktop_audio_capture/windows/mic_capture_plugin.cpp +1117 -0
  186. package/flutter_app/third_party/desktop_audio_capture/windows/mic_capture_plugin.h +115 -0
  187. package/flutter_app/third_party/desktop_audio_capture/windows/system_audio_capture_plugin.cpp +777 -0
  188. package/flutter_app/third_party/desktop_audio_capture/windows/system_audio_capture_plugin.h +87 -0
  189. package/flutter_app/third_party/flutter_secure_storage_linux/linux/CMakeLists.txt +30 -0
  190. package/flutter_app/third_party/flutter_secure_storage_linux/linux/flutter_secure_storage_linux_plugin.cc +215 -0
  191. package/flutter_app/third_party/flutter_secure_storage_linux/linux/include/flutter_secure_storage_linux/flutter_secure_storage_linux_plugin.h +27 -0
  192. package/flutter_app/third_party/flutter_secure_storage_linux/pubspec.yaml +20 -0
  193. package/flutter_app/tool/generate_desktop_branding.py +219 -0
  194. package/flutter_app/web/favicon.png +0 -0
  195. package/flutter_app/web/favicon.svg +12 -0
  196. package/flutter_app/web/icons/Icon-192.png +0 -0
  197. package/flutter_app/web/icons/Icon-512.png +0 -0
  198. package/flutter_app/web/icons/Icon-maskable-192.png +0 -0
  199. package/flutter_app/web/icons/Icon-maskable-512.png +0 -0
  200. package/flutter_app/web/index.html +39 -0
  201. package/flutter_app/web/manifest.json +35 -0
  202. package/flutter_app/windows/CMakeLists.txt +108 -0
  203. package/flutter_app/windows/flutter/CMakeLists.txt +109 -0
  204. package/flutter_app/windows/flutter/generated_plugin_registrant.cc +47 -0
  205. package/flutter_app/windows/flutter/generated_plugin_registrant.h +15 -0
  206. package/flutter_app/windows/flutter/generated_plugins.cmake +35 -0
  207. package/flutter_app/windows/runner/CMakeLists.txt +41 -0
  208. package/flutter_app/windows/runner/Runner.rc +121 -0
  209. package/flutter_app/windows/runner/flutter_window.cpp +533 -0
  210. package/flutter_app/windows/runner/flutter_window.h +37 -0
  211. package/flutter_app/windows/runner/main.cpp +53 -0
  212. package/flutter_app/windows/runner/resource.h +16 -0
  213. package/flutter_app/windows/runner/resources/app_icon.ico +0 -0
  214. package/flutter_app/windows/runner/runner.exe.manifest +14 -0
  215. package/flutter_app/windows/runner/utils.cpp +65 -0
  216. package/flutter_app/windows/runner/utils.h +19 -0
  217. package/flutter_app/windows/runner/win32_window.cpp +299 -0
  218. package/flutter_app/windows/runner/win32_window.h +102 -0
  219. package/lib/manager.js +231 -7
  220. package/package.json +3 -1
  221. package/server/db/database.js +68 -0
  222. package/server/http/middleware.js +50 -0
  223. package/server/http/routes.js +3 -1
  224. package/server/index.js +1 -0
  225. package/server/public/.last_build_id +1 -1
  226. package/server/public/assets/NOTICES +61 -0
  227. package/server/public/assets/fonts/MaterialIcons-Regular.otf +0 -0
  228. package/server/public/flutter_bootstrap.js +1 -1
  229. package/server/public/main.dart.js +65262 -64422
  230. package/server/routes/integrations.js +86 -0
  231. package/server/routes/memory.js +11 -2
  232. package/server/routes/screenHistory.js +46 -0
  233. package/server/routes/triggers.js +81 -0
  234. package/server/services/ai/models.js +30 -0
  235. package/server/services/ai/providers/githubCopilot.js +97 -0
  236. package/server/services/ai/providers/openai.js +2 -1
  237. package/server/services/ai/providers/openaiCodex.js +31 -0
  238. package/server/services/ai/settings.js +20 -0
  239. package/server/services/ai/systemPrompt.js +1 -1
  240. package/server/services/ai/tools.js +35 -6
  241. package/server/services/browser/controller.js +47 -3
  242. package/server/services/desktop/screenRecorder.js +172 -0
  243. package/server/services/integrations/env.js +5 -0
  244. package/server/services/integrations/github/common.js +106 -0
  245. package/server/services/integrations/github/provider.js +499 -0
  246. package/server/services/integrations/github/repos.js +1124 -0
  247. package/server/services/integrations/home_assistant/provider.js +306 -26
  248. package/server/services/integrations/manager.js +63 -7
  249. package/server/services/integrations/oauth_provider.js +13 -6
  250. package/server/services/integrations/provider_config_store.js +76 -0
  251. package/server/services/integrations/registry.js +4 -0
  252. package/server/services/integrations/trello/provider.js +744 -0
  253. package/server/services/integrations/whatsapp/provider.js +6 -2
  254. package/server/services/manager.js +22 -0
  255. package/server/services/memory/manager.js +39 -2
  256. package/server/services/skills/base_catalog.js +33 -0
  257. package/server/services/tasks/adapters/index.js +1 -0
  258. package/server/services/tasks/adapters/manual.js +12 -0
  259. package/server/services/tasks/runtime.js +1 -1
  260. package/server/services/voice/openaiClient.js +4 -1
  261. package/server/services/voice/providers.js +2 -1
  262. package/server/services/widgets/service.js +49 -4
  263. package/server/utils/local_secrets.js +56 -0
  264. package/server/utils/logger.js +37 -9
@@ -0,0 +1,253 @@
1
+ package com.neoagent.flutter_app.health
2
+
3
+ import android.content.Context
4
+ import androidx.work.CoroutineWorker
5
+ import androidx.work.WorkerParameters
6
+ import org.json.JSONObject
7
+ import java.io.IOException
8
+ import java.net.HttpURLConnection
9
+ import java.net.URI
10
+ import java.net.URL
11
+ import java.time.Instant
12
+
13
+ class HealthSyncWorker(
14
+ appContext: Context,
15
+ params: WorkerParameters,
16
+ ) : CoroutineWorker(appContext, params) {
17
+
18
+ override suspend fun doWork(): Result {
19
+ val prefs = HealthSyncPrefs.read(applicationContext)
20
+ val enabled = prefs.getBoolean(HealthSyncPrefs.KEY_ENABLED, false)
21
+ val backendUrl =
22
+ prefs.getString(HealthSyncPrefs.KEY_BACKEND_URL, "")?.trim().orEmpty()
23
+ if (!enabled || backendUrl.isBlank()) {
24
+ return Result.success()
25
+ }
26
+
27
+ HealthSyncNotifications.register(applicationContext)
28
+ setForeground(
29
+ HealthSyncNotifications.foregroundInfo(
30
+ applicationContext,
31
+ "Checking Health Connect permissions…",
32
+ ),
33
+ )
34
+
35
+ return try {
36
+ val gateway = HealthConnectGateway(applicationContext)
37
+ val client = gateway.getClientOrNull()
38
+ ?: return finishFailure(
39
+ retry = false,
40
+ message = "Health Connect is unavailable on this device.",
41
+ )
42
+ val requiredPermissions = gateway.getRequestedPermissions(client)
43
+ val grantedPermissions = client.permissionController.getGrantedPermissions()
44
+ if (!grantedPermissions.containsAll(requiredPermissions)) {
45
+ return finishFailure(
46
+ retry = false,
47
+ message = "Health permissions are missing. Open NeoAgent and grant Health Connect access.",
48
+ )
49
+ }
50
+
51
+ var cookie =
52
+ prefs.getString(HealthSyncPrefs.KEY_SESSION_COOKIE, "")?.trim().orEmpty()
53
+ cookie = ensureSessionCookie(backendUrl, cookie)
54
+ ?: return finishFailure(
55
+ retry = true,
56
+ message = "NeoAgent background sync could not refresh your session.",
57
+ )
58
+
59
+ var statusResponse = request(
60
+ method = "GET",
61
+ baseUrl = backendUrl,
62
+ path = "/api/mobile/health/status",
63
+ cookie = cookie,
64
+ )
65
+
66
+ if (statusResponse.code == HttpURLConnection.HTTP_UNAUTHORIZED) {
67
+ cookie = ensureSessionCookie(backendUrl, "")
68
+ ?: return finishFailure(
69
+ retry = true,
70
+ message = "NeoAgent background sync could not refresh your session.",
71
+ )
72
+ statusResponse = request(
73
+ method = "GET",
74
+ baseUrl = backendUrl,
75
+ path = "/api/mobile/health/status",
76
+ cookie = cookie,
77
+ )
78
+ }
79
+
80
+ if (statusResponse.code !in 200..299) {
81
+ return finishFailure(
82
+ retry = statusResponse.code >= 500,
83
+ message =
84
+ "NeoAgent server rejected health status check (${statusResponse.code}).",
85
+ )
86
+ }
87
+
88
+ val lastWindowEndRaw = JSONObject(statusResponse.body)
89
+ .optJSONObject("lastRun")
90
+ ?.optString("sync_window_end")
91
+ ?.takeIf { it.isNotBlank() }
92
+
93
+ val windowEnd = Instant.now()
94
+ val windowStart = lastWindowEndRaw?.let {
95
+ runCatching { Instant.parse(it) }.getOrNull()
96
+ }?.minusSeconds(300) ?: windowEnd.minusSeconds(24 * 60 * 60)
97
+
98
+ setForeground(
99
+ HealthSyncNotifications.foregroundInfo(
100
+ applicationContext,
101
+ "Syncing steps, heart rate, sleep, exercise, and weight…",
102
+ ),
103
+ )
104
+ val payload = gateway.collectBatch(client, windowStart, windowEnd)
105
+ val uploadResponse = request(
106
+ method = "POST",
107
+ baseUrl = backendUrl,
108
+ path = "/api/mobile/health/sync",
109
+ cookie = cookie,
110
+ jsonBody = payload.toJson().toString(),
111
+ )
112
+
113
+ if (uploadResponse.code !in 200..299) {
114
+ return finishFailure(
115
+ retry = uploadResponse.code >= 500,
116
+ message =
117
+ "NeoAgent server rejected health upload (${uploadResponse.code}).",
118
+ )
119
+ }
120
+
121
+ prefs.edit()
122
+ .putString(HealthSyncPrefs.KEY_SESSION_COOKIE, cookie)
123
+ .putString(HealthSyncPrefs.KEY_LAST_SUCCESS_AT, Instant.now().toString())
124
+ .putInt(HealthSyncPrefs.KEY_CONSECUTIVE_FAILURES, 0)
125
+ .remove(HealthSyncPrefs.KEY_LAST_ERROR)
126
+ .apply()
127
+ HealthSyncNotifications.clearFailure(applicationContext)
128
+ Result.success()
129
+ } catch (err: IOException) {
130
+ finishFailure(
131
+ retry = true,
132
+ message = err.message ?: "Network error while syncing health data.",
133
+ )
134
+ } catch (err: Exception) {
135
+ finishFailure(
136
+ retry = false,
137
+ message = err.message ?: err.javaClass.simpleName,
138
+ )
139
+ }
140
+ }
141
+
142
+ private fun finishFailure(retry: Boolean, message: String): Result {
143
+ val prefs = HealthSyncPrefs.read(applicationContext)
144
+ val failures =
145
+ prefs.getInt(HealthSyncPrefs.KEY_CONSECUTIVE_FAILURES, 0) + 1
146
+ prefs.edit()
147
+ .putInt(HealthSyncPrefs.KEY_CONSECUTIVE_FAILURES, failures)
148
+ .putString(HealthSyncPrefs.KEY_LAST_ERROR, message)
149
+ .apply()
150
+
151
+ if (!retry || failures >= 2) {
152
+ HealthSyncNotifications.showFailure(applicationContext, message)
153
+ }
154
+
155
+ return if (retry) Result.retry() else Result.failure()
156
+ }
157
+
158
+ private fun ensureSessionCookie(
159
+ backendUrl: String,
160
+ currentCookie: String,
161
+ ): String? {
162
+ if (currentCookie.isNotBlank()) {
163
+ return currentCookie
164
+ }
165
+
166
+ val flutterPrefs =
167
+ applicationContext.getSharedPreferences(
168
+ "FlutterSharedPreferences",
169
+ Context.MODE_PRIVATE,
170
+ )
171
+ val username = flutterPrefs.getString("flutter.username", "")?.trim().orEmpty()
172
+ val password = flutterPrefs.getString("flutter.password", "")?.trim().orEmpty()
173
+ if (username.isBlank() || password.isBlank()) {
174
+ return null
175
+ }
176
+
177
+ val response = request(
178
+ method = "POST",
179
+ baseUrl = backendUrl,
180
+ path = "/api/auth/login",
181
+ jsonBody =
182
+ JSONObject()
183
+ .put("username", username)
184
+ .put("password", password)
185
+ .toString(),
186
+ )
187
+ if (response.code !in 200..299 || response.cookie.isNullOrBlank()) {
188
+ return null
189
+ }
190
+
191
+ val cookie = response.cookie.substringBefore(";")
192
+ HealthSyncPrefs.read(applicationContext)
193
+ .edit()
194
+ .putString(HealthSyncPrefs.KEY_SESSION_COOKIE, cookie)
195
+ .apply()
196
+ return cookie
197
+ }
198
+
199
+ private fun request(
200
+ method: String,
201
+ baseUrl: String,
202
+ path: String,
203
+ cookie: String? = null,
204
+ jsonBody: String? = null,
205
+ ): HttpResponse {
206
+ val url = resolveUrl(baseUrl, path)
207
+ val connection = (url.openConnection() as HttpURLConnection).apply {
208
+ requestMethod = method
209
+ connectTimeout = 15_000
210
+ readTimeout = 20_000
211
+ doInput = true
212
+ instanceFollowRedirects = false
213
+ setRequestProperty("Accept", "application/json")
214
+ if (!cookie.isNullOrBlank()) {
215
+ setRequestProperty("Cookie", cookie)
216
+ }
217
+ if (jsonBody != null) {
218
+ doOutput = true
219
+ setRequestProperty("Content-Type", "application/json")
220
+ outputStream.use { stream ->
221
+ stream.write(jsonBody.toByteArray(Charsets.UTF_8))
222
+ }
223
+ }
224
+ }
225
+
226
+ return try {
227
+ val code = connection.responseCode
228
+ val body = (if (code in 200..299) connection.inputStream else connection.errorStream)
229
+ ?.bufferedReader()
230
+ ?.use { it.readText() }
231
+ .orEmpty()
232
+ HttpResponse(
233
+ code = code,
234
+ body = body,
235
+ cookie = connection.getHeaderField("Set-Cookie"),
236
+ )
237
+ } finally {
238
+ connection.disconnect()
239
+ }
240
+ }
241
+
242
+ private fun resolveUrl(baseUrl: String, path: String): URL {
243
+ return URI(baseUrl.trim().ifBlank { "http://localhost:3333" })
244
+ .resolve(path)
245
+ .toURL()
246
+ }
247
+
248
+ private data class HttpResponse(
249
+ val code: Int,
250
+ val body: String,
251
+ val cookie: String? = null,
252
+ )
253
+ }
@@ -0,0 +1,46 @@
1
+ package com.neoagent.flutter_app.health
2
+
3
+ import android.app.Activity
4
+ import android.graphics.Color
5
+ import android.os.Bundle
6
+ import android.widget.ScrollView
7
+ import android.widget.TextView
8
+
9
+ class PermissionsRationaleActivity : Activity() {
10
+
11
+ override fun onCreate(savedInstanceState: Bundle?) {
12
+ super.onCreate(savedInstanceState)
13
+
14
+ val content = TextView(this).apply {
15
+ setBackgroundColor(Color.parseColor("#081317"))
16
+ setTextColor(Color.parseColor("#F4F0E8"))
17
+ textSize = 16f
18
+ setPadding(40, 48, 40, 48)
19
+ text = """
20
+ NeoAgent Health Sync
21
+
22
+ This Flutter app reads only the Health Connect data you explicitly grant and uploads it to your NeoAgent backend.
23
+
24
+ Data types:
25
+ - steps
26
+ - heart rate
27
+ - sleep sessions
28
+ - exercise sessions
29
+ - weight
30
+
31
+ Purpose:
32
+ - keep your NeoAgent server aware of recent health context
33
+ - let the app sync that context on demand without the old notification workflow
34
+
35
+ You can revoke permissions at any time in Android's Health Connect settings.
36
+ """.trimIndent()
37
+ }
38
+
39
+ setContentView(
40
+ ScrollView(this).apply {
41
+ setBackgroundColor(Color.parseColor("#081317"))
42
+ addView(content)
43
+ },
44
+ )
45
+ }
46
+ }
@@ -0,0 +1,21 @@
1
+ package com.neoagent.flutter_app.recording
2
+
3
+ import android.content.BroadcastReceiver
4
+ import android.content.Context
5
+ import android.content.Intent
6
+ import androidx.core.content.ContextCompat
7
+
8
+ class RecordingBootReceiver : BroadcastReceiver() {
9
+ override fun onReceive(context: Context, intent: Intent?) {
10
+ if (intent?.action != Intent.ACTION_BOOT_COMPLETED) {
11
+ return
12
+ }
13
+ val store = RecordingStateStore(context)
14
+ val config = store.loadConfig() ?: return
15
+ if (!config.active || config.paused) {
16
+ return
17
+ }
18
+ val serviceIntent = RecordingForegroundService.buildRestoreIntent(context)
19
+ ContextCompat.startForegroundService(context, serviceIntent)
20
+ }
21
+ }