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.
- package/.env.example +39 -0
- package/README.md +2 -0
- package/docs/capabilities.md +2 -2
- package/docs/configuration.md +13 -5
- package/docs/integrations.md +4 -1
- package/flutter_app/.metadata +42 -0
- package/flutter_app/README.md +21 -0
- package/flutter_app/analysis_options.yaml +32 -0
- package/flutter_app/android/app/build.gradle.kts +109 -0
- package/flutter_app/android/app/src/debug/AndroidManifest.xml +7 -0
- package/flutter_app/android/app/src/main/AndroidManifest.xml +147 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/MainActivity.kt +747 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/HealthConnectGateway.kt +280 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/HealthSyncNotifications.kt +113 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/HealthSyncPayload.kt +57 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/HealthSyncScheduler.kt +78 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/HealthSyncWorker.kt +253 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/PermissionsRationaleActivity.kt +46 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/recording/RecordingBootReceiver.kt +21 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/recording/RecordingForegroundService.kt +586 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/recording/RecordingStateStore.kt +78 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/recording/RecordingUploadClient.kt +104 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/AiHomeWidgetProvider.kt +457 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/AiWidgetStore.kt +194 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/VoiceLaunchWidgetProvider.kt +67 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/WidgetConfigActivity.kt +228 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/WidgetSyncScheduler.kt +72 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/WidgetSyncWorker.kt +186 -0
- package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/widgets/WidgetTaskRunWorker.kt +210 -0
- package/flutter_app/android/app/src/main/res/drawable/launch_background.xml +12 -0
- package/flutter_app/android/app/src/main/res/drawable/neoagent_ai_widget_bg.xml +11 -0
- package/flutter_app/android/app/src/main/res/drawable/neoagent_ai_widget_task_bg.xml +8 -0
- package/flutter_app/android/app/src/main/res/drawable-v21/launch_background.xml +12 -0
- package/flutter_app/android/app/src/main/res/layout/neoagent_ai_widget.xml +138 -0
- package/flutter_app/android/app/src/main/res/layout/neoagent_ai_widget_task_row.xml +52 -0
- package/flutter_app/android/app/src/main/res/layout/neoagent_voice_widget.xml +49 -0
- package/flutter_app/android/app/src/main/res/mipmap-hdpi/ic_launcher.png +0 -0
- package/flutter_app/android/app/src/main/res/mipmap-mdpi/ic_launcher.png +0 -0
- package/flutter_app/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png +0 -0
- package/flutter_app/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png +0 -0
- package/flutter_app/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png +0 -0
- package/flutter_app/android/app/src/main/res/values/strings.xml +12 -0
- package/flutter_app/android/app/src/main/res/values/styles.xml +18 -0
- package/flutter_app/android/app/src/main/res/values-night/styles.xml +18 -0
- package/flutter_app/android/app/src/main/res/xml/file_paths.xml +6 -0
- package/flutter_app/android/app/src/main/res/xml/neoagent_ai_widget_info.xml +12 -0
- package/flutter_app/android/app/src/main/res/xml/neoagent_voice_widget_info.xml +12 -0
- package/flutter_app/android/app/src/profile/AndroidManifest.xml +7 -0
- package/flutter_app/android/build.gradle.kts +24 -0
- package/flutter_app/android/ci-release.keystore +0 -0
- package/flutter_app/android/gradle/wrapper/gradle-wrapper.properties +5 -0
- package/flutter_app/android/gradle.properties +3 -0
- package/flutter_app/android/key.properties +4 -0
- package/flutter_app/android/settings.gradle.kts +26 -0
- package/flutter_app/assets/branding/app_icon_1024.png +0 -0
- package/flutter_app/assets/branding/app_icon_128.png +0 -0
- package/flutter_app/assets/branding/app_icon_192.png +0 -0
- package/flutter_app/assets/branding/app_icon_256.png +0 -0
- package/flutter_app/assets/branding/app_icon_32.png +0 -0
- package/flutter_app/assets/branding/app_icon_512.png +0 -0
- package/flutter_app/assets/branding/app_icon_64.png +0 -0
- package/flutter_app/assets/branding/tray_icon_template.png +0 -0
- package/flutter_app/lib/features/location/location_service.dart +119 -0
- package/flutter_app/lib/features/notifications/notification_interceptor.dart +97 -0
- package/flutter_app/lib/main.dart +23057 -0
- package/flutter_app/lib/main_app_shell.dart +1682 -0
- package/flutter_app/lib/main_integrations.dart +931 -0
- package/flutter_app/lib/main_launcher.dart +959 -0
- package/flutter_app/lib/main_launcher_entry.dart +5 -0
- package/flutter_app/lib/main_models.dart +3473 -0
- package/flutter_app/lib/main_shared.dart +2861 -0
- package/flutter_app/lib/main_theme.dart +204 -0
- package/flutter_app/lib/main_voice_assistant.dart +831 -0
- package/flutter_app/lib/src/android_apk_drop_zone.dart +32 -0
- package/flutter_app/lib/src/android_apk_drop_zone_stub.dart +16 -0
- package/flutter_app/lib/src/android_apk_drop_zone_web.dart +348 -0
- package/flutter_app/lib/src/android_app_installer.dart +22 -0
- package/flutter_app/lib/src/android_app_installer_io.dart +122 -0
- package/flutter_app/lib/src/android_app_installer_stub.dart +21 -0
- package/flutter_app/lib/src/android_launcher_bridge.dart +239 -0
- package/flutter_app/lib/src/app_launch_bridge.dart +29 -0
- package/flutter_app/lib/src/app_release_updater.dart +511 -0
- package/flutter_app/lib/src/backend_client.dart +1833 -0
- package/flutter_app/lib/src/desktop_companion.dart +2 -0
- package/flutter_app/lib/src/desktop_companion_actions.dart +586 -0
- package/flutter_app/lib/src/desktop_companion_io.dart +538 -0
- package/flutter_app/lib/src/desktop_companion_stub.dart +59 -0
- package/flutter_app/lib/src/desktop_native_bridge.dart +91 -0
- package/flutter_app/lib/src/desktop_screen_capture.dart +21 -0
- package/flutter_app/lib/src/desktop_screen_capture_io.dart +142 -0
- package/flutter_app/lib/src/desktop_screen_capture_stub.dart +12 -0
- package/flutter_app/lib/src/diagnostics_logger.dart +119 -0
- package/flutter_app/lib/src/health_bridge.dart +136 -0
- package/flutter_app/lib/src/live_voice_capture.dart +85 -0
- package/flutter_app/lib/src/messaging_access_summary.dart +46 -0
- package/flutter_app/lib/src/network/app_http_client.dart +53 -0
- package/flutter_app/lib/src/network/app_http_client_factory.dart +6 -0
- package/flutter_app/lib/src/network/app_http_client_io.dart +138 -0
- package/flutter_app/lib/src/network/app_http_client_stub.dart +3 -0
- package/flutter_app/lib/src/network/app_http_client_web.dart +94 -0
- package/flutter_app/lib/src/oauth_launcher.dart +33 -0
- package/flutter_app/lib/src/oauth_launcher_io.dart +77 -0
- package/flutter_app/lib/src/oauth_launcher_stub.dart +33 -0
- package/flutter_app/lib/src/oauth_launcher_web.dart +107 -0
- package/flutter_app/lib/src/recording_bridge.dart +232 -0
- package/flutter_app/lib/src/recording_bridge_io.dart +1019 -0
- package/flutter_app/lib/src/recording_bridge_stub.dart +120 -0
- package/flutter_app/lib/src/recording_bridge_web.dart +689 -0
- package/flutter_app/lib/src/recording_payloads.dart +86 -0
- package/flutter_app/lib/src/theme/palette.dart +81 -0
- package/flutter_app/lib/src/widget_bridge.dart +49 -0
- package/flutter_app/linux/CMakeLists.txt +128 -0
- package/flutter_app/linux/flutter/CMakeLists.txt +88 -0
- package/flutter_app/linux/flutter/generated_plugin_registrant.cc +43 -0
- package/flutter_app/linux/flutter/generated_plugin_registrant.h +15 -0
- package/flutter_app/linux/flutter/generated_plugins.cmake +31 -0
- package/flutter_app/linux/runner/CMakeLists.txt +26 -0
- package/flutter_app/linux/runner/main.cc +6 -0
- package/flutter_app/linux/runner/my_application.cc +144 -0
- package/flutter_app/linux/runner/my_application.h +18 -0
- package/flutter_app/linux/runner/resources/app_icon.png +0 -0
- package/flutter_app/macos/Flutter/Flutter-Debug.xcconfig +2 -0
- package/flutter_app/macos/Flutter/Flutter-Release.xcconfig +2 -0
- package/flutter_app/macos/Flutter/GeneratedPluginRegistrant.swift +40 -0
- package/flutter_app/macos/Podfile +42 -0
- package/flutter_app/macos/Podfile.lock +87 -0
- package/flutter_app/macos/Runner/AppDelegate.swift +576 -0
- package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json +68 -0
- package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_1024.png +0 -0
- package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_128.png +0 -0
- package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_16.png +0 -0
- package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_256.png +0 -0
- package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_32.png +0 -0
- package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_512.png +0 -0
- package/flutter_app/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_64.png +0 -0
- package/flutter_app/macos/Runner/Base.lproj/MainMenu.xib +342 -0
- package/flutter_app/macos/Runner/Configs/AppInfo.xcconfig +14 -0
- package/flutter_app/macos/Runner/Configs/Debug.xcconfig +2 -0
- package/flutter_app/macos/Runner/Configs/Release.xcconfig +2 -0
- package/flutter_app/macos/Runner/Configs/Warnings.xcconfig +13 -0
- package/flutter_app/macos/Runner/DebugProfile.entitlements +16 -0
- package/flutter_app/macos/Runner/Info.plist +36 -0
- package/flutter_app/macos/Runner/MainFlutterWindow.swift +19 -0
- package/flutter_app/macos/Runner/Release.entitlements +12 -0
- package/flutter_app/macos/Runner.xcodeproj/project.pbxproj +801 -0
- package/flutter_app/macos/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist +8 -0
- package/flutter_app/macos/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme +99 -0
- package/flutter_app/macos/Runner.xcworkspace/contents.xcworkspacedata +10 -0
- package/flutter_app/macos/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist +8 -0
- package/flutter_app/macos/RunnerTests/RunnerTests.swift +12 -0
- package/flutter_app/patch_strings.py +12 -0
- package/flutter_app/pubspec.lock +1088 -0
- package/flutter_app/pubspec.yaml +53 -0
- package/flutter_app/test/messaging_access_summary_test.dart +22 -0
- package/flutter_app/test/recording_payloads_test.dart +53 -0
- package/flutter_app/third_party/desktop_audio_capture/LICENSE +21 -0
- package/flutter_app/third_party/desktop_audio_capture/README.md +262 -0
- package/flutter_app/third_party/desktop_audio_capture/lib/audio_capture.dart +65 -0
- package/flutter_app/third_party/desktop_audio_capture/lib/config/mic_audio_config.dart +153 -0
- package/flutter_app/third_party/desktop_audio_capture/lib/config/system_adudio_config.dart +110 -0
- package/flutter_app/third_party/desktop_audio_capture/lib/mic/mic_audio_capture.dart +461 -0
- package/flutter_app/third_party/desktop_audio_capture/lib/model/audio_status.dart +91 -0
- package/flutter_app/third_party/desktop_audio_capture/lib/model/decibel_data.dart +106 -0
- package/flutter_app/third_party/desktop_audio_capture/lib/model/input_device_type.dart +219 -0
- package/flutter_app/third_party/desktop_audio_capture/lib/system/system_audio_capture.dart +336 -0
- package/flutter_app/third_party/desktop_audio_capture/linux/CMakeLists.txt +101 -0
- package/flutter_app/third_party/desktop_audio_capture/linux/audio_capture_plugin.cc +692 -0
- package/flutter_app/third_party/desktop_audio_capture/linux/include/audio_capture/audio_capture_plugin.h +35 -0
- package/flutter_app/third_party/desktop_audio_capture/linux/include/audio_capture/mic_capture_plugin.h +36 -0
- package/flutter_app/third_party/desktop_audio_capture/linux/include/desktop_audio_capture/audio_capture_plugin.h +32 -0
- package/flutter_app/third_party/desktop_audio_capture/linux/include/desktop_audio_capture/mic_capture_plugin.h +32 -0
- package/flutter_app/third_party/desktop_audio_capture/linux/mic_capture_plugin.cc +878 -0
- package/flutter_app/third_party/desktop_audio_capture/macos/Classes/AudioCapturePlugin.swift +27 -0
- package/flutter_app/third_party/desktop_audio_capture/macos/Classes/MicCapturePlugin.swift +1172 -0
- package/flutter_app/third_party/desktop_audio_capture/macos/Classes/SystemCapturePlugin.swift +655 -0
- package/flutter_app/third_party/desktop_audio_capture/macos/Resources/PrivacyInfo.xcprivacy +12 -0
- package/flutter_app/third_party/desktop_audio_capture/macos/desktop_audio_capture.podspec +30 -0
- package/flutter_app/third_party/desktop_audio_capture/pubspec.yaml +87 -0
- package/flutter_app/third_party/desktop_audio_capture/windows/CMakeLists.txt +105 -0
- package/flutter_app/third_party/desktop_audio_capture/windows/audio_capture_plugin.cpp +80 -0
- package/flutter_app/third_party/desktop_audio_capture/windows/audio_capture_plugin.h +31 -0
- package/flutter_app/third_party/desktop_audio_capture/windows/audio_capture_plugin_c_api.cpp +12 -0
- package/flutter_app/third_party/desktop_audio_capture/windows/include/audio_capture/audio_capture_plugin_c_api.h +23 -0
- package/flutter_app/third_party/desktop_audio_capture/windows/include/desktop_audio_capture/audio_capture_plugin.h +25 -0
- package/flutter_app/third_party/desktop_audio_capture/windows/mic_capture_plugin.cpp +1117 -0
- package/flutter_app/third_party/desktop_audio_capture/windows/mic_capture_plugin.h +115 -0
- package/flutter_app/third_party/desktop_audio_capture/windows/system_audio_capture_plugin.cpp +777 -0
- package/flutter_app/third_party/desktop_audio_capture/windows/system_audio_capture_plugin.h +87 -0
- package/flutter_app/third_party/flutter_secure_storage_linux/linux/CMakeLists.txt +30 -0
- package/flutter_app/third_party/flutter_secure_storage_linux/linux/flutter_secure_storage_linux_plugin.cc +215 -0
- package/flutter_app/third_party/flutter_secure_storage_linux/linux/include/flutter_secure_storage_linux/flutter_secure_storage_linux_plugin.h +27 -0
- package/flutter_app/third_party/flutter_secure_storage_linux/pubspec.yaml +20 -0
- package/flutter_app/tool/generate_desktop_branding.py +219 -0
- package/flutter_app/web/favicon.png +0 -0
- package/flutter_app/web/favicon.svg +12 -0
- package/flutter_app/web/icons/Icon-192.png +0 -0
- package/flutter_app/web/icons/Icon-512.png +0 -0
- package/flutter_app/web/icons/Icon-maskable-192.png +0 -0
- package/flutter_app/web/icons/Icon-maskable-512.png +0 -0
- package/flutter_app/web/index.html +39 -0
- package/flutter_app/web/manifest.json +35 -0
- package/flutter_app/windows/CMakeLists.txt +108 -0
- package/flutter_app/windows/flutter/CMakeLists.txt +109 -0
- package/flutter_app/windows/flutter/generated_plugin_registrant.cc +47 -0
- package/flutter_app/windows/flutter/generated_plugin_registrant.h +15 -0
- package/flutter_app/windows/flutter/generated_plugins.cmake +35 -0
- package/flutter_app/windows/runner/CMakeLists.txt +41 -0
- package/flutter_app/windows/runner/Runner.rc +121 -0
- package/flutter_app/windows/runner/flutter_window.cpp +533 -0
- package/flutter_app/windows/runner/flutter_window.h +37 -0
- package/flutter_app/windows/runner/main.cpp +53 -0
- package/flutter_app/windows/runner/resource.h +16 -0
- package/flutter_app/windows/runner/resources/app_icon.ico +0 -0
- package/flutter_app/windows/runner/runner.exe.manifest +14 -0
- package/flutter_app/windows/runner/utils.cpp +65 -0
- package/flutter_app/windows/runner/utils.h +19 -0
- package/flutter_app/windows/runner/win32_window.cpp +299 -0
- package/flutter_app/windows/runner/win32_window.h +102 -0
- package/lib/manager.js +231 -7
- package/package.json +3 -1
- package/server/db/database.js +68 -0
- package/server/http/middleware.js +50 -0
- package/server/http/routes.js +3 -1
- package/server/index.js +1 -0
- package/server/public/.last_build_id +1 -1
- package/server/public/assets/NOTICES +61 -0
- package/server/public/assets/fonts/MaterialIcons-Regular.otf +0 -0
- package/server/public/flutter_bootstrap.js +1 -1
- package/server/public/main.dart.js +65262 -64422
- package/server/routes/integrations.js +86 -0
- package/server/routes/memory.js +11 -2
- package/server/routes/screenHistory.js +46 -0
- package/server/routes/triggers.js +81 -0
- package/server/services/ai/models.js +30 -0
- package/server/services/ai/providers/githubCopilot.js +97 -0
- package/server/services/ai/providers/openai.js +2 -1
- package/server/services/ai/providers/openaiCodex.js +31 -0
- package/server/services/ai/settings.js +20 -0
- package/server/services/ai/systemPrompt.js +1 -1
- package/server/services/ai/tools.js +35 -6
- package/server/services/browser/controller.js +47 -3
- package/server/services/desktop/screenRecorder.js +172 -0
- package/server/services/integrations/env.js +5 -0
- package/server/services/integrations/github/common.js +106 -0
- package/server/services/integrations/github/provider.js +499 -0
- package/server/services/integrations/github/repos.js +1124 -0
- package/server/services/integrations/home_assistant/provider.js +306 -26
- package/server/services/integrations/manager.js +63 -7
- package/server/services/integrations/oauth_provider.js +13 -6
- package/server/services/integrations/provider_config_store.js +76 -0
- package/server/services/integrations/registry.js +4 -0
- package/server/services/integrations/trello/provider.js +744 -0
- package/server/services/integrations/whatsapp/provider.js +6 -2
- package/server/services/manager.js +22 -0
- package/server/services/memory/manager.js +39 -2
- package/server/services/skills/base_catalog.js +33 -0
- package/server/services/tasks/adapters/index.js +1 -0
- package/server/services/tasks/adapters/manual.js +12 -0
- package/server/services/tasks/runtime.js +1 -1
- package/server/services/voice/openaiClient.js +4 -1
- package/server/services/voice/providers.js +2 -1
- package/server/services/widgets/service.js +49 -4
- package/server/utils/local_secrets.js +56 -0
- package/server/utils/logger.js +37 -9
package/flutter_app/android/app/src/main/kotlin/com/neoagent/flutter_app/health/HealthSyncWorker.kt
ADDED
|
@@ -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
|
+
}
|