v0.1.3 — DeFlock via Overpass + per-source health UI
The cdn.deflock.me CDN is gated behind Cloudflare bot mitigation that mobile HTTP clients can't pass. The live deflock-app Flutter client abandoned that path; it POSTs Overpass-QL queries directly to overpass.deflock.org (with overpass-api.de as a fallback). Verified by hitting the same endpoint from curl — 22 ALPRs returned for the Springfield VA bbox, matching the user's screenshot of the working app. DeflockClient rewrite: - POST [out:json][timeout:25];(node[surveillance][type=ALPR](bbox););out body; - 5 km half-width bbox around the user - 24h on-disk cache keyed by 0.05° grid cell (revisits don't refetch) - Returns sealed FetchResult: Success(points) | Failed(reason) DeflockScanner update: - Replaces 20° tile concept with distance-based refetch (1.5 km threshold) - Records SourceHealth on each fetch outcome Waze: reCAPTCHA gating confirmed. WazeClient.fetchPoliceNear now returns sealed FetchResult; WazeScanner records SourceHealth.FAILED with "Upstream blocked (HTTP 403)" so the user sees why no Waze data is flowing instead of silent zeros. New fusion/SourceHealth.kt — per-source MutableStateFlow registry, record(source, ok, message) + reset() called on service start/stop. UI: SourceRow in the bottom-sheet drill-down now shows the health message in orange when status = FAILED instead of "no detections". versionCode 3 → 4, versionName 0.1.2 → 0.1.3.
This commit is contained in:
@@ -19,8 +19,8 @@ camera, or police presence near you.
|
|||||||
|---|---|---|
|
|---|---|---|
|
||||||
| **BLE** | Bluetooth-LE advertisements: vendor MAC OUIs (Axon, Flock Penguin / Raven, XUNTONG mfg id `0x09C8`, "TN" serial pattern), Raven service UUIDs, device-name patterns | Local radio scan (BLE callback API) |
|
| **BLE** | Bluetooth-LE advertisements: vendor MAC OUIs (Axon, Flock Penguin / Raven, XUNTONG mfg id `0x09C8`, "TN" serial pattern), Raven service UUIDs, device-name patterns | Local radio scan (BLE callback API) |
|
||||||
| **WiFi** | BSSID OUI prefixes for Flock infrastructure (31-prefix superset), `Flock-XXXX` and other generic SSID patterns | `WifiManager.getScanResults()` polled every 35 s (just under the Android 11+ 4-scans/2-min throttle) |
|
| **WiFi** | BSSID OUI prefixes for Flock infrastructure (31-prefix superset), `Flock-XXXX` and other generic SSID patterns | `WifiManager.getScanResults()` polled every 35 s (just under the Android 11+ 4-scans/2-min throttle) |
|
||||||
| **DEFLOCK** | Crowdsourced ALPR locations within configurable proximity (default 200 m) | Public CDN tile fetch from `cdn.deflock.me`, 24h on-disk cache |
|
| **DEFLOCK** | Crowdsourced ALPR locations within configurable proximity (default 200 m) | POST to Overpass API (`overpass.deflock.org` → fallback `overpass-api.de`) for `man_made=surveillance + surveillance:type=ALPR` in a 5 km bbox; 24 h on-disk cache by 0.05° grid cell. Refetches when the user moves > 1.5 km from the last fetch center. |
|
||||||
| **WAZE** | Live `POLICE` reports within configurable proximity (default 500 m) and < 10 min old | `live-map/api/georss` polled every 60 s with a small bbox around the user |
|
| **WAZE** | Live `POLICE` reports within configurable proximity (default 500 m) and < 10 min old | `live-map/api/georss` polled every 60 s with a small bbox around the user. **Note:** Waze added reCAPTCHA gating to this endpoint in 2025/2026; mobile clients now receive HTTP 403. The bottom-sheet drill-down surfaces this as a per-source health indicator. |
|
||||||
|
|
||||||
Every observation is scored 0-100 by `ConfidenceEngine`. The on-screen tier is
|
Every observation is scored 0-100 by `ConfidenceEngine`. The on-screen tier is
|
||||||
the maximum live score across all sources:
|
the maximum live score across all sources:
|
||||||
|
|||||||
@@ -12,8 +12,8 @@ android {
|
|||||||
applicationId = "org.soulstone.overwatch"
|
applicationId = "org.soulstone.overwatch"
|
||||||
minSdk = 26
|
minSdk = 26
|
||||||
targetSdk = 35
|
targetSdk = 35
|
||||||
versionCode = 3
|
versionCode = 4
|
||||||
versionName = "0.1.2"
|
versionName = "0.1.3"
|
||||||
}
|
}
|
||||||
|
|
||||||
buildTypes {
|
buildTypes {
|
||||||
|
|||||||
@@ -0,0 +1,63 @@
|
|||||||
|
package org.soulstone.overwatch.fusion
|
||||||
|
|
||||||
|
import kotlinx.coroutines.flow.MutableStateFlow
|
||||||
|
import kotlinx.coroutines.flow.StateFlow
|
||||||
|
import kotlinx.coroutines.flow.asStateFlow
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Per-source upstream-health registry.
|
||||||
|
*
|
||||||
|
* Network sources (DEFLOCK, WAZE) record OK/FAILED so the UI can distinguish
|
||||||
|
* "scanned, found nothing" from "couldn't reach the data source." BLE/WIFI
|
||||||
|
* are radio-only and don't currently report; they default to UNKNOWN, which
|
||||||
|
* the UI treats the same as OK.
|
||||||
|
*/
|
||||||
|
object SourceHealth {
|
||||||
|
|
||||||
|
enum class Status { UNKNOWN, OK, FAILED }
|
||||||
|
|
||||||
|
data class Health(
|
||||||
|
val status: Status = Status.UNKNOWN,
|
||||||
|
val lastFetchMs: Long = 0L,
|
||||||
|
/** Short reason shown in the UI when status = FAILED. */
|
||||||
|
val message: String? = null
|
||||||
|
)
|
||||||
|
|
||||||
|
private val _ble = MutableStateFlow(Health())
|
||||||
|
private val _wifi = MutableStateFlow(Health())
|
||||||
|
private val _deflock = MutableStateFlow(Health())
|
||||||
|
private val _waze = MutableStateFlow(Health())
|
||||||
|
|
||||||
|
val ble: StateFlow<Health> = _ble.asStateFlow()
|
||||||
|
val wifi: StateFlow<Health> = _wifi.asStateFlow()
|
||||||
|
val deflock: StateFlow<Health> = _deflock.asStateFlow()
|
||||||
|
val waze: StateFlow<Health> = _waze.asStateFlow()
|
||||||
|
|
||||||
|
fun flowFor(source: DetectionSource): StateFlow<Health> = when (source) {
|
||||||
|
DetectionSource.BLE -> ble
|
||||||
|
DetectionSource.WIFI -> wifi
|
||||||
|
DetectionSource.DEFLOCK -> deflock
|
||||||
|
DetectionSource.WAZE -> waze
|
||||||
|
}
|
||||||
|
|
||||||
|
fun record(source: DetectionSource, ok: Boolean, message: String? = null) {
|
||||||
|
val target = when (source) {
|
||||||
|
DetectionSource.BLE -> _ble
|
||||||
|
DetectionSource.WIFI -> _wifi
|
||||||
|
DetectionSource.DEFLOCK -> _deflock
|
||||||
|
DetectionSource.WAZE -> _waze
|
||||||
|
}
|
||||||
|
target.value = Health(
|
||||||
|
status = if (ok) Status.OK else Status.FAILED,
|
||||||
|
lastFetchMs = System.currentTimeMillis(),
|
||||||
|
message = message
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun reset() {
|
||||||
|
_ble.value = Health()
|
||||||
|
_wifi.value = Health()
|
||||||
|
_deflock.value = Health()
|
||||||
|
_waze.value = Health()
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,31 +5,39 @@ import android.util.Log
|
|||||||
import java.io.File
|
import java.io.File
|
||||||
import java.net.HttpURLConnection
|
import java.net.HttpURLConnection
|
||||||
import java.net.URL
|
import java.net.URL
|
||||||
|
import java.net.URLEncoder
|
||||||
|
import kotlin.math.floor
|
||||||
import kotlinx.coroutines.Dispatchers
|
import kotlinx.coroutines.Dispatchers
|
||||||
import kotlinx.coroutines.withContext
|
import kotlinx.coroutines.withContext
|
||||||
import kotlin.math.floor
|
import org.json.JSONObject
|
||||||
import org.json.JSONArray
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fetches DeFlock ALPR tile data from the public CDN, with a 24h on-disk cache.
|
* Fetches DeFlock ALPR data from the Overpass API (matching the live deflock-app
|
||||||
|
* Flutter client). The earlier `cdn.deflock.me/regions/...json` path is now
|
||||||
|
* gated behind Cloudflare bot mitigation that we cannot pass from a mobile HTTP
|
||||||
|
* client.
|
||||||
*
|
*
|
||||||
* Tile scheme (from REFERENCES/deflock/serverless/alpr_cache):
|
* Strategy:
|
||||||
* tile_lat = floor(lat / 20) * 20
|
* - POST an Overpass-QL query for `man_made=surveillance + surveillance:type=ALPR`
|
||||||
* tile_lon = floor(lon / 20) * 20
|
* inside a small bbox around the user.
|
||||||
* url = https://cdn.deflock.me/regions/{tile_lat}/{tile_lon}.json
|
* - Try `overpass.deflock.org` first (less rate-limited for this use case),
|
||||||
* body = JSON array of { id: number, lat: number, lon: number, tags: {…} }
|
* fall back to public `overpass-api.de`.
|
||||||
*
|
* - Cache the JSON response on disk by 0.05° grid cell (24h TTL). Revisits to
|
||||||
* 20° tiles → ≤16 tiles cover the entire globe; one user typically only ever touches one.
|
* the same cell don't re-hit the API.
|
||||||
*/
|
*/
|
||||||
class DeflockClient(context: Context) {
|
class DeflockClient(context: Context) {
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
private const val TAG = "DeflockClient"
|
private const val TAG = "DeflockClient"
|
||||||
private const val TILE_SIZE_DEG = 20
|
private const val FETCH_RADIUS_DEG = 0.05 // ~5.5 km half-width bbox
|
||||||
private const val CACHE_TTL_MS = 24L * 60L * 60L * 1000L
|
private const val CACHE_TTL_MS = 24L * 60L * 60L * 1000L
|
||||||
private const val CDN_BASE = "https://cdn.deflock.me/regions"
|
|
||||||
private const val USER_AGENT = "OVERWATCH/0.1 (+github.com/KaraZajac/OVERWATCH)"
|
private const val USER_AGENT = "OVERWATCH/0.1 (+github.com/KaraZajac/OVERWATCH)"
|
||||||
private const val TIMEOUT_MS = 15_000
|
private const val TIMEOUT_MS = 30_000
|
||||||
|
private const val OVERPASS_QUERY_TIMEOUT_S = 25
|
||||||
|
private val ENDPOINTS = listOf(
|
||||||
|
"https://overpass.deflock.org/api/interpreter",
|
||||||
|
"https://overpass-api.de/api/interpreter"
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
data class AlprPoint(
|
data class AlprPoint(
|
||||||
@@ -40,63 +48,94 @@ class DeflockClient(context: Context) {
|
|||||||
val manufacturer: String? = null
|
val manufacturer: String? = null
|
||||||
)
|
)
|
||||||
|
|
||||||
data class TileKey(val tileLat: Int, val tileLon: Int) {
|
/** Outcome of a fetch — distinguishes "no ALPRs in area" from "couldn't reach the API." */
|
||||||
fun fileName() = "deflock_${tileLat}_${tileLon}.json"
|
sealed class FetchResult {
|
||||||
|
data class Success(val points: List<AlprPoint>) : FetchResult()
|
||||||
|
data class Failed(val reason: String) : FetchResult()
|
||||||
}
|
}
|
||||||
|
|
||||||
private val cacheDir: File = File(context.cacheDir, "deflock").apply { mkdirs() }
|
private val cacheDir: File = File(context.cacheDir, "deflock").apply { mkdirs() }
|
||||||
|
|
||||||
fun tileFor(lat: Double, lon: Double): TileKey = TileKey(
|
suspend fun fetchAround(lat: Double, lon: Double): FetchResult = withContext(Dispatchers.IO) {
|
||||||
tileLat = floor(lat / TILE_SIZE_DEG).toInt() * TILE_SIZE_DEG,
|
val key = cacheKeyFor(lat, lon)
|
||||||
tileLon = floor(lon / TILE_SIZE_DEG).toInt() * TILE_SIZE_DEG
|
val cached = cachedJson(key)
|
||||||
)
|
|
||||||
|
|
||||||
/** Returns parsed ALPR points for the tile; empty list on any failure (logged). */
|
|
||||||
suspend fun fetchTile(tile: TileKey): List<AlprPoint> = withContext(Dispatchers.IO) {
|
|
||||||
val cached = cachedJson(tile)
|
|
||||||
if (cached != null) {
|
if (cached != null) {
|
||||||
return@withContext parseSafely(cached)
|
Log.d(TAG, "Cache hit for $key")
|
||||||
|
return@withContext FetchResult.Success(parseSafely(cached))
|
||||||
|
}
|
||||||
|
val south = lat - FETCH_RADIUS_DEG
|
||||||
|
val north = lat + FETCH_RADIUS_DEG
|
||||||
|
val west = lon - FETCH_RADIUS_DEG
|
||||||
|
val east = lon + FETCH_RADIUS_DEG
|
||||||
|
val query = buildQuery(south, west, north, east)
|
||||||
|
val (body, lastError) = downloadFromAny(query)
|
||||||
|
if (body == null) {
|
||||||
|
return@withContext FetchResult.Failed(lastError ?: "Network error")
|
||||||
}
|
}
|
||||||
val downloaded = downloadTile(tile) ?: return@withContext emptyList()
|
|
||||||
try {
|
try {
|
||||||
File(cacheDir, tile.fileName()).writeText(downloaded)
|
File(cacheDir, "$key.json").writeText(body)
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
Log.w(TAG, "Failed to write tile cache for $tile: ${e.message}")
|
Log.w(TAG, "Failed to write cache for $key: ${e.message}")
|
||||||
}
|
}
|
||||||
parseSafely(downloaded)
|
FetchResult.Success(parseSafely(body))
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun cachedJson(tile: TileKey): String? {
|
private fun cacheKeyFor(lat: Double, lon: Double): String {
|
||||||
val f = File(cacheDir, tile.fileName())
|
// 0.05° grid cell. Two consecutive points within the same cell get the
|
||||||
|
// same cache key, so micro-movements don't refetch.
|
||||||
|
val latStep = floor(lat / FETCH_RADIUS_DEG).toInt()
|
||||||
|
val lonStep = floor(lon / FETCH_RADIUS_DEG).toInt()
|
||||||
|
return "deflock_${latStep}_${lonStep}"
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun cachedJson(key: String): String? {
|
||||||
|
val f = File(cacheDir, "$key.json")
|
||||||
if (!f.exists()) return null
|
if (!f.exists()) return null
|
||||||
val age = System.currentTimeMillis() - f.lastModified()
|
if (System.currentTimeMillis() - f.lastModified() > CACHE_TTL_MS) return null
|
||||||
if (age > CACHE_TTL_MS) return null
|
|
||||||
return try { f.readText() } catch (e: Exception) { null }
|
return try { f.readText() } catch (e: Exception) { null }
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun downloadTile(tile: TileKey): String? {
|
private fun buildQuery(south: Double, west: Double, north: Double, east: Double): String =
|
||||||
val url = URL("$CDN_BASE/${tile.tileLat}/${tile.tileLon}.json")
|
"[out:json][timeout:$OVERPASS_QUERY_TIMEOUT_S];" +
|
||||||
|
"(node[\"man_made\"=\"surveillance\"][\"surveillance:type\"=\"ALPR\"]" +
|
||||||
|
"($south,$west,$north,$east););out body;"
|
||||||
|
|
||||||
|
/** Try each endpoint in order until one returns 2xx. Returns body + last error message. */
|
||||||
|
private fun downloadFromAny(query: String): Pair<String?, String?> {
|
||||||
|
var lastError: String? = null
|
||||||
|
for (endpoint in ENDPOINTS) {
|
||||||
|
val (body, err) = postQuery(endpoint, query)
|
||||||
|
if (body != null) return body to null
|
||||||
|
lastError = err
|
||||||
|
}
|
||||||
|
return null to lastError
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun postQuery(endpoint: String, query: String): Pair<String?, String?> {
|
||||||
|
val url = URL(endpoint)
|
||||||
val conn = (url.openConnection() as HttpURLConnection).apply {
|
val conn = (url.openConnection() as HttpURLConnection).apply {
|
||||||
connectTimeout = TIMEOUT_MS
|
connectTimeout = TIMEOUT_MS
|
||||||
readTimeout = TIMEOUT_MS
|
readTimeout = TIMEOUT_MS
|
||||||
requestMethod = "GET"
|
requestMethod = "POST"
|
||||||
|
doOutput = true
|
||||||
setRequestProperty("User-Agent", USER_AGENT)
|
setRequestProperty("User-Agent", USER_AGENT)
|
||||||
|
setRequestProperty("Content-Type", "application/x-www-form-urlencoded")
|
||||||
setRequestProperty("Accept", "application/json")
|
setRequestProperty("Accept", "application/json")
|
||||||
}
|
}
|
||||||
return try {
|
return try {
|
||||||
|
val payload = "data=" + URLEncoder.encode(query, "UTF-8")
|
||||||
|
conn.outputStream.use { it.write(payload.toByteArray()) }
|
||||||
val code = conn.responseCode
|
val code = conn.responseCode
|
||||||
if (code == 404) {
|
if (code in 200..299) {
|
||||||
Log.i(TAG, "Tile $tile not present on CDN (no ALPRs in this region)")
|
val body = conn.inputStream.bufferedReader().use { it.readText() }
|
||||||
"" // cache the empty result by writing an empty string
|
body to null
|
||||||
} else if (code in 200..299) {
|
|
||||||
conn.inputStream.bufferedReader().use { it.readText() }
|
|
||||||
} else {
|
} else {
|
||||||
Log.w(TAG, "CDN returned $code for $tile")
|
Log.w(TAG, "$endpoint returned $code")
|
||||||
null
|
null to "HTTP $code"
|
||||||
}
|
}
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
Log.w(TAG, "Download failed for $tile: ${e.message}")
|
Log.w(TAG, "$endpoint failed: ${e.message}")
|
||||||
null
|
null to (e.message ?: e.javaClass.simpleName)
|
||||||
} finally {
|
} finally {
|
||||||
conn.disconnect()
|
conn.disconnect()
|
||||||
}
|
}
|
||||||
@@ -105,17 +144,19 @@ class DeflockClient(context: Context) {
|
|||||||
private fun parseSafely(json: String): List<AlprPoint> {
|
private fun parseSafely(json: String): List<AlprPoint> {
|
||||||
if (json.isBlank()) return emptyList()
|
if (json.isBlank()) return emptyList()
|
||||||
return try {
|
return try {
|
||||||
val arr = JSONArray(json)
|
val root = JSONObject(json)
|
||||||
val out = ArrayList<AlprPoint>(arr.length())
|
val elements = root.optJSONArray("elements") ?: return emptyList()
|
||||||
for (i in 0 until arr.length()) {
|
val out = ArrayList<AlprPoint>(elements.length())
|
||||||
val o = arr.getJSONObject(i)
|
for (i in 0 until elements.length()) {
|
||||||
val lat = o.optDouble("lat")
|
val el = elements.optJSONObject(i) ?: continue
|
||||||
val lon = o.optDouble("lon")
|
if (el.optString("type") != "node") continue
|
||||||
|
val lat = el.optDouble("lat")
|
||||||
|
val lon = el.optDouble("lon")
|
||||||
if (lat.isNaN() || lon.isNaN()) continue
|
if (lat.isNaN() || lon.isNaN()) continue
|
||||||
val tags = o.optJSONObject("tags")
|
val tags = el.optJSONObject("tags")
|
||||||
out.add(
|
out.add(
|
||||||
AlprPoint(
|
AlprPoint(
|
||||||
id = o.optLong("id", 0L),
|
id = el.optLong("id", 0L),
|
||||||
lat = lat,
|
lat = lat,
|
||||||
lon = lon,
|
lon = lon,
|
||||||
operator = tags?.optString("operator")?.ifBlank { null }
|
operator = tags?.optString("operator")?.ifBlank { null }
|
||||||
@@ -129,7 +170,7 @@ class DeflockClient(context: Context) {
|
|||||||
}
|
}
|
||||||
out
|
out
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
Log.w(TAG, "Failed to parse tile JSON: ${e.message}")
|
Log.w(TAG, "Failed to parse Overpass response: ${e.message}")
|
||||||
emptyList()
|
emptyList()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,17 +11,16 @@ import org.soulstone.overwatch.fusion.ConfidenceEngine
|
|||||||
import org.soulstone.overwatch.fusion.DetectionEvent
|
import org.soulstone.overwatch.fusion.DetectionEvent
|
||||||
import org.soulstone.overwatch.fusion.DetectionSource
|
import org.soulstone.overwatch.fusion.DetectionSource
|
||||||
import org.soulstone.overwatch.fusion.DetectionStore
|
import org.soulstone.overwatch.fusion.DetectionStore
|
||||||
|
import org.soulstone.overwatch.fusion.SourceHealth
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* DeFlock orchestrator.
|
* DeFlock orchestrator.
|
||||||
*
|
*
|
||||||
* Subscribes to [LocationProvider]; for each new fix, looks up the matching 20° tile
|
* Subscribes to [LocationProvider]; when the user has moved more than
|
||||||
* (loaded from [DeflockClient] cache or downloaded once / 24h) and submits a
|
* [REFETCH_THRESHOLD_M] from the last fetch center (or there is no last
|
||||||
* detection event for every ALPR within [PROXIMITY_M].
|
* center), runs an Overpass query via [DeflockClient] for the surrounding
|
||||||
*
|
* 5-km bbox. For each cached ALPR within [proximityMeters], submits a
|
||||||
* Tile-boundary edge case: at lat ≈ tile_lat or lon ≈ tile_lon ±0.002°, ALPRs across
|
* detection event.
|
||||||
* the boundary won't be visible until the user crosses it. Acceptable for v0.1 — a
|
|
||||||
* 5-tile fetch (current + 4 neighbours) is a polish item.
|
|
||||||
*/
|
*/
|
||||||
class DeflockScanner(
|
class DeflockScanner(
|
||||||
private val store: DetectionStore,
|
private val store: DetectionStore,
|
||||||
@@ -32,10 +31,12 @@ class DeflockScanner(
|
|||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
private const val TAG = "DeflockScanner"
|
private const val TAG = "DeflockScanner"
|
||||||
|
private const val REFETCH_THRESHOLD_M = 1500f
|
||||||
}
|
}
|
||||||
|
|
||||||
private var job: Job? = null
|
private var job: Job? = null
|
||||||
private var lastTile: DeflockClient.TileKey? = null
|
private var lastFetchLat: Double? = null
|
||||||
|
private var lastFetchLon: Double? = null
|
||||||
private var cachedPoints: List<DeflockClient.AlprPoint> = emptyList()
|
private var cachedPoints: List<DeflockClient.AlprPoint> = emptyList()
|
||||||
|
|
||||||
fun start(scope: CoroutineScope): Boolean {
|
fun start(scope: CoroutineScope): Boolean {
|
||||||
@@ -52,17 +53,36 @@ class DeflockScanner(
|
|||||||
fun stop() {
|
fun stop() {
|
||||||
job?.cancel()
|
job?.cancel()
|
||||||
job = null
|
job = null
|
||||||
lastTile = null
|
lastFetchLat = null
|
||||||
|
lastFetchLon = null
|
||||||
cachedPoints = emptyList()
|
cachedPoints = emptyList()
|
||||||
Log.i(TAG, "DeflockScanner stopped")
|
Log.i(TAG, "DeflockScanner stopped")
|
||||||
}
|
}
|
||||||
|
|
||||||
private suspend fun handleFix(fix: Location) {
|
private suspend fun handleFix(fix: Location) {
|
||||||
val tile = client.tileFor(fix.latitude, fix.longitude)
|
if (shouldRefetch(fix)) {
|
||||||
if (tile != lastTile) {
|
when (val result = client.fetchAround(fix.latitude, fix.longitude)) {
|
||||||
cachedPoints = client.fetchTile(tile)
|
is DeflockClient.FetchResult.Success -> {
|
||||||
lastTile = tile
|
cachedPoints = result.points
|
||||||
Log.i(TAG, "Loaded tile $tile with ${cachedPoints.size} ALPRs")
|
lastFetchLat = fix.latitude
|
||||||
|
lastFetchLon = fix.longitude
|
||||||
|
SourceHealth.record(DetectionSource.DEFLOCK, ok = true)
|
||||||
|
Log.i(
|
||||||
|
TAG,
|
||||||
|
"Loaded ${cachedPoints.size} ALPRs around " +
|
||||||
|
"(${fix.latitude}, ${fix.longitude})"
|
||||||
|
)
|
||||||
|
}
|
||||||
|
is DeflockClient.FetchResult.Failed -> {
|
||||||
|
SourceHealth.record(
|
||||||
|
DetectionSource.DEFLOCK,
|
||||||
|
ok = false,
|
||||||
|
message = "Overpass unreachable: ${result.reason}"
|
||||||
|
)
|
||||||
|
Log.w(TAG, "Overpass fetch failed: ${result.reason}")
|
||||||
|
// Keep using cachedPoints (may be empty on first failure).
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (cachedPoints.isEmpty()) return
|
if (cachedPoints.isEmpty()) return
|
||||||
|
|
||||||
@@ -91,4 +111,12 @@ class DeflockScanner(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun shouldRefetch(fix: Location): Boolean {
|
||||||
|
val lat = lastFetchLat ?: return true
|
||||||
|
val lon = lastFetchLon ?: return true
|
||||||
|
val out = FloatArray(1)
|
||||||
|
Location.distanceBetween(lat, lon, fix.latitude, fix.longitude, out)
|
||||||
|
return out[0] > REFETCH_THRESHOLD_M
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -48,7 +48,13 @@ class WazeClient {
|
|||||||
val reportedBy: String?
|
val reportedBy: String?
|
||||||
)
|
)
|
||||||
|
|
||||||
suspend fun fetchPoliceNear(lat: Double, lon: Double): List<Alert> = withContext(Dispatchers.IO) {
|
/** Outcome — distinguishes "no police alerts in area" from "couldn't reach Waze." */
|
||||||
|
sealed class FetchResult {
|
||||||
|
data class Success(val alerts: List<Alert>) : FetchResult()
|
||||||
|
data class Failed(val reason: String) : FetchResult()
|
||||||
|
}
|
||||||
|
|
||||||
|
suspend fun fetchPoliceNear(lat: Double, lon: Double): FetchResult = withContext(Dispatchers.IO) {
|
||||||
val top = lat + BBOX_HALF_DEG
|
val top = lat + BBOX_HALF_DEG
|
||||||
val bottom = lat - BBOX_HALF_DEG
|
val bottom = lat - BBOX_HALF_DEG
|
||||||
val left = lon - BBOX_HALF_DEG
|
val left = lon - BBOX_HALF_DEG
|
||||||
@@ -67,15 +73,22 @@ class WazeClient {
|
|||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
val code = conn.responseCode
|
val code = conn.responseCode
|
||||||
|
if (code == 403) {
|
||||||
|
// Waze added reCAPTCHA gating to live-map in 2025/2026; mobile
|
||||||
|
// clients can no longer hit this endpoint without browser-level
|
||||||
|
// automation. Surface this distinctly so the UI can say so.
|
||||||
|
Log.w(TAG, "Waze returned 403 (upstream reCAPTCHA gating)")
|
||||||
|
return@withContext FetchResult.Failed("Upstream blocked (HTTP 403)")
|
||||||
|
}
|
||||||
if (code !in 200..299) {
|
if (code !in 200..299) {
|
||||||
Log.w(TAG, "Waze returned $code")
|
Log.w(TAG, "Waze returned $code")
|
||||||
return@withContext emptyList()
|
return@withContext FetchResult.Failed("HTTP $code")
|
||||||
}
|
}
|
||||||
val body = conn.inputStream.bufferedReader().use { it.readText() }
|
val body = conn.inputStream.bufferedReader().use { it.readText() }
|
||||||
parsePolice(body)
|
FetchResult.Success(parsePolice(body))
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
Log.w(TAG, "Waze fetch failed: ${e.message}")
|
Log.w(TAG, "Waze fetch failed: ${e.message}")
|
||||||
emptyList()
|
FetchResult.Failed(e.message ?: e.javaClass.simpleName)
|
||||||
} finally {
|
} finally {
|
||||||
conn.disconnect()
|
conn.disconnect()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import org.soulstone.overwatch.fusion.ConfidenceEngine
|
|||||||
import org.soulstone.overwatch.fusion.DetectionEvent
|
import org.soulstone.overwatch.fusion.DetectionEvent
|
||||||
import org.soulstone.overwatch.fusion.DetectionSource
|
import org.soulstone.overwatch.fusion.DetectionSource
|
||||||
import org.soulstone.overwatch.fusion.DetectionStore
|
import org.soulstone.overwatch.fusion.DetectionStore
|
||||||
|
import org.soulstone.overwatch.fusion.SourceHealth
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Polls Waze every 60s for live POLICE alerts in a small bounding box around the
|
* Polls Waze every 60s for live POLICE alerts in a small bounding box around the
|
||||||
@@ -59,7 +60,21 @@ class WazeScanner(
|
|||||||
}
|
}
|
||||||
|
|
||||||
private suspend fun pollOnce(fix: Location) {
|
private suspend fun pollOnce(fix: Location) {
|
||||||
val alerts = client.fetchPoliceNear(fix.latitude, fix.longitude)
|
val result = client.fetchPoliceNear(fix.latitude, fix.longitude)
|
||||||
|
val alerts = when (result) {
|
||||||
|
is WazeClient.FetchResult.Success -> {
|
||||||
|
SourceHealth.record(DetectionSource.WAZE, ok = true)
|
||||||
|
result.alerts
|
||||||
|
}
|
||||||
|
is WazeClient.FetchResult.Failed -> {
|
||||||
|
SourceHealth.record(
|
||||||
|
DetectionSource.WAZE,
|
||||||
|
ok = false,
|
||||||
|
message = "Waze unreachable: ${result.reason}"
|
||||||
|
)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
if (alerts.isEmpty()) return
|
if (alerts.isEmpty()) return
|
||||||
val now = System.currentTimeMillis()
|
val now = System.currentTimeMillis()
|
||||||
val limit = proximityMeters()
|
val limit = proximityMeters()
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ import org.soulstone.overwatch.R
|
|||||||
import org.soulstone.overwatch.data.location.LocationProvider
|
import org.soulstone.overwatch.data.location.LocationProvider
|
||||||
import org.soulstone.overwatch.data.settings.Settings
|
import org.soulstone.overwatch.data.settings.Settings
|
||||||
import org.soulstone.overwatch.fusion.DetectionStore
|
import org.soulstone.overwatch.fusion.DetectionStore
|
||||||
|
import org.soulstone.overwatch.fusion.SourceHealth
|
||||||
import org.soulstone.overwatch.scan.BleScanner
|
import org.soulstone.overwatch.scan.BleScanner
|
||||||
import org.soulstone.overwatch.scan.DeflockClient
|
import org.soulstone.overwatch.scan.DeflockClient
|
||||||
import org.soulstone.overwatch.scan.DeflockScanner
|
import org.soulstone.overwatch.scan.DeflockScanner
|
||||||
@@ -116,6 +117,7 @@ class DetectionService : LifecycleService() {
|
|||||||
|
|
||||||
private fun beginScanning() {
|
private fun beginScanning() {
|
||||||
if (_running.value) return
|
if (_running.value) return
|
||||||
|
SourceHealth.reset()
|
||||||
startInForeground()
|
startInForeground()
|
||||||
if (settings.bleEnabled.value) {
|
if (settings.bleEnabled.value) {
|
||||||
bleStarted = bleScanner.start()
|
bleStarted = bleScanner.start()
|
||||||
@@ -157,6 +159,7 @@ class DetectionService : LifecycleService() {
|
|||||||
if (wazeStarted) { wazeScanner.stop(); wazeStarted = false }
|
if (wazeStarted) { wazeScanner.stop(); wazeStarted = false }
|
||||||
locationProvider.stop()
|
locationProvider.stop()
|
||||||
store.clear()
|
store.clear()
|
||||||
|
SourceHealth.reset()
|
||||||
pruneJob?.cancel()
|
pruneJob?.cancel()
|
||||||
pruneJob = null
|
pruneJob = null
|
||||||
_running.value = false
|
_running.value = false
|
||||||
|
|||||||
@@ -33,6 +33,7 @@ import androidx.compose.material3.ModalBottomSheet
|
|||||||
import androidx.compose.material3.Text
|
import androidx.compose.material3.Text
|
||||||
import androidx.compose.material3.rememberModalBottomSheetState
|
import androidx.compose.material3.rememberModalBottomSheetState
|
||||||
import androidx.compose.runtime.Composable
|
import androidx.compose.runtime.Composable
|
||||||
|
import androidx.compose.runtime.collectAsState
|
||||||
import androidx.compose.runtime.getValue
|
import androidx.compose.runtime.getValue
|
||||||
import androidx.compose.runtime.mutableStateOf
|
import androidx.compose.runtime.mutableStateOf
|
||||||
import androidx.compose.runtime.remember
|
import androidx.compose.runtime.remember
|
||||||
@@ -50,6 +51,7 @@ import androidx.compose.ui.unit.sp
|
|||||||
import kotlinx.coroutines.launch
|
import kotlinx.coroutines.launch
|
||||||
import org.soulstone.overwatch.fusion.DetectionEvent
|
import org.soulstone.overwatch.fusion.DetectionEvent
|
||||||
import org.soulstone.overwatch.fusion.DetectionSource
|
import org.soulstone.overwatch.fusion.DetectionSource
|
||||||
|
import org.soulstone.overwatch.fusion.SourceHealth
|
||||||
import org.soulstone.overwatch.fusion.ThreatLevel
|
import org.soulstone.overwatch.fusion.ThreatLevel
|
||||||
import org.soulstone.overwatch.ui.theme.ThreatColors
|
import org.soulstone.overwatch.ui.theme.ThreatColors
|
||||||
|
|
||||||
@@ -272,6 +274,9 @@ private fun SourcesPanel(events: List<DetectionEvent>) {
|
|||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
private fun SourceRow(source: DetectionSource, events: List<DetectionEvent>) {
|
private fun SourceRow(source: DetectionSource, events: List<DetectionEvent>) {
|
||||||
|
val health by SourceHealth.flowFor(source).collectAsState()
|
||||||
|
val unreachable = health.status == SourceHealth.Status.FAILED
|
||||||
|
|
||||||
Card(
|
Card(
|
||||||
modifier = Modifier.fillMaxWidth(),
|
modifier = Modifier.fillMaxWidth(),
|
||||||
colors = CardDefaults.cardColors(
|
colors = CardDefaults.cardColors(
|
||||||
@@ -294,6 +299,7 @@ private fun SourceRow(source: DetectionSource, events: List<DetectionEvent>) {
|
|||||||
)
|
)
|
||||||
val maxScore = events.maxOfOrNull { it.score } ?: 0
|
val maxScore = events.maxOfOrNull { it.score } ?: 0
|
||||||
val statusColor = when {
|
val statusColor = when {
|
||||||
|
unreachable -> MaterialTheme.colorScheme.onSurfaceVariant
|
||||||
maxScore >= ThreatLevel.RED.minScore -> ThreatColors.Red
|
maxScore >= ThreatLevel.RED.minScore -> ThreatColors.Red
|
||||||
maxScore >= ThreatLevel.ORANGE.minScore -> ThreatColors.Orange
|
maxScore >= ThreatLevel.ORANGE.minScore -> ThreatColors.Orange
|
||||||
maxScore >= ThreatLevel.YELLOW.minScore -> ThreatColors.Yellow
|
maxScore >= ThreatLevel.YELLOW.minScore -> ThreatColors.Yellow
|
||||||
@@ -306,7 +312,15 @@ private fun SourceRow(source: DetectionSource, events: List<DetectionEvent>) {
|
|||||||
.background(statusColor)
|
.background(statusColor)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
if (events.isEmpty()) {
|
if (unreachable) {
|
||||||
|
Spacer(Modifier.height(4.dp))
|
||||||
|
Text(
|
||||||
|
text = health.message ?: "Source unavailable",
|
||||||
|
color = ThreatColors.Orange,
|
||||||
|
fontSize = 11.sp,
|
||||||
|
fontFamily = FontFamily.Monospace
|
||||||
|
)
|
||||||
|
} else if (events.isEmpty()) {
|
||||||
Spacer(Modifier.height(4.dp))
|
Spacer(Modifier.height(4.dp))
|
||||||
Text(
|
Text(
|
||||||
text = "no detections",
|
text = "no detections",
|
||||||
|
|||||||
Reference in New Issue
Block a user