-
-
Notifications
You must be signed in to change notification settings - Fork 51
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Convert Discovery class to object - Simplify cross-thread handling with dedicated executor - Fixed simultaneous resolution of multiple services - Acquire multicast lock to improve reliability
- Loading branch information
Showing
4 changed files
with
236 additions
and
115 deletions.
There are no files selected for viewing
112 changes: 112 additions & 0 deletions
112
app/src/androidTest/java/com/gaurav/avnc/viewmodel/service/DiscoveryTest.kt
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,112 @@ | ||
/* | ||
* Copyright (c) 2024 Gaurav Ujjwal. | ||
* | ||
* SPDX-License-Identifier: GPL-3.0-or-later | ||
* | ||
* See COPYING.txt for more details. | ||
*/ | ||
|
||
package com.gaurav.avnc.viewmodel.service | ||
|
||
import android.net.nsd.NsdManager | ||
import android.net.nsd.NsdServiceInfo | ||
import android.util.Log | ||
import androidx.core.content.ContextCompat | ||
import com.gaurav.avnc.pollingAssert | ||
import com.gaurav.avnc.runOnMainSync | ||
import com.gaurav.avnc.targetContext | ||
import org.junit.After | ||
import org.junit.Assert.assertTrue | ||
import org.junit.Assume | ||
import org.junit.Before | ||
import org.junit.Test | ||
|
||
class DiscoveryTest { | ||
private val TAG = "DiscoveryTest" | ||
private var nsdManager: NsdManager? = null | ||
private var listeners = mutableListOf<NsdManager.RegistrationListener>() | ||
|
||
/** | ||
* If advertisement fails, this method will cause the calling test to be skipped. | ||
* Advertisement can fail if there is no suitable network on the device | ||
* (e.g. in Airplane mode on newer Android versions). | ||
*/ | ||
private fun advertiseService(advertisedName: String, advertisedPort: Int) { | ||
var registeredService = false | ||
val listener = object : NsdManager.RegistrationListener { | ||
override fun onRegistrationFailed(serviceInfo: NsdServiceInfo?, errorCode: Int) { | ||
Log.e(TAG, "Registration failed: si: $serviceInfo, error:$errorCode") | ||
} | ||
|
||
override fun onUnregistrationFailed(serviceInfo: NsdServiceInfo?, errorCode: Int) {} | ||
override fun onServiceRegistered(serviceInfo: NsdServiceInfo?) { | ||
Log.d(TAG, "Registered si: $serviceInfo") | ||
registeredService = true | ||
} | ||
|
||
override fun onServiceUnregistered(serviceInfo: NsdServiceInfo?) {} | ||
} | ||
val si = NsdServiceInfo().apply { | ||
serviceType = "_rfb._tcp" | ||
serviceName = advertisedName | ||
port = advertisedPort | ||
} | ||
nsdManager?.registerService(si, NsdManager.PROTOCOL_DNS_SD, listener) | ||
listeners.add(listener) | ||
|
||
Assume.assumeTrue(runCatching { pollingAssert { assertTrue(registeredService) } }.isSuccess) | ||
} | ||
|
||
private fun assertDiscoveryState(test: Discovery.() -> Boolean) { | ||
pollingAssert { runOnMainSync { assertTrue(Discovery.test()) } } | ||
} | ||
|
||
@Before | ||
fun before() { | ||
nsdManager = ContextCompat.getSystemService(targetContext, NsdManager::class.java) | ||
} | ||
|
||
@After | ||
fun after() { | ||
assertDiscoveryState { Log.d(TAG, "AfterTest: state: ${isRunning.value}, list: ${servers.value}"); true } | ||
listeners.forEach { nsdManager?.unregisterService(it) } | ||
listeners.clear() | ||
assertDiscoveryState { servers.value!!.isEmpty() } | ||
Discovery.stop() | ||
assertDiscoveryState { isRunning.value == false } | ||
} | ||
|
||
@Test | ||
fun startStop() { | ||
assertDiscoveryState { isRunning.value == false } | ||
Discovery.start(targetContext) | ||
assertDiscoveryState { isRunning.value == true } | ||
Discovery.stop() | ||
assertDiscoveryState { isRunning.value == false } | ||
} | ||
|
||
@Test | ||
fun singleService() { | ||
Discovery.start(targetContext) | ||
advertiseService("Server 1", 5999) | ||
assertDiscoveryState { | ||
isRunning.value == true && servers.value!!.size == 1 && servers.value!![0].port == 5999 | ||
} | ||
} | ||
|
||
@Test | ||
fun multipleServices() { | ||
Discovery.start(targetContext) | ||
|
||
val count = 10 | ||
for (i in 1..count) | ||
advertiseService("Server $i", 5900 + i) | ||
|
||
assertDiscoveryState { | ||
isRunning.value == true && servers.value!!.size == count && | ||
servers.value!!.find { it.port == 5901 } != null && | ||
servers.value!!.find { it.port == 5904 } != null && | ||
servers.value!!.find { it.port == 5908 } != null | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.