Fix websocket global ID bug
This commit is contained in:
parent
82177253a7
commit
74b260f9a0
4 changed files with 51 additions and 49 deletions
|
@ -1,20 +1,15 @@
|
||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
package="io.heckel.ntfy">
|
package="io.heckel.ntfy">
|
||||||
<!--
|
<!-- Permissions -->
|
||||||
Permissions
|
|
||||||
- INTERNET is needed because we need to talk to the ntfy server(s)
|
|
||||||
- FOREGROUND_SERVICE is needed to support "use another server" feature
|
|
||||||
- WAKE_LOCK & RECEIVE_BOOT_COMPLETED are required to restart the foreground service
|
|
||||||
if it is stopped; see https://robertohuertas.com/2019/06/29/android_foreground_services/
|
|
||||||
-->
|
|
||||||
<uses-permission android:name="android.permission.INTERNET"/>
|
<uses-permission android:name="android.permission.INTERNET"/>
|
||||||
<uses-permission android:name="android.permission.FOREGROUND_SERVICE"/>
|
<uses-permission android:name="android.permission.FOREGROUND_SERVICE"/> <!-- For instant delivery foregrounds service -->
|
||||||
<uses-permission android:name="android.permission.WAKE_LOCK"/>
|
<uses-permission android:name="android.permission.WAKE_LOCK"/> <!-- To keep foreground service awake; soon not needed anymore -->
|
||||||
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
|
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/> <!-- To restart service on reboot -->
|
||||||
<uses-permission android:name="android.permission.VIBRATE"/>
|
<uses-permission android:name="android.permission.VIBRATE"/> <!-- Incoming notifications should be able to vibrate the phone -->
|
||||||
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" android:maxSdkVersion="28"/> <!-- Only required on SDK <= 28 -->
|
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" android:maxSdkVersion="28"/> <!-- Only required on SDK <= 28 -->
|
||||||
<uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES"/> <!-- Required to install packages downloaded through ntfy; craazyy! -->
|
<uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES"/> <!-- To install packages downloaded through ntfy; craazyy! -->
|
||||||
|
<uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM"/> <!-- To reschedule the websocket retry -->
|
||||||
|
|
||||||
<application
|
<application
|
||||||
android:name=".app.Application"
|
android:name=".app.Application"
|
||||||
|
|
|
@ -32,7 +32,7 @@ class ApiService {
|
||||||
val url = topicUrl(baseUrl, topic)
|
val url = topicUrl(baseUrl, topic)
|
||||||
Log.d(TAG, "Publishing to $url")
|
Log.d(TAG, "Publishing to $url")
|
||||||
|
|
||||||
val builder = builder(url, user)
|
val builder = requestBuilder(url, user)
|
||||||
.put(message.toRequestBody())
|
.put(message.toRequestBody())
|
||||||
if (priority in 1..5) {
|
if (priority in 1..5) {
|
||||||
builder.addHeader("X-Priority", priority.toString())
|
builder.addHeader("X-Priority", priority.toString())
|
||||||
|
@ -59,7 +59,7 @@ class ApiService {
|
||||||
val url = topicUrlJsonPoll(baseUrl, topic, sinceVal)
|
val url = topicUrlJsonPoll(baseUrl, topic, sinceVal)
|
||||||
Log.d(TAG, "Polling topic $url")
|
Log.d(TAG, "Polling topic $url")
|
||||||
|
|
||||||
val request = builder(url, user).build()
|
val request = requestBuilder(url, user).build()
|
||||||
client.newCall(request).execute().use { response ->
|
client.newCall(request).execute().use { response ->
|
||||||
if (!response.isSuccessful) {
|
if (!response.isSuccessful) {
|
||||||
throw Exception("Unexpected response ${response.code} when polling topic $url")
|
throw Exception("Unexpected response ${response.code} when polling topic $url")
|
||||||
|
@ -86,7 +86,7 @@ class ApiService {
|
||||||
val sinceVal = if (since == 0L) "all" else since.toString()
|
val sinceVal = if (since == 0L) "all" else since.toString()
|
||||||
val url = topicUrlJson(baseUrl, topics, sinceVal)
|
val url = topicUrlJson(baseUrl, topics, sinceVal)
|
||||||
Log.d(TAG, "Opening subscription connection to $url")
|
Log.d(TAG, "Opening subscription connection to $url")
|
||||||
val request = builder(url, user).build()
|
val request = requestBuilder(url, user).build()
|
||||||
val call = subscriberClient.newCall(request)
|
val call = subscriberClient.newCall(request)
|
||||||
call.enqueue(object : Callback {
|
call.enqueue(object : Callback {
|
||||||
override fun onResponse(call: Call, response: Response) {
|
override fun onResponse(call: Call, response: Response) {
|
||||||
|
@ -122,7 +122,7 @@ class ApiService {
|
||||||
Log.d(TAG, "Checking read access for user ${user.username} against ${topicUrl(baseUrl, topic)}")
|
Log.d(TAG, "Checking read access for user ${user.username} against ${topicUrl(baseUrl, topic)}")
|
||||||
}
|
}
|
||||||
val url = topicUrlAuth(baseUrl, topic)
|
val url = topicUrlAuth(baseUrl, topic)
|
||||||
val request = builder(url, user).build()
|
val request = requestBuilder(url, user).build()
|
||||||
client.newCall(request).execute().use { response ->
|
client.newCall(request).execute().use { response ->
|
||||||
return if (user == null) {
|
return if (user == null) {
|
||||||
response.isSuccessful || response.code == 404 // Treat 404 as success (old server; to be removed in future versions)
|
response.isSuccessful || response.code == 404 // Treat 404 as success (old server; to be removed in future versions)
|
||||||
|
@ -132,16 +132,6 @@ class ApiService {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun builder(url: String, user: User?): Request.Builder {
|
|
||||||
val builder = Request.Builder()
|
|
||||||
.url(url)
|
|
||||||
.addHeader("User-Agent", USER_AGENT)
|
|
||||||
if (user != null) {
|
|
||||||
builder.addHeader("Authorization", Credentials.basic(user.username, user.password, UTF_8))
|
|
||||||
}
|
|
||||||
return builder
|
|
||||||
}
|
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
val USER_AGENT = "ntfy/${BuildConfig.VERSION_NAME} (${BuildConfig.FLAVOR}; Android ${Build.VERSION.RELEASE}; SDK ${Build.VERSION.SDK_INT})"
|
val USER_AGENT = "ntfy/${BuildConfig.VERSION_NAME} (${BuildConfig.FLAVOR}; Android ${Build.VERSION.RELEASE}; SDK ${Build.VERSION.SDK_INT})"
|
||||||
private const val TAG = "NtfyApiService"
|
private const val TAG = "NtfyApiService"
|
||||||
|
@ -151,5 +141,15 @@ class ApiService {
|
||||||
const val EVENT_MESSAGE = "message"
|
const val EVENT_MESSAGE = "message"
|
||||||
const val EVENT_KEEPALIVE = "keepalive"
|
const val EVENT_KEEPALIVE = "keepalive"
|
||||||
const val EVENT_POLL_REQUEST = "poll_request"
|
const val EVENT_POLL_REQUEST = "poll_request"
|
||||||
|
|
||||||
|
fun requestBuilder(url: String, user: User?): Request.Builder {
|
||||||
|
val builder = Request.Builder()
|
||||||
|
.url(url)
|
||||||
|
.addHeader("User-Agent", USER_AGENT)
|
||||||
|
if (user != null) {
|
||||||
|
builder.addHeader("Authorization", Credentials.basic(user.username, user.password, UTF_8))
|
||||||
|
}
|
||||||
|
return builder
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,7 @@ import android.os.Looper
|
||||||
import io.heckel.ntfy.db.*
|
import io.heckel.ntfy.db.*
|
||||||
import io.heckel.ntfy.log.Log
|
import io.heckel.ntfy.log.Log
|
||||||
import io.heckel.ntfy.msg.ApiService
|
import io.heckel.ntfy.msg.ApiService
|
||||||
|
import io.heckel.ntfy.msg.ApiService.Companion.requestBuilder
|
||||||
import io.heckel.ntfy.msg.NotificationParser
|
import io.heckel.ntfy.msg.NotificationParser
|
||||||
import io.heckel.ntfy.util.topicUrl
|
import io.heckel.ntfy.util.topicUrl
|
||||||
import io.heckel.ntfy.util.topicUrlWs
|
import io.heckel.ntfy.util.topicUrlWs
|
||||||
|
@ -44,6 +45,7 @@ class WsConnection(
|
||||||
.build()
|
.build()
|
||||||
private var errorCount = 0
|
private var errorCount = 0
|
||||||
private var webSocket: WebSocket? = null
|
private var webSocket: WebSocket? = null
|
||||||
|
private val webSocketId = AtomicLong(0)
|
||||||
private var state: State? = null
|
private var state: State? = null
|
||||||
private var closed = false
|
private var closed = false
|
||||||
|
|
||||||
|
@ -57,24 +59,18 @@ class WsConnection(
|
||||||
@Synchronized
|
@Synchronized
|
||||||
override fun start() {
|
override fun start() {
|
||||||
if (closed || state == State.Connecting || state == State.Connected) {
|
if (closed || state == State.Connecting || state == State.Connected) {
|
||||||
|
Log.d(TAG,"[$url] WebSocket: Not (re-)starting, because connection is marked closed/connecting/connected")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if (webSocket != null) {
|
if (webSocket != null) {
|
||||||
webSocket!!.close(WS_CLOSE_NORMAL, "")
|
webSocket!!.close(WS_CLOSE_NORMAL, "")
|
||||||
}
|
}
|
||||||
state = State.Connecting
|
state = State.Connecting
|
||||||
val nextId = ID.incrementAndGet()
|
val nextId = webSocketId.incrementAndGet()
|
||||||
val sinceVal = if (since == 0L) "all" else since.toString()
|
val sinceVal = if (since == 0L) "all" else since.toString()
|
||||||
val urlWithSince = topicUrlWs(baseUrl, topicsStr, sinceVal)
|
val urlWithSince = topicUrlWs(baseUrl, topicsStr, sinceVal)
|
||||||
val builder = Request.Builder()
|
val request = requestBuilder(urlWithSince, user).build()
|
||||||
.get()
|
Log.d(TAG, "[$url] WebSocket: Opening $urlWithSince with listener ID $nextId ...")
|
||||||
.url(urlWithSince)
|
|
||||||
.addHeader("User-Agent", ApiService.USER_AGENT)
|
|
||||||
if (user != null) {
|
|
||||||
builder.addHeader("Authorization", Credentials.basic(user.username, user.password, StandardCharsets.UTF_8))
|
|
||||||
}
|
|
||||||
val request = builder.build()
|
|
||||||
Log.d(TAG, "[$url] WebSocket($nextId): opening $urlWithSince ...")
|
|
||||||
webSocket = client.newWebSocket(request, Listener(nextId))
|
webSocket = client.newWebSocket(request, Listener(nextId))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -82,9 +78,10 @@ class WsConnection(
|
||||||
override fun close() {
|
override fun close() {
|
||||||
closed = true
|
closed = true
|
||||||
if (webSocket == null) {
|
if (webSocket == null) {
|
||||||
|
Log.d(TAG,"[$url] WebSocket: Not closing existing connection, because there is no active web socket")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
Log.d(TAG, "[$url] WebSocket(${ID.get()}): closing existing connection")
|
Log.d(TAG, "[$url] WebSocket: Closing existing connection")
|
||||||
state = State.Disconnected
|
state = State.Disconnected
|
||||||
webSocket!!.close(WS_CLOSE_NORMAL, "")
|
webSocket!!.close(WS_CLOSE_NORMAL, "")
|
||||||
webSocket = null
|
webSocket = null
|
||||||
|
@ -98,6 +95,7 @@ class WsConnection(
|
||||||
@Synchronized
|
@Synchronized
|
||||||
fun scheduleReconnect(seconds: Int) {
|
fun scheduleReconnect(seconds: Int) {
|
||||||
if (closed || state == State.Connecting || state == State.Connected) {
|
if (closed || state == State.Connecting || state == State.Connected) {
|
||||||
|
Log.d(TAG,"[$url] WebSocket: Not rescheduling connection, because connection is marked closed/connecting/connected")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
state = State.Scheduled
|
state = State.Scheduled
|
||||||
|
@ -115,8 +113,8 @@ class WsConnection(
|
||||||
|
|
||||||
private inner class Listener(private val id: Long) : WebSocketListener() {
|
private inner class Listener(private val id: Long) : WebSocketListener() {
|
||||||
override fun onOpen(webSocket: WebSocket, response: Response) {
|
override fun onOpen(webSocket: WebSocket, response: Response) {
|
||||||
syncExec {
|
syncExec("onOpen") {
|
||||||
Log.d(TAG, "[$url] WebSocket($id): opened")
|
Log.d(TAG, "[$url] WebSocket ($id): Opened connection")
|
||||||
state = State.Connected
|
state = State.Connected
|
||||||
if (errorCount > 0) {
|
if (errorCount > 0) {
|
||||||
errorCount = 0
|
errorCount = 0
|
||||||
|
@ -126,10 +124,11 @@ class WsConnection(
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onMessage(webSocket: WebSocket, text: String) {
|
override fun onMessage(webSocket: WebSocket, text: String) {
|
||||||
syncExec {
|
syncExec("onMessage") {
|
||||||
Log.d(TAG, "[$url] WebSocket($id): received message: $text")
|
Log.d(TAG, "[$url] WebSocket ($id): Received message: $text")
|
||||||
val notificationWithTopic = parser.parseWithTopic(text, subscriptionId = 0, notificationId = Random.nextInt())
|
val notificationWithTopic = parser.parseWithTopic(text, subscriptionId = 0, notificationId = Random.nextInt())
|
||||||
if (notificationWithTopic == null) {
|
if (notificationWithTopic == null) {
|
||||||
|
Log.d(TAG, "[$url] WebSocket ($id): Unable to parse message. Discarding.")
|
||||||
return@syncExec
|
return@syncExec
|
||||||
}
|
}
|
||||||
val topic = notificationWithTopic.topic
|
val topic = notificationWithTopic.topic
|
||||||
|
@ -143,17 +142,21 @@ class WsConnection(
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onClosed(webSocket: WebSocket, code: Int, reason: String) {
|
override fun onClosed(webSocket: WebSocket, code: Int, reason: String) {
|
||||||
syncExec {
|
syncExec("onClosed") {
|
||||||
Log.w(TAG, "[$url] WebSocket($id): closed")
|
Log.w(TAG, "[$url] WebSocket ($id): Closed connection")
|
||||||
state = State.Disconnected
|
state = State.Disconnected
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onFailure(webSocket: WebSocket, t: Throwable, response: Response?) {
|
override fun onFailure(webSocket: WebSocket, t: Throwable, response: Response?) {
|
||||||
Log.e(TAG, "[$url] WebSocket($id): failure ${response?.code}: ${response?.message}", t)
|
syncExec("onFailure") {
|
||||||
syncExec {
|
if (response == null) {
|
||||||
|
Log.e(TAG, "[$url] WebSocket ($id): Connection failed (response is null): ${t.message}", t)
|
||||||
|
} else {
|
||||||
|
Log.e(TAG, "[$url] WebSocket ($id): Connection failed (response code ${response.code}, message: ${response.message}): ${t.message}", t)
|
||||||
|
}
|
||||||
if (closed) {
|
if (closed) {
|
||||||
Log.d(TAG, "WebSocket($id): Connection marked as closed. Not retrying.")
|
Log.d(TAG, "WebSocket ($id): Connection marked as closed. Not retrying.")
|
||||||
return@syncExec
|
return@syncExec
|
||||||
}
|
}
|
||||||
stateChangeListener(subscriptionIds, ConnectionState.CONNECTING)
|
stateChangeListener(subscriptionIds, ConnectionState.CONNECTING)
|
||||||
|
@ -164,10 +167,14 @@ class WsConnection(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun syncExec(fn: () -> Unit) {
|
private fun syncExec(tag: String, fn: () -> Unit) {
|
||||||
synchronized(this) {
|
synchronized(this) {
|
||||||
if (ID.get() == id) {
|
if (webSocketId.get() == id) {
|
||||||
|
Log.d(TAG, "[$url] WebSocket ($id): Begin $tag")
|
||||||
fn()
|
fn()
|
||||||
|
Log.d(TAG, "[$url] WebSocket ($id): End $tag")
|
||||||
|
} else {
|
||||||
|
Log.d(TAG, "[$url] WebSocket ($id): Skipping synchronized block '$tag', because ID does not match ${webSocketId.get()}")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -182,6 +189,5 @@ class WsConnection(
|
||||||
private const val RECONNECT_TAG = "WsReconnect"
|
private const val RECONNECT_TAG = "WsReconnect"
|
||||||
private const val WS_CLOSE_NORMAL = 1000
|
private const val WS_CLOSE_NORMAL = 1000
|
||||||
private val RETRY_SECONDS = listOf(5, 10, 15, 20, 30, 45, 60, 120)
|
private val RETRY_SECONDS = listOf(5, 10, 15, 20, 30, 45, 60, 120)
|
||||||
private val ID = AtomicLong(0)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -66,6 +66,7 @@ class SettingsActivity : AppCompatActivity(), PreferenceFragmentCompat.OnPrefere
|
||||||
.beginTransaction()
|
.beginTransaction()
|
||||||
.replace(R.id.settings_layout, settingsFragment)
|
.replace(R.id.settings_layout, settingsFragment)
|
||||||
.commit()
|
.commit()
|
||||||
|
title = getString(R.string.settings_title)
|
||||||
} else {
|
} else {
|
||||||
title = savedInstanceState.getCharSequence(TITLE_TAG)
|
title = savedInstanceState.getCharSequence(TITLE_TAG)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue