blob: e04771f84d216e30bf3a3b999c144b44fcd1c168 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 org.apache.tuweni.devp2p.v5.internal
import org.apache.tuweni.bytes.Bytes
import org.apache.tuweni.crypto.Hash
import org.apache.tuweni.crypto.SECP256K1
import org.apache.tuweni.devp2p.EthereumNodeRecord
import org.apache.tuweni.devp2p.v5.AuthenticationProvider
import org.apache.tuweni.devp2p.v5.PacketCodec
import org.apache.tuweni.devp2p.v5.storage.RoutingTable
import org.apache.tuweni.devp2p.v5.encrypt.AES128GCM
import org.apache.tuweni.devp2p.v5.misc.SessionKey
import org.apache.tuweni.devp2p.v5.packet.FindNodeMessage
import org.apache.tuweni.devp2p.v5.packet.RandomMessage
import org.apache.tuweni.devp2p.v5.packet.UdpMessage
import org.apache.tuweni.devp2p.v5.packet.WhoAreYouMessage
import org.apache.tuweni.junit.BouncyCastleExtension
import org.junit.jupiter.api.Test
import org.junit.jupiter.api.extension.ExtendWith
import java.net.InetAddress
@ExtendWith(BouncyCastleExtension::class)
class DefaultPacketCodecTest {
private val keyPair: SECP256K1.KeyPair = SECP256K1.KeyPair.random()
private val enr: Bytes = EthereumNodeRecord.toRLP(keyPair, ip = InetAddress.getLocalHost())
private val nodeId: Bytes = Hash.sha2_256(enr)
private val routingTable: RoutingTable = RoutingTable(enr)
private val authenticationProvider: AuthenticationProvider = DefaultAuthenticationProvider(keyPair, routingTable)
private val codec: PacketCodec = DefaultPacketCodec(keyPair, routingTable, nodeId, authenticationProvider)
private val destNodeId: Bytes = Bytes.random(32)
@Test
fun encodePerformsValidEncodingOfRandomMessage() {
val message = RandomMessage()
val encodedResult = codec.encode(message, destNodeId)
val encodedContent = encodedResult.content.slice(45)
val result = RandomMessage.create(UdpMessage.authTag(), encodedContent)
assert(result.data == message.data)
}
@Test
fun encodePerformsValidEncodingOfWhoAreYouMessage() {
val message = WhoAreYouMessage()
val encodedResult = codec.encode(message, destNodeId)
val encodedContent = encodedResult.content.slice(32)
val result = WhoAreYouMessage.create(encodedContent)
assert(result.idNonce == message.idNonce)
assert(result.enrSeq == message.enrSeq)
assert(result.authTag == message.authTag)
}
@Test
fun encodePerformsValidEncodingOfMessagesWithTypeIncluded() {
val message = FindNodeMessage()
val key = Bytes.random(16)
val sessionKey = SessionKey(key, key, key)
authenticationProvider.setSessionKey(destNodeId.toHexString(), sessionKey)
val encodedResult = codec.encode(message, destNodeId)
val tag = encodedResult.content.slice(0, UdpMessage.TAG_LENGTH)
val encryptedContent = encodedResult.content.slice(45)
val content = AES128GCM.decrypt(encryptedContent, sessionKey.initiatorKey, tag).slice(1)
val result = FindNodeMessage.create(content)
assert(result.requestId == message.requestId)
assert(result.distance == message.distance)
}
@Test
fun decodePerformsValidDecodingOfRandomMessasge() {
val message = RandomMessage()
val encodedResult = codec.encode(message, destNodeId)
val result = codec.decode(encodedResult.content).message as? RandomMessage
assert(null != result)
assert(result!!.data == message.data)
}
@Test
fun decodePerformsValidDecodingOfWhoAreYouMessage() {
val message = WhoAreYouMessage()
val encodedResult = codec.encode(message, destNodeId)
val result = codec.decode(encodedResult.content).message as? WhoAreYouMessage
assert(null != result)
assert(result!!.idNonce == message.idNonce)
assert(result.enrSeq == message.enrSeq)
assert(result.authTag == message.authTag)
}
@Test
fun decodePerformsValidDecodingOfMessagesWithTypeIncluded() {
val message = FindNodeMessage()
val key = Bytes.random(16)
val sessionKey = SessionKey(key, key, key)
authenticationProvider.setSessionKey(destNodeId.toHexString(), sessionKey)
authenticationProvider.setSessionKey(nodeId.toHexString(), sessionKey)
val encodedResult = codec.encode(message, nodeId)
val result = codec.decode(encodedResult.content).message as? FindNodeMessage
assert(result!!.requestId == message.requestId)
assert(result.distance == message.distance)
}
}