Network/test/dorkboxTest/network/ConnectionFilterTest.kt

644 lines
16 KiB
Kotlin

/*
* Copyright 2023 dorkbox, llc
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package dorkboxTest.network
import dorkbox.netUtil.IPv4
import dorkbox.netUtil.IPv6
import dorkbox.network.Client
import dorkbox.network.Server
import dorkbox.network.connection.Connection
import dorkbox.network.exceptions.ClientException
import dorkbox.network.ipFilter.IpSubnetFilterRule
import kotlinx.atomicfu.atomic
import org.junit.Assert
import org.junit.Test
@Suppress("UNUSED_ANONYMOUS_PARAMETER")
class ConnectionFilterTest : BaseTest() {
@Test
fun autoAcceptAll() {
val serverConnectSuccess = atomic(false)
val clientConnectSuccess = atomic(false)
val server = run {
val configuration = serverConfig()
val server: Server<Connection> = Server(configuration)
addEndPoint(server)
server.onConnect {
serverConnectSuccess.value = true
close()
}
server
}
val client = run {
val config = clientConfig()
val client: Client<Connection> = Client(config)
addEndPoint(client)
client.onConnect {
clientConnectSuccess.value = true
}
client.onDisconnect {
stopEndPoints()
}
client
}
server.bind(2000)
try {
client.connect(LOCALHOST, 2000)
} catch (e: Exception) {
stopEndPoints()
waitForThreads()
throw e
}
waitForThreads()
Assert.assertTrue(serverConnectSuccess.value)
Assert.assertTrue(clientConnectSuccess.value)
}
@Test
fun acceptWildcardServer() {
val serverConnectSuccess = atomic(false)
val clientConnectSuccess = atomic(false)
val server = run {
val configuration = serverConfig()
val server: Server<Connection> = Server(configuration)
addEndPoint(server)
server.filter(IpSubnetFilterRule(IPv4.WILDCARD, 0))
server.filter(IpSubnetFilterRule(IPv6.WILDCARD, 0))
server.onConnect {
serverConnectSuccess.value = true
close()
}
server
}
val client = run {
val config = clientConfig()
val client: Client<Connection> = Client(config)
addEndPoint(client)
client.onConnect {
clientConnectSuccess.value = true
}
client.onDisconnect {
stopEndPoints()
}
client
}
server.bind(2000)
try {
client.connect(LOCALHOST, 2000)
} catch (e: Exception) {
stopEndPoints()
waitForThreads()
throw e
}
waitForThreads()
Assert.assertTrue(serverConnectSuccess.value)
Assert.assertTrue(clientConnectSuccess.value)
}
@Test
fun acceptZeroCidrServer() {
val serverConnectSuccess = atomic(false)
val clientConnectSuccess = atomic(false)
val server = run {
val configuration = serverConfig()
val server: Server<Connection> = Server(configuration)
addEndPoint(server)
server.filter(IpSubnetFilterRule("1.1.1.1", 0))
server.filter(IpSubnetFilterRule("::1.1.1.1", 0)) // compressed ipv6
server.onConnect {
serverConnectSuccess.value = true
close()
}
server
}
val client = run {
val config = clientConfig()
val client: Client<Connection> = Client(config)
addEndPoint(client)
client.onConnect {
clientConnectSuccess.value = true
}
client.onDisconnect {
stopEndPoints()
}
client
}
server.bind(2000)
try {
client.connect(LOCALHOST, 2000)
} catch (e: Exception) {
stopEndPoints()
waitForThreads()
throw e
}
waitForThreads()
Assert.assertTrue(serverConnectSuccess.value)
Assert.assertTrue(clientConnectSuccess.value)
}
@Test
fun acceptWildcardClient() {
val serverConnectSuccess = atomic(false)
val clientConnectSuccess = atomic(false)
val server = run {
val configuration = serverConfig()
val server: Server<Connection> = Server(configuration)
addEndPoint(server)
server.filter(IpSubnetFilterRule(IPv4.WILDCARD, 0))
server.filter(IpSubnetFilterRule(IPv6.WILDCARD, 0))
server.onConnect {
serverConnectSuccess.value = true
close()
}
server
}
val client = run {
val config = clientConfig()
val client: Client<Connection> = Client(config)
addEndPoint(client)
client.onConnect {
clientConnectSuccess.value = true
}
client.onDisconnect {
stopEndPoints()
}
client
}
server.bind(2000)
try {
client.connect(LOCALHOST, 2000)
} catch (e: Exception) {
stopEndPoints()
waitForThreads()
throw e
}
waitForThreads()
Assert.assertTrue(serverConnectSuccess.value)
Assert.assertTrue(clientConnectSuccess.value)
}
@Test(expected = ClientException::class)
fun rejectServer() {
val server = run {
val serverConfig = serverConfig()
val server: Server<Connection> = Server(serverConfig)
addEndPoint(server)
server.filter(IpSubnetFilterRule("1.1.1.1", 32)) // this address will NEVER actually connect. we just use it for testing
server.onConnect {
close()
}
server
}
val client = run {
val clientConfig = clientConfig()
val client: Client<Connection> = Client(clientConfig)
addEndPoint(client)
client.onDisconnect {
stopEndPoints()
}
client
}
server.bind(2000)
try {
client.connect(LOCALHOST, 2000)
} catch (e: Exception) {
stopEndPoints()
waitForThreads { errors ->
errors.filterNot {
it.message?.contains("Connection was not permitted!") ?: false
}
}
throw e
}
// fail, since we should have thrown an exception
Assert.assertFalse(true)
}
@Test
fun rejectServerIpc() {
// we do not want to limit loopback addresses! Even with filtering, IPC is always allowed to connect
val serverConnectSuccess = atomic(false)
val clientConnectSuccess = atomic(false)
val server = run {
val configuration = serverConfig() {
enableIpc = true
}
val server: Server<Connection> = Server(configuration)
addEndPoint(server)
server.filter(IpSubnetFilterRule("1.1.1.1", 32)) // this address will NEVER actually connect. we just use it for testing
server.onConnect {
serverConnectSuccess.value = true
close()
}
server
}
val client = run {
val config = clientConfig() {
enableIpc = true
}
val client: Client<Connection> = Client(config)
addEndPoint(client)
client.onConnect {
clientConnectSuccess.value = true
}
client.onDisconnect {
logger.error("STARTING TO CLOSE CLIENT")
stopEndPoints()
}
client
}
server.bind(2000)
try {
client.connect(LOCALHOST, 2000)
} catch (e: Exception) {
stopEndPoints()
// this is expected.
}
waitForThreads()
Assert.assertTrue(serverConnectSuccess.value)
Assert.assertTrue(clientConnectSuccess.value)
}
@Test(expected = ClientException::class)
fun rejectClient() {
val server = run {
val configuration = serverConfig()
val server: Server<Connection> = Server(configuration)
addEndPoint(server)
server.filter(IpSubnetFilterRule("1.1.1.1", 32)) // this address will NEVER actually connect. we just use it for testing
server.onConnect {
close()
}
server
}
val client = run {
val config = clientConfig()
val client: Client<Connection> = Client(config)
addEndPoint(client)
client.onDisconnect {
stopEndPoints()
}
client
}
server.bind(2000)
try {
client.connect(LOCALHOST, 2000)
} catch (e: Exception) {
stopEndPoints()
waitForThreads { errors ->
errors.filterNot {
it.message?.contains("Connection was not permitted!") ?: false
}
}
throw e
}
waitForThreads()
}
@Test
fun acceptAllCustomServer() {
val serverConnectSuccess = atomic(false)
val clientConnectSuccess = atomic(false)
val server = run {
val configuration = serverConfig()
val server: Server<Connection> = Server(configuration)
addEndPoint(server)
server.filter { clientAddress, tagName ->
true
}
server.onConnect {
serverConnectSuccess.value = true
close()
}
server
}
val client = run {
val config = clientConfig()
val client: Client<Connection> = Client(config)
addEndPoint(client)
client.onConnect {
clientConnectSuccess.value = true
}
client.onDisconnect {
stopEndPoints()
}
client
}
server.bind(2000)
try {
client.connect(LOCALHOST, 2000)
} catch (e: Exception) {
stopEndPoints()
waitForThreads()
throw e
}
waitForThreads()
Assert.assertTrue(serverConnectSuccess.value)
Assert.assertTrue(clientConnectSuccess.value)
}
@Test
fun acceptAllCustomClient() {
val serverConnectSuccess = atomic(false)
val clientConnectSuccess = atomic(false)
val server = run {
val configuration = serverConfig()
val server: Server<Connection> = Server(configuration)
addEndPoint(server)
server.onConnect {
serverConnectSuccess.value = true
close()
}
server
}
val client = run {
val config = clientConfig()
val client: Client<Connection> = Client(config)
addEndPoint(client)
client.onConnect {
clientConnectSuccess.value = true
}
client.onDisconnect {
stopEndPoints()
}
client
}
server.bind(2000)
try {
client.connect(LOCALHOST, 2000)
} catch (e: Exception) {
stopEndPoints()
waitForThreads()
throw e
}
waitForThreads()
Assert.assertTrue(serverConnectSuccess.value)
Assert.assertTrue(clientConnectSuccess.value)
}
@Test(expected = ClientException::class)
fun rejectCustomServer() {
val server = run {
val configuration = serverConfig()
val server: Server<Connection> = Server(configuration)
addEndPoint(server)
server.filter { clientAddress, tagName ->
false
}
server.onConnect {
close()
}
server
}
val client = run {
val config = clientConfig()
val client: Client<Connection> = Client(config)
addEndPoint(client)
client.onDisconnect {
stopEndPoints()
}
client
}
server.bind(2000)
try {
client.connect(LOCALHOST, 2000)
} catch (e: Exception) {
stopEndPoints()
waitForThreads{ errors ->
errors.filterNot {
it.message?.contains("Connection was not permitted!") ?: false
}
}
throw e
}
waitForThreads()
}
@Test(expected = ClientException::class)
fun rejectCustomClient() {
val server = run {
val configuration = serverConfig()
val server: Server<Connection> = Server(configuration)
addEndPoint(server)
server.filter { clientAddress, tagName ->
false
}
server.onConnect {
close()
}
server
}
val client = run {
val config = clientConfig()
val client: Client<Connection> = Client(config)
addEndPoint(client)
client.onDisconnect {
stopEndPoints()
}
client
}
server.bind(2000)
try {
client.connect(LOCALHOST, 2000)
} catch (e: Exception) {
stopEndPoints()
waitForThreads{ errors ->
errors.filterNot {
it.message?.contains("Connection was not permitted!") ?: false
}
}
throw e
}
waitForThreads()
}
@Test
fun acceptAllCustomClientNoPendingMessages() {
val serverConnectSuccess = atomic(false)
val clientConnectSuccess = atomic(false)
val server = run {
val configuration = serverConfig()
val server: Server<Connection> = Server(configuration)
addEndPoint(server)
server.enableBufferedMessages { clientAddress, tagName ->
false
}
server.onConnect {
serverConnectSuccess.value = true
close()
}
server
}
val client = run {
val config = clientConfig()
val client: Client<Connection> = Client(config)
addEndPoint(client)
client.onConnect {
clientConnectSuccess.value = true
}
client.onDisconnect {
stopEndPoints()
}
client
}
server.bind(2000)
try {
client.connect(LOCALHOST, 2000)
} catch (e: Exception) {
stopEndPoints()
waitForThreads()
throw e
}
waitForThreads()
Assert.assertTrue(serverConnectSuccess.value)
Assert.assertTrue(clientConnectSuccess.value)
}
}