From d7f3fa1b3a432bcbb7d27105eb147d1a927451f4 Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Wed, 9 Oct 2024 19:12:21 +0900 Subject: [PATCH 01/16] Update FlowCollectionGuarantee --- .../main/kotlin/org/onflow/flow/sdk/models.kt | 35 +++++++++++++++++-- 1 file changed, 33 insertions(+), 2 deletions(-) diff --git a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt index 9006b2b..78837ce 100644 --- a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt +++ b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt @@ -918,13 +918,19 @@ data class FlowBlockExecutionData( data class FlowCollectionGuarantee( val id: FlowId, - val signatures: List + val signatures: List, + val referenceBlockId: FlowId, + val signature: FlowSignature, + val signerIndices: ByteArray ) : Serializable { companion object { @JvmStatic fun of(value: CollectionOuterClass.CollectionGuarantee) = FlowCollectionGuarantee( id = FlowId.of(value.collectionId.toByteArray()), - signatures = value.signaturesList.map { FlowSignature(it.toByteArray()) } + signatures = value.signaturesList.map { FlowSignature(it.toByteArray()) }, + referenceBlockId = FlowId.of(value.referenceBlockId.toByteArray()), + signature = FlowSignature(value.signature.toByteArray()), + signerIndices = value.signerIndices.toByteArray() ) } @@ -932,6 +938,31 @@ data class FlowCollectionGuarantee( fun builder(builder: CollectionOuterClass.CollectionGuarantee.Builder = CollectionOuterClass.CollectionGuarantee.newBuilder()): CollectionOuterClass.CollectionGuarantee.Builder = builder .setCollectionId(id.byteStringValue) .addAllSignatures(signatures.map { it.byteStringValue }) + .setReferenceBlockId(referenceBlockId.byteStringValue) + .setSignature(signature.byteStringValue) + .setSignerIndices(UnsafeByteOperations.unsafeWrap(signerIndices)) + + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (other !is FlowCollectionGuarantee) return false + + if (id != other.id) return false + if (signatures != other.signatures) return false + if (referenceBlockId != other.referenceBlockId) return false + if (signature != other.signature) return false + if (!signerIndices.contentEquals(other.signerIndices)) return false + + return true + } + + override fun hashCode(): Int { + var result = id.hashCode() + result = 31 * result + signatures.hashCode() + result = 31 * result + referenceBlockId.hashCode() + result = 31 * result + signature.hashCode() + result = 31 * result + signerIndices.contentHashCode() + return result + } } data class FlowBlockSeal( From 11f2ad5285c9421eb186ea436bb6a11ed6854b94 Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Wed, 9 Oct 2024 19:18:58 +0900 Subject: [PATCH 02/16] Update tests --- .../sdk/models/FlowCollectionGuaranteeTest.kt | 19 ++++++++++++++++++- 1 file changed, 18 insertions(+), 1 deletion(-) diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowCollectionGuaranteeTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowCollectionGuaranteeTest.kt index 3b3f530..78ed1ff 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowCollectionGuaranteeTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowCollectionGuaranteeTest.kt @@ -7,18 +7,26 @@ import org.onflow.flow.sdk.FlowSignature import org.junit.jupiter.api.Assertions.assertEquals import org.junit.jupiter.api.Test import org.onflow.protobuf.entities.CollectionOuterClass +import com.google.protobuf.UnsafeByteOperations class FlowCollectionGuaranteeTest { + @Test fun `Test building FlowCollectionGuarantee from CollectionOuterClass`() { val collectionIdBytes = byteArrayOf(1, 2, 3) + val referenceBlockIdBytes = byteArrayOf(10, 11, 12) + val signatureBytes = byteArrayOf(13, 14, 15) val signature1Bytes = byteArrayOf(4, 5, 6) val signature2Bytes = byteArrayOf(7, 8, 9) + val signerIndicesBytes = byteArrayOf(16, 17) val collectionGuaranteeBuilder = CollectionOuterClass.CollectionGuarantee.newBuilder() .setCollectionId(ByteString.copyFrom(collectionIdBytes)) .addSignatures(ByteString.copyFrom(signature1Bytes)) .addSignatures(ByteString.copyFrom(signature2Bytes)) + .setReferenceBlockId(ByteString.copyFrom(referenceBlockIdBytes)) + .setSignature(ByteString.copyFrom(signatureBytes)) + .setSignerIndices(UnsafeByteOperations.unsafeWrap(signerIndicesBytes)) val flowCollectionGuarantee = FlowCollectionGuarantee.of(collectionGuaranteeBuilder.build()) @@ -26,14 +34,20 @@ class FlowCollectionGuaranteeTest { assertEquals(2, flowCollectionGuarantee.signatures.size) assertEquals(FlowSignature(signature1Bytes), flowCollectionGuarantee.signatures[0]) assertEquals(FlowSignature(signature2Bytes), flowCollectionGuarantee.signatures[1]) + assertEquals(FlowId.of(referenceBlockIdBytes), flowCollectionGuarantee.referenceBlockId) + assertEquals(FlowSignature(signatureBytes), flowCollectionGuarantee.signature) + assertEquals(signerIndicesBytes.toList(), flowCollectionGuarantee.signerIndices.toList()) } @Test fun `Test building CollectionOuterClass from FlowCollectionGuarantee`() { val collectionId = FlowId.of(byteArrayOf(1, 2, 3)) + val referenceBlockId = FlowId.of(byteArrayOf(10, 11, 12)) + val signature = FlowSignature(byteArrayOf(13, 14, 15)) val signatures = listOf(FlowSignature(byteArrayOf(4, 5, 6)), FlowSignature(byteArrayOf(7, 8, 9))) + val signerIndices = byteArrayOf(16, 17) - val flowCollectionGuarantee = FlowCollectionGuarantee(collectionId, signatures) + val flowCollectionGuarantee = FlowCollectionGuarantee(collectionId, signatures, referenceBlockId, signature, signerIndices) val collectionGuaranteeBuilder = flowCollectionGuarantee.builder() val collectionGuarantee = collectionGuaranteeBuilder.build() @@ -42,5 +56,8 @@ class FlowCollectionGuaranteeTest { assertEquals(2, collectionGuarantee.signaturesCount) assertEquals(ByteString.copyFrom(signatures[0].bytes), collectionGuarantee.getSignatures(0)) assertEquals(ByteString.copyFrom(signatures[1].bytes), collectionGuarantee.getSignatures(1)) + assertEquals(ByteString.copyFrom(referenceBlockId.bytes), collectionGuarantee.referenceBlockId) + assertEquals(ByteString.copyFrom(signature.bytes), collectionGuarantee.signature) + assertEquals(ByteString.copyFrom(signerIndices), collectionGuarantee.signerIndices) } } From fbd0e87901bbc5129c5b7c6734e899f0e193add4 Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Wed, 9 Oct 2024 19:27:10 +0900 Subject: [PATCH 03/16] Lint --- .../org/onflow/flow/sdk/models/FlowCollectionGuaranteeTest.kt | 1 - 1 file changed, 1 deletion(-) diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowCollectionGuaranteeTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowCollectionGuaranteeTest.kt index 78ed1ff..6656ca5 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowCollectionGuaranteeTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowCollectionGuaranteeTest.kt @@ -10,7 +10,6 @@ import org.onflow.protobuf.entities.CollectionOuterClass import com.google.protobuf.UnsafeByteOperations class FlowCollectionGuaranteeTest { - @Test fun `Test building FlowCollectionGuarantee from CollectionOuterClass`() { val collectionIdBytes = byteArrayOf(1, 2, 3) From f9bb646c4572de99387e9a83999a41b670e83077 Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Fri, 11 Oct 2024 00:00:55 +0900 Subject: [PATCH 04/16] Working on models --- sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt index 78837ce..c4a321b 100644 --- a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt +++ b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt @@ -644,6 +644,10 @@ data class FlowBlock( val collectionGuarantees: List, val blockSeals: List, val signatures: List, + val executionReceiptMetalist, + val exectionResultList, + val blockHeader, + val protocolStateId ) : Serializable { companion object { @JvmStatic From 31666fbc35cbf0a3ed5e5a9e38a0bd11925ab380 Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Tue, 15 Oct 2024 15:41:57 +0900 Subject: [PATCH 05/16] WIP: Update models for block --- sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt index c4a321b..4465445 100644 --- a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt +++ b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt @@ -644,10 +644,10 @@ data class FlowBlock( val collectionGuarantees: List, val blockSeals: List, val signatures: List, - val executionReceiptMetalist, - val exectionResultList, - val blockHeader, - val protocolStateId + val executionReceiptMetalist: List, + val exectionResultList: List, + val blockHeader: FlowBlockHeader, + val protocolStateId: FlowId ) : Serializable { companion object { @JvmStatic @@ -659,6 +659,10 @@ data class FlowBlock( collectionGuarantees = value.collectionGuaranteesList.map { FlowCollectionGuarantee.of(it) }, blockSeals = value.blockSealsList.map { FlowBlockSeal.of(it) }, signatures = value.signaturesList.map { FlowSignature(it.toByteArray()) }, + executionReceiptMetalist = value.executionReceiptMetaListList.map { FlowExecutionReceiptMeta(it.toByteArray()) }, + exectionResultList = value.executionResultListList.map { FlowExecutionResult.of(it)}, + blockHeader = FlowBlockHeader.of(value.blockHeader), + protocolStateId = FlowId.of(value.protocolStateId.toByteArray()) ) } From 46be4273c676ddd53242af08cee6eda3c8c4da8e Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Tue, 15 Oct 2024 17:43:47 +0900 Subject: [PATCH 06/16] WIP: Update models for block --- .../main/kotlin/org/onflow/flow/sdk/models.kt | 74 ++++++++++++++----- 1 file changed, 55 insertions(+), 19 deletions(-) diff --git a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt index 4465445..a3f603b 100644 --- a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt +++ b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt @@ -193,14 +193,15 @@ data class FlowAccountKey( .setSequenceNumber(sequenceNumber) .setRevoked(revoked) - val encoded: ByteArray get() = RLPCodec.encode( - arrayOf( - publicKey.bytes, - signAlgo.code, - hashAlgo.code, - weight + val encoded: ByteArray + get() = RLPCodec.encode( + arrayOf( + publicKey.bytes, + signAlgo.code, + hashAlgo.code, + weight + ) ) - ) } data class FlowEventResult( @@ -250,6 +251,7 @@ data class FlowEvent( val event: EventField get() = payload.jsonCadence as EventField private fun > getField(name: String): T? = event[name] + @Suppress("UNCHECKED_CAST") operator fun get(name: String): T? = getField>(name) as T operator fun contains(name: String): Boolean = name in event @@ -312,7 +314,13 @@ data class FlowTransactionResult( @JvmOverloads fun getEventsOfType(type: String, exact: Boolean = false, expectedCount: Int? = null): List { val ret = this.events - .filter { if (exact) { it.type == type } else { it.type.endsWith(type) } } + .filter { + if (exact) { + it.type == type + } else { + it.type.endsWith(type) + } + } .map { it.event } check(expectedCount == null || ret.size == expectedCount) { "Expected $expectedCount events of type $type but there were ${ret.size}" } return ret @@ -449,12 +457,13 @@ data class FlowTransaction( return ret } - val signerMap: Map get() { - return signerList - .withIndex() - .map { it.value to it.index } - .toMap() - } + val signerMap: Map + get() { + return signerList + .withIndex() + .map { it.value to it.index } + .toMap() + } companion object { @JvmStatic @@ -469,6 +478,7 @@ data class FlowTransaction( payloadSignatures = value.payloadSignaturesList.map { FlowTransactionSignature.of(it) }, envelopeSignatures = value.envelopeSignaturesList.map { FlowTransactionSignature.of(it) } ) + @JvmStatic fun of(bytes: ByteArray): FlowTransaction { val txEnvelope: TransactionEnvelope = RLPCodec.decode(bytes, TransactionEnvelope::class.java) @@ -608,6 +618,7 @@ data class FlowTransactionSignature( signature = FlowSignature(value.signature.toByteArray()) ) } + @JvmOverloads fun builder(builder: TransactionOuterClass.Transaction.Signature.Builder = TransactionOuterClass.Transaction.Signature.newBuilder()): TransactionOuterClass.Transaction.Signature.Builder = builder .setAddress(address.byteStringValue) @@ -644,8 +655,8 @@ data class FlowBlock( val collectionGuarantees: List, val blockSeals: List, val signatures: List, - val executionReceiptMetalist: List, - val exectionResultList: List, + val executionReceiptMetaList: List, + val executionResultList: List, val blockHeader: FlowBlockHeader, val protocolStateId: FlowId ) : Serializable { @@ -659,10 +670,10 @@ data class FlowBlock( collectionGuarantees = value.collectionGuaranteesList.map { FlowCollectionGuarantee.of(it) }, blockSeals = value.blockSealsList.map { FlowBlockSeal.of(it) }, signatures = value.signaturesList.map { FlowSignature(it.toByteArray()) }, - executionReceiptMetalist = value.executionReceiptMetaListList.map { FlowExecutionReceiptMeta(it.toByteArray()) }, - exectionResultList = value.executionResultListList.map { FlowExecutionResult.of(it)}, + executionReceiptMetaList = value.executionReceiptMetaListList.map { FlowExecutionReceiptMeta.of(it) }, + executionResultList = value.executionResultListList.map { FlowExecutionResult.of(it) }, blockHeader = FlowBlockHeader.of(value.blockHeader), - protocolStateId = FlowId.of(value.protocolStateId.toByteArray()) + protocolStateId = FlowId.of(value.protocolStateId.toByteArray()) ) } @@ -778,6 +789,31 @@ data class FlowExecutionResult( chunks = grpcExecutionResult.executionResult.chunksList.map { FlowChunk.of(it) }, serviceEvents = grpcExecutionResult.executionResult.serviceEventsList.map { FlowServiceEvent.of(it) }, ) + + fun of(grpcExecutionResult: ExecutionResultOuterClass.ExecutionResult) = FlowExecutionResult( + blockId = FlowId.of(grpcExecutionResult.blockId.toByteArray()), + previousResultId = FlowId.of(grpcExecutionResult.previousResultId.toByteArray()), + chunks = grpcExecutionResult.chunksList.map { FlowChunk.of(it) }, + serviceEvents = grpcExecutionResult.serviceEventsList.map { FlowServiceEvent.of(it) }, + ) + + + } +} + +data class FlowExecutionReceiptMeta( + val executorId: FlowId, + val resultId: FlowId, + val spocks: List, + val executorSignature: FlowSignature, +) : Serializable { + companion object { + fun of(grpcExecutionResult: ExecutionResultOuterClass.ExecutionReceiptMeta) = FlowExecutionReceiptMeta( + executorId = FlowId.of(grpcExecutionResult.executorId.toByteArray()), + resultId = FlowId.of(grpcExecutionResult.resultId.toByteArray()), + spocks = grpcExecutionResult.spocksList.map { it.toByteArray() }, + executorSignature = FlowSignature(grpcExecutionResult.executorSignature.toByteArray()) + ) } } From 585c458d3595465892691dfca5f56d58ac730b6a Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Tue, 15 Oct 2024 18:04:13 +0900 Subject: [PATCH 07/16] WIP: Update models for block --- .../main/kotlin/org/onflow/flow/sdk/models.kt | 138 +++++++++++++++++- 1 file changed, 136 insertions(+), 2 deletions(-) diff --git a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt index a3f603b..29b5565 100644 --- a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt +++ b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt @@ -629,14 +629,34 @@ data class FlowTransactionSignature( data class FlowBlockHeader( val id: FlowId, val parentId: FlowId, - val height: Long + val height: Long, + val timestamp: LocalDateTime, + val payloadHash: ByteArray, + val view: Long, + val parentVoterSigData: ByteArray, + val proposerId: FlowId, + val proposerSigData: ByteArray, + val chainId: FlowChainId, + val parentVoterIndices: ByteArray, + val lastViewTc: FlowTimeoutCertificate, + val parentView: Long ) : Serializable { companion object { @JvmStatic fun of(value: BlockHeaderOuterClass.BlockHeader): FlowBlockHeader = FlowBlockHeader( id = FlowId.of(value.id.toByteArray()), parentId = FlowId.of(value.parentId.toByteArray()), - height = value.height + height = value.height, + timestamp = value.timestamp.asLocalDateTime(), + payloadHash = value.payloadHash.toByteArray(), + view = value.view, + parentVoterSigData = value.parentVoterSigData.toByteArray(), + proposerId = FlowId.of(value.proposerId.toByteArray()), + proposerSigData = value.proposerSigData.toByteArray(), + chainId = FlowChainId.of(value.chainId), + parentVoterIndices = value.parentVoterIndices.toByteArray(), + lastViewTc = FlowTimeoutCertificate.of(value.lastViewTc), + parentView = value.parentView ) } @@ -645,6 +665,44 @@ data class FlowBlockHeader( .setId(id.byteStringValue) .setParentId(parentId.byteStringValue) .setHeight(height) + + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (other !is FlowBlockHeader) return false + + if (id != other.id) return false + if (parentId != other.parentId) return false + if (height != other.height) return false + if (timestamp != other.timestamp) return false + if (!payloadHash.contentEquals(other.payloadHash)) return false + if (view != other.view) return false + if (!parentVoterSigData.contentEquals(other.parentVoterSigData)) return false + if (proposerId != other.proposerId) return false + if (!proposerSigData.contentEquals(other.proposerSigData)) return false + if (chainId != other.chainId) return false + if (!parentVoterIndices.contentEquals(other.parentVoterIndices)) return false + if (lastViewTc != other.lastViewTc) return false + if (parentView != other.parentView) return false + + return true + } + + override fun hashCode(): Int { + var result = id.hashCode() + result = 31 * result + parentId.hashCode() + result = 31 * result + height.hashCode() + result = 31 * result + timestamp.hashCode() + result = 31 * result + payloadHash.contentHashCode() + result = 31 * result + view.hashCode() + result = 31 * result + parentVoterSigData.contentHashCode() + result = 31 * result + proposerId.hashCode() + result = 31 * result + proposerSigData.contentHashCode() + result = 31 * result + chainId.hashCode() + result = 31 * result + parentVoterIndices.contentHashCode() + result = 31 * result + lastViewTc.hashCode() + result = 31 * result + parentView.hashCode() + return result + } } data class FlowBlock( @@ -817,6 +875,82 @@ data class FlowExecutionReceiptMeta( } } +data class FlowTimeoutCertificate( + val view: Long, + val highQcViews: List, + val highestQc: FlowQuorumCertificate, + val signerIndices: ByteArray, + val sigData: ByteArray +) : Serializable { + companion object { + fun of(grpcExecutionResult: BlockHeaderOuterClass.TimeoutCertificate) = FlowTimeoutCertificate( + view = grpcExecutionResult.view, + highQcViews = grpcExecutionResult.highQcViewsList, + highestQc = FlowQuorumCertificate.of(grpcExecutionResult.highestQc), + signerIndices = grpcExecutionResult.signerIndices.toByteArray(), + sigData = grpcExecutionResult.sigData.toByteArray() + ) + } + + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (other !is FlowTimeoutCertificate) return false + + if (view != other.view) return false + if (highQcViews != other.highQcViews) return false + if (highestQc != other.highestQc) return false + if (!signerIndices.contentEquals(other.signerIndices)) return false + if (!sigData.contentEquals(other.sigData)) return false + + return true + } + + override fun hashCode(): Int { + var result = view.hashCode() + result = 31 * result + highQcViews.hashCode() + result = 31 * result + highestQc.hashCode() + result = 31 * result + signerIndices.contentHashCode() + result = 31 * result + sigData.contentHashCode() + return result + } +} + +data class FlowQuorumCertificate( + val view: Long, + val blockId: FlowId, + val signerIndices: ByteArray, + val sigData: ByteArray +) : Serializable { + companion object { + fun of(grpcExecutionResult: BlockHeaderOuterClass.QuorumCertificate) = FlowQuorumCertificate( + view = grpcExecutionResult.view, + blockId = FlowId.of(grpcExecutionResult.blockId.toByteArray()), + signerIndices = grpcExecutionResult.signerIndices.toByteArray(), + sigData = grpcExecutionResult.sigData.toByteArray() + ) + } + + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (other !is FlowQuorumCertificate) return false + + if (view != other.view) return false + if (blockId != other.blockId) return false + if (!signerIndices.contentEquals(other.signerIndices)) return false + if (!sigData.contentEquals(other.sigData)) return false + + return true + } + + override fun hashCode(): Int { + var result = view.hashCode() + result = 31 * result + blockId.hashCode() + result = 31 * result + signerIndices.contentHashCode() + result = 31 * result + sigData.contentHashCode() + return result + } +} + data class FlowChunkExecutionData( val collection: FlowExecutionDataCollection, val events: List, From 2c8caef76620f097705e39f7cd58a9362dff8ace Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Tue, 15 Oct 2024 19:12:19 +0900 Subject: [PATCH 08/16] Update unit tests, add builder methods --- .../main/kotlin/org/onflow/flow/sdk/models.kt | 64 +++++++++++++- .../sdk/impl/AsyncFlowAccessApiImplTest.kt | 86 ++++++++++++++++--- .../flow/sdk/impl/FlowAccessApiImplTest.kt | 39 ++++++--- .../onflow/flow/sdk/models/FlowBlockTest.kt | 46 ++++++++-- 4 files changed, 204 insertions(+), 31 deletions(-) diff --git a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt index 29b5565..83be771 100644 --- a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt +++ b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt @@ -665,6 +665,16 @@ data class FlowBlockHeader( .setId(id.byteStringValue) .setParentId(parentId.byteStringValue) .setHeight(height) + .setTimestamp(timestamp.asTimestamp()) + .setPayloadHash(UnsafeByteOperations.unsafeWrap(payloadHash)) + .setView(view) + .setParentVoterSigData(UnsafeByteOperations.unsafeWrap(parentVoterSigData)) + .setProposerId(proposerId.byteStringValue) + .setProposerSigData(UnsafeByteOperations.unsafeWrap(proposerSigData)) + .setChainId(chainId.id) + .setParentVoterIndices(UnsafeByteOperations.unsafeWrap(parentVoterIndices)) + .setLastViewTc(lastViewTc.builder().build()) + .setParentView(parentView) override fun equals(other: Any?): Boolean { if (this === other) return true @@ -744,6 +754,11 @@ data class FlowBlock( .addAllCollectionGuarantees(collectionGuarantees.map { it.builder().build() }) .addAllBlockSeals(blockSeals.map { it.builder().build() }) .addAllSignatures(signatures.map { it.byteStringValue }) + .addAllExecutionReceiptMetaList(executionReceiptMetaList.map { it.builder().build() }) + .addAllExecutionResultList(executionResultList.map { it.builder().build() }) + .setBlockHeader(blockHeader.builder().build()) + .setProtocolStateId(protocolStateId.byteStringValue) + } data class FlowChunk( @@ -773,6 +788,19 @@ data class FlowChunk( ) } + @JvmOverloads + fun builder(builder: ExecutionResultOuterClass.Chunk.Builder = ExecutionResultOuterClass.Chunk.newBuilder()): ExecutionResultOuterClass.Chunk.Builder = builder + .setCollectionIndex(collectionIndex) + .setStartState(ByteString.copyFrom(startState)) + .setEventCollection(ByteString.copyFrom(eventCollection)) + .setBlockId(blockId.byteStringValue) + .setTotalComputationUsed(totalComputationUsed) + .setNumberOfTransactions(numberOfTransactions) + .setIndex(index) + .setEndState(ByteString.copyFrom(endState)) + .setExecutionDataId(executionDataId.byteStringValue) + .setStateDeltaCommitment(ByteString.copyFrom(stateDeltaCommitment)) + override fun equals(other: Any?): Boolean { if (this === other) return true if (other !is FlowChunk) return false @@ -817,6 +845,11 @@ data class FlowServiceEvent( ) } + @JvmOverloads + fun builder(builder: ExecutionResultOuterClass.ServiceEvent.Builder = ExecutionResultOuterClass.ServiceEvent.newBuilder()): ExecutionResultOuterClass.ServiceEvent.Builder = builder + .setType(type) + .setPayload(ByteString.copyFrom(payload)) + override fun equals(other: Any?): Boolean { if (this === other) return true if (other !is FlowServiceEvent) return false @@ -854,9 +887,14 @@ data class FlowExecutionResult( chunks = grpcExecutionResult.chunksList.map { FlowChunk.of(it) }, serviceEvents = grpcExecutionResult.serviceEventsList.map { FlowServiceEvent.of(it) }, ) - - } + + @JvmOverloads + fun builder(builder: ExecutionResultOuterClass.ExecutionResult.Builder = ExecutionResultOuterClass.ExecutionResult.newBuilder()): ExecutionResultOuterClass.ExecutionResult.Builder = builder + .setBlockId(blockId.byteStringValue) + .setPreviousResultId(previousResultId.byteStringValue) + .addAllChunks(chunks.map { it.builder().build() }) + .addAllServiceEvents(serviceEvents.map { it.builder().build() }) } data class FlowExecutionReceiptMeta( @@ -873,6 +911,13 @@ data class FlowExecutionReceiptMeta( executorSignature = FlowSignature(grpcExecutionResult.executorSignature.toByteArray()) ) } + + @JvmOverloads + fun builder(builder: ExecutionResultOuterClass.ExecutionReceiptMeta.Builder = ExecutionResultOuterClass.ExecutionReceiptMeta.newBuilder()): ExecutionResultOuterClass.ExecutionReceiptMeta.Builder = builder + .setExecutorId(executorId.byteStringValue) + .setResultId(resultId.byteStringValue) + .addAllSpocks(spocks.map { ByteString.copyFrom(it) }) + .setExecutorSignature(executorSignature.byteStringValue) } data class FlowTimeoutCertificate( @@ -892,6 +937,14 @@ data class FlowTimeoutCertificate( ) } + @JvmOverloads + fun builder(builder: BlockHeaderOuterClass.TimeoutCertificate.Builder = BlockHeaderOuterClass.TimeoutCertificate.newBuilder()): BlockHeaderOuterClass.TimeoutCertificate.Builder = builder + .setView(view) + .addAllHighQcViews(highQcViews) + .setHighestQc(highestQc.builder().build()) + .setSignerIndices(ByteString.copyFrom(signerIndices)) + .setSigData(ByteString.copyFrom(sigData)) + override fun equals(other: Any?): Boolean { if (this === other) return true if (other !is FlowTimeoutCertificate) return false @@ -930,6 +983,13 @@ data class FlowQuorumCertificate( ) } + @JvmOverloads + fun builder(builder: BlockHeaderOuterClass.QuorumCertificate.Builder = BlockHeaderOuterClass.QuorumCertificate.newBuilder()): BlockHeaderOuterClass.QuorumCertificate.Builder = builder + .setView(view) + .setBlockId(blockId.byteStringValue) + .setSignerIndices(ByteString.copyFrom(signerIndices)) + .setSigData(ByteString.copyFrom(sigData)) + override fun equals(other: Any?): Boolean { if (this === other) return true if (other !is FlowQuorumCertificate) return false diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/impl/AsyncFlowAccessApiImplTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/impl/AsyncFlowAccessApiImplTest.kt index 6122092..11d6798 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/impl/AsyncFlowAccessApiImplTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/impl/AsyncFlowAccessApiImplTest.kt @@ -29,6 +29,36 @@ class AsyncFlowAccessApiImplTest { companion object { val BLOCK_ID_BYTES = byteArrayOf(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1) val PARENT_ID_BYTES = byteArrayOf(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2) + + const val HEIGHT = 123L + + val mockBlockHeader = FlowBlockHeader( + id = FlowId.of(BLOCK_ID_BYTES), + parentId = FlowId.of(PARENT_ID_BYTES), + height = 123L, + timestamp = LocalDateTime.of(2024, 10, 15, 18, 33, 12), + payloadHash = ByteArray(32) { 0 }, + view = 1L, + parentVoterSigData = ByteArray(32) { 0 }, + proposerId = FlowId.of(PARENT_ID_BYTES), + proposerSigData = ByteArray(32) { 0 }, + chainId = FlowChainId.MAINNET, + parentVoterIndices = ByteArray(32) { 0 }, + lastViewTc = FlowTimeoutCertificate( + view = 1L, + highQcViews = emptyList(), + highestQc = FlowQuorumCertificate( + view = 1L, + blockId = FlowId.of(BLOCK_ID_BYTES), + signerIndices = ByteArray(32) { 0 }, + sigData = ByteArray(32) { 0 } + ), + signerIndices = ByteArray(32) { 0 }, + sigData = ByteArray(32) { 0 } + ), + parentView = 1L + ) + } private fun setupFutureMock(response: T): ListenableFuture { @@ -48,7 +78,7 @@ class AsyncFlowAccessApiImplTest { @Test fun `test getLatestBlockHeader`() { - val mockBlockHeader = FlowBlockHeader(FlowId("01"), FlowId("01"), 123L) + val mockBlockHeader = mockBlockHeader val blockHeaderResponse = Access.BlockHeaderResponse.newBuilder().setBlock(mockBlockHeader.builder().build()).build() `when`(api.getLatestBlockHeader(any())).thenReturn(setupFutureMock(blockHeaderResponse)) @@ -60,8 +90,8 @@ class AsyncFlowAccessApiImplTest { @Test fun `test getBlockHeaderById`() { - val blockId = FlowId("01") - val mockBlockHeader = FlowBlockHeader(blockId, FlowId("01"), 123L) + val blockId = FlowId.of(BLOCK_ID_BYTES) + val mockBlockHeader = mockBlockHeader val blockHeaderResponse = Access.BlockHeaderResponse.newBuilder().setBlock(mockBlockHeader.builder().build()).build() `when`(api.getBlockHeaderByID(any())).thenReturn(setupFutureMock(blockHeaderResponse)) @@ -73,8 +103,8 @@ class AsyncFlowAccessApiImplTest { @Test fun `test getBlockHeaderByHeight`() { - val height = 123L - val mockBlockHeader = FlowBlockHeader(FlowId("01"), FlowId("01"), height) + val height = HEIGHT + val mockBlockHeader = mockBlockHeader val blockHeaderResponse = Access.BlockHeaderResponse.newBuilder().setBlock(mockBlockHeader.builder().build()).build() `when`(api.getBlockHeaderByHeight(any())).thenReturn(setupFutureMock(blockHeaderResponse)) @@ -86,7 +116,19 @@ class AsyncFlowAccessApiImplTest { @Test fun `test getLatestBlock`() { - val mockBlock = FlowBlock(FlowId("01"), FlowId("01"), 123L, LocalDateTime.now(), emptyList(), emptyList(), emptyList()) + val mockBlock = FlowBlock( + id = FlowId.of(BLOCK_ID_BYTES), + parentId = FlowId.of(PARENT_ID_BYTES), + height = 123L, + timestamp = LocalDateTime.now(), + collectionGuarantees = emptyList(), + blockSeals = emptyList(), + signatures = emptyList(), + executionReceiptMetaList = emptyList(), + executionResultList = emptyList(), + blockHeader = mockBlockHeader, + protocolStateId = FlowId.of(ByteArray(32)) + ) val blockResponse = Access.BlockResponse.newBuilder().setBlock(mockBlock.builder().build()).build() `when`(api.getLatestBlock(any())).thenReturn(setupFutureMock(blockResponse)) @@ -98,8 +140,20 @@ class AsyncFlowAccessApiImplTest { @Test fun `test getBlockById`() { - val blockId = FlowId("01") - val mockBlock = FlowBlock(blockId, FlowId("01"), 123L, LocalDateTime.now(), emptyList(), emptyList(), emptyList()) + val blockId = FlowId.of(BLOCK_ID_BYTES) + val mockBlock = FlowBlock( + id = blockId, + parentId = FlowId.of(PARENT_ID_BYTES), + height = 123L, + timestamp = LocalDateTime.now(), + collectionGuarantees = emptyList(), + blockSeals = emptyList(), + signatures = emptyList(), + executionReceiptMetaList = emptyList(), + executionResultList = emptyList(), + blockHeader = mockBlockHeader, + protocolStateId = FlowId.of(ByteArray(32)) + ) val blockResponse = Access.BlockResponse.newBuilder().setBlock(mockBlock.builder().build()).build() `when`(api.getBlockByID(any())).thenReturn(setupFutureMock(blockResponse)) @@ -111,8 +165,20 @@ class AsyncFlowAccessApiImplTest { @Test fun `test getBlockByHeight`() { - val height = 123L - val mockBlock = FlowBlock(FlowId("01"), FlowId("01"), height, LocalDateTime.now(), emptyList(), emptyList(), emptyList()) + val height = HEIGHT + val mockBlock = FlowBlock( + FlowId("01"), + FlowId("01"), + height, + LocalDateTime.now(), + emptyList(), + emptyList(), + emptyList(), + emptyList(), + emptyList(), + mockBlockHeader, + FlowId.of(ByteArray(32)) + ) val blockResponse = Access.BlockResponse.newBuilder().setBlock(mockBlock.builder().build()).build() `when`(api.getBlockByHeight(any())).thenReturn(setupFutureMock(blockResponse)) diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/impl/FlowAccessApiImplTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/impl/FlowAccessApiImplTest.kt index 8f52169..c50341d 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/impl/FlowAccessApiImplTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/impl/FlowAccessApiImplTest.kt @@ -32,11 +32,30 @@ class FlowAccessApiImplTest { private lateinit var outputStreamCaptor: ByteArrayOutputStream private lateinit var originalOut: PrintStream - private val api = mock(AccessAPIGrpc.AccessAPIBlockingStub::class.java) - private val executionDataApi = mock(ExecutionDataAPIGrpc.ExecutionDataAPIBlockingStub::class.java) private val testDispatcher = TestCoroutineDispatcher() private val testScope = TestCoroutineScope(testDispatcher) + companion object { + val mockBlockHeader = FlowBlockHeader( + id = FlowId.of(AsyncFlowAccessApiImplTest.BLOCK_ID_BYTES), + parentId = FlowId.of(AsyncFlowAccessApiImplTest.PARENT_ID_BYTES), + height = 123L, + timestamp = LocalDateTime.now(), + payloadHash = ByteArray(32), + view = 1L, + parentVoterSigData = ByteArray(32), + proposerId = FlowId.of(AsyncFlowAccessApiImplTest.PARENT_ID_BYTES), + proposerSigData = ByteArray(32), + chainId = FlowChainId.MAINNET, + parentVoterIndices = ByteArray(32), + lastViewTc = FlowTimeoutCertificate(1L, emptyList(), FlowQuorumCertificate(1L, FlowId.of(AsyncFlowAccessApiImplTest.BLOCK_ID_BYTES), ByteArray(32), ByteArray(32)), ByteArray(32), ByteArray(32)), + parentView = 1L + ) + + val blockId = FlowId("01") + val mockBlock = FlowBlock(blockId, FlowId("01"), 123L, LocalDateTime.now(), emptyList(), emptyList(), emptyList(), emptyList(), emptyList(), mockBlockHeader, FlowId("01")) + } + @BeforeEach fun setUp() { mockApi = mock(AccessAPIGrpc.AccessAPIBlockingStub::class.java) @@ -60,7 +79,7 @@ class FlowAccessApiImplTest { @Test fun `Test getLatestBlockHeader`() { - val mockBlockHeader = FlowBlockHeader(FlowId("01"), FlowId("01"), 123L) + val mockBlockHeader = mockBlockHeader val blockHeaderProto = Access.BlockHeaderResponse.newBuilder().setBlock(mockBlockHeader.builder().build()).build() `when`(mockApi.getLatestBlockHeader(any())).thenReturn(blockHeaderProto) @@ -71,8 +90,8 @@ class FlowAccessApiImplTest { @Test fun `Test getBlockHeaderById`() { - val blockId = FlowId("01") - val mockBlockHeader = FlowBlockHeader(blockId, FlowId("01"), 123L) + val blockId = blockId + val mockBlockHeader = mockBlockHeader val blockHeaderProto = Access.BlockHeaderResponse.newBuilder().setBlock(mockBlockHeader.builder().build()).build() `when`(mockApi.getBlockHeaderByID(any())).thenReturn(blockHeaderProto) @@ -84,7 +103,7 @@ class FlowAccessApiImplTest { @Test fun `Test getBlockHeaderByHeight`() { val height = 123L - val mockBlockHeader = FlowBlockHeader(FlowId("01"), FlowId("01"), height) + val mockBlockHeader = mockBlockHeader val blockHeaderProto = Access.BlockHeaderResponse.newBuilder().setBlock(mockBlockHeader.builder().build()).build() `when`(mockApi.getBlockHeaderByHeight(any())).thenReturn(blockHeaderProto) @@ -95,7 +114,7 @@ class FlowAccessApiImplTest { @Test fun `Test getLatestBlock`() { - val mockBlock = FlowBlock(FlowId("01"), FlowId("01"), 123L, LocalDateTime.now(), emptyList(), emptyList(), emptyList()) + val mockBlock = mockBlock val blockProto = Access.BlockResponse.newBuilder().setBlock(mockBlock.builder().build()).build() `when`(mockApi.getLatestBlock(any())).thenReturn(blockProto) @@ -106,9 +125,9 @@ class FlowAccessApiImplTest { @Test fun `Test getBlockById`() { - val blockId = FlowId("01") - val mockBlock = FlowBlock(blockId, FlowId("01"), 123L, LocalDateTime.now(), emptyList(), emptyList(), emptyList()) + val blockId = blockId val blockProto = Access.BlockResponse.newBuilder().setBlock(mockBlock.builder().build()).build() + val mockBlock = mockBlock `when`(mockApi.getBlockByID(any())).thenReturn(blockProto) @@ -119,7 +138,7 @@ class FlowAccessApiImplTest { @Test fun `Test getBlockByHeight`() { val height = 123L - val mockBlock = FlowBlock(FlowId("01"), FlowId("01"), height, LocalDateTime.now(), emptyList(), emptyList(), emptyList()) + val mockBlock = mockBlock val blockProto = Access.BlockResponse.newBuilder().setBlock(mockBlock.builder().build()).build() `when`(mockApi.getBlockByHeight(any())).thenReturn(blockProto) diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockTest.kt index 8c1ca7f..6728991 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockTest.kt @@ -2,12 +2,10 @@ package org.onflow.flow.sdk.models import com.google.protobuf.ByteString import com.google.protobuf.Timestamp -import org.onflow.flow.sdk.FlowBlock -import org.onflow.flow.sdk.FlowId -import org.onflow.flow.sdk.asTimestamp -import org.onflow.flow.sdk.fixedSize import org.junit.jupiter.api.Assertions.assertEquals import org.junit.jupiter.api.Test +import org.onflow.flow.sdk.* +import org.onflow.protobuf.entities.BlockHeaderOuterClass import org.onflow.protobuf.entities.BlockOuterClass import java.time.Instant import java.time.LocalDateTime @@ -24,6 +22,14 @@ class FlowBlockTest { .setParentId(ByteString.copyFromUtf8("parent_id")) .setHeight(123) .setTimestamp(timestamp) + .setBlockHeader( + BlockHeaderOuterClass.BlockHeader.newBuilder() + .setId(ByteString.copyFromUtf8("header_id")) + .setParentId(ByteString.copyFromUtf8("header_parent_id")) + .setHeight(124) + .setTimestamp(timestamp) + .build()) + .setProtocolStateId(ByteString.copyFromUtf8("protocol_state_id")) val flowBlock = FlowBlock.of(blockBuilder.build()) @@ -40,13 +46,14 @@ class FlowBlockTest { assert(flowBlock.id.bytes.contentEquals(fixedSize("id".toByteArray(), 32))) assert(flowBlock.parentId.bytes.contentEquals(fixedSize("parent_id".toByteArray(), 32))) assertEquals(flowBlock.height, 123L) - assertEquals( - expectedUtcDateTime, - actualUtcDateTime, - ) + assertEquals(expectedUtcDateTime, actualUtcDateTime) assert(flowBlock.collectionGuarantees.isEmpty()) assert(flowBlock.blockSeals.isEmpty()) assert(flowBlock.signatures.isEmpty()) + assert(flowBlock.executionReceiptMetaList.isEmpty()) + assert(flowBlock.executionResultList.isEmpty()) + assert(flowBlock.blockHeader.id.bytes.contentEquals(fixedSize("header_id".toByteArray(), 32))) + assert(flowBlock.protocolStateId.bytes.contentEquals(fixedSize("protocol_state_id".toByteArray(), 32))) } @Test @@ -58,7 +65,25 @@ class FlowBlockTest { timestamp = LocalDateTime.now(), collectionGuarantees = emptyList(), blockSeals = emptyList(), - signatures = emptyList() + signatures = emptyList(), + executionReceiptMetaList = emptyList(), + executionResultList = emptyList(), + blockHeader = FlowBlockHeader( + id = FlowId.of("header_id".toByteArray()), + parentId = FlowId.of("header_parent_id".toByteArray()), + height = 124, + timestamp = LocalDateTime.now(), + payloadHash = ByteArray(32), + view = 1, + parentVoterSigData = ByteArray(32), + proposerId = FlowId.of("proposer_id".toByteArray()), + proposerSigData = ByteArray(32), + chainId = FlowChainId.MAINNET, + parentVoterIndices = ByteArray(32), + lastViewTc = FlowTimeoutCertificate(1L, emptyList(), FlowQuorumCertificate(1L, FlowId.of("block_id".toByteArray()), ByteArray(32), ByteArray(32)), ByteArray(32), ByteArray(32)), + parentView = 1 + ), + protocolStateId = FlowId.of("protocol_state_id".toByteArray()) ) val blockBuilder = flowBlock.builder() @@ -70,5 +95,8 @@ class FlowBlockTest { assert(blockBuilder.collectionGuaranteesList.isEmpty()) assert(blockBuilder.blockSealsList.isEmpty()) assert(blockBuilder.signaturesList.isEmpty()) + assert(blockBuilder.blockHeader.id.toByteArray().contentEquals(fixedSize("header_id".toByteArray(), 32))) + assert(blockBuilder.protocolStateId.toByteArray().contentEquals(fixedSize("protocol_state_id".toByteArray(), 32))) } } + From d7b50123263b4014c5424a3be94a05c730e22457 Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Tue, 15 Oct 2024 19:18:03 +0900 Subject: [PATCH 09/16] Setup files for missing tests --- .../kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt | 4 ++++ .../test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt | 4 ++++ .../onflow/flow/sdk/models/FlowExecutionReceiptMetaTest.kt | 4 ++++ .../org/onflow/flow/sdk/models/FlowExecutionResultTest.kt | 4 ++++ .../org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt | 4 ++++ .../org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt | 4 ++++ 6 files changed, 24 insertions(+) create mode 100644 sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt create mode 100644 sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt create mode 100644 sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionReceiptMetaTest.kt create mode 100644 sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt create mode 100644 sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt create mode 100644 sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt new file mode 100644 index 0000000..42a9676 --- /dev/null +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt @@ -0,0 +1,4 @@ +package org.onflow.flow.sdk.models + +class FlowBlockHeaderTest { +} diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt new file mode 100644 index 0000000..f3ad7f9 --- /dev/null +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt @@ -0,0 +1,4 @@ +package org.onflow.flow.sdk.models + +class FlowChunkTest { +} diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionReceiptMetaTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionReceiptMetaTest.kt new file mode 100644 index 0000000..2a0731f --- /dev/null +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionReceiptMetaTest.kt @@ -0,0 +1,4 @@ +package org.onflow.flow.sdk.models + +class FlowExecutionReceiptMetaTest { +} diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt new file mode 100644 index 0000000..6584548 --- /dev/null +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt @@ -0,0 +1,4 @@ +package org.onflow.flow.sdk.models + +class FlowExecutionResultTest { +} diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt new file mode 100644 index 0000000..5aef581 --- /dev/null +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt @@ -0,0 +1,4 @@ +package org.onflow.flow.sdk.models + +class FlowQuorumCertificateTest { +} diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt new file mode 100644 index 0000000..01a28ef --- /dev/null +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt @@ -0,0 +1,4 @@ +package org.onflow.flow.sdk.models + +class FlowTimeoutCertificateTest { +} From d311cff0be15cee0d9ef8e4b8b1db55dab6ce75b Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Tue, 15 Oct 2024 19:31:04 +0900 Subject: [PATCH 10/16] Setup tests for FlowQuorumCertificateTest --- .../sdk/models/FlowQuorumCertificateTest.kt | 87 +++++++++++++++++++ 1 file changed, 87 insertions(+) diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt index 5aef581..ab5cbb3 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt @@ -1,4 +1,91 @@ package org.onflow.flow.sdk.models +import org.junit.jupiter.api.Assertions.* +import org.junit.jupiter.api.Test +import org.mockito.Mockito.* +import com.google.protobuf.ByteString +import org.onflow.flow.sdk.FlowId +import org.onflow.flow.sdk.FlowQuorumCertificate +import org.onflow.protobuf.entities.BlockHeaderOuterClass + class FlowQuorumCertificateTest { + + @Test + fun `test of() method`() { + val view = 123L + val blockIdBytes = ByteArray(32) { 0x01 } + val signerIndicesBytes = ByteArray(32) { 0x02 } + val sigDataBytes = ByteArray(32) { 0x03 } + + val grpcQuorumCertificate = mock(BlockHeaderOuterClass.QuorumCertificate::class.java) + `when`(grpcQuorumCertificate.view).thenReturn(view) + `when`(grpcQuorumCertificate.blockId).thenReturn(ByteString.copyFrom(blockIdBytes)) + `when`(grpcQuorumCertificate.signerIndices).thenReturn(ByteString.copyFrom(signerIndicesBytes)) + `when`(grpcQuorumCertificate.sigData).thenReturn(ByteString.copyFrom(sigDataBytes)) + + val flowQuorumCertificate = FlowQuorumCertificate.of(grpcQuorumCertificate) + + assertEquals(view, flowQuorumCertificate.view) + assertArrayEquals(blockIdBytes, flowQuorumCertificate.blockId.bytes) + assertArrayEquals(signerIndicesBytes, flowQuorumCertificate.signerIndices) + assertArrayEquals(sigDataBytes, flowQuorumCertificate.sigData) + } + + @Test + fun `test builder() method`() { + val view = 123L + val blockId = FlowId.of(ByteArray(32) { 0x01 }) + val signerIndices = ByteArray(32) { 0x02 } + val sigData = ByteArray(32) { 0x03 } + + val flowQuorumCertificate = FlowQuorumCertificate(view, blockId, signerIndices, sigData) + + val builder = flowQuorumCertificate.builder().build() + + assertEquals(view, builder.view) + assertArrayEquals(blockId.bytes, builder.blockId.toByteArray()) + assertArrayEquals(signerIndices, builder.signerIndices.toByteArray()) + assertArrayEquals(sigData, builder.sigData.toByteArray()) + } + + @Test + fun `test equals() method`() { + val view1 = 123L + val blockId1 = FlowId.of(ByteArray(32) { 0x01 }) + val signerIndices1 = ByteArray(32) { 0x02 } + val sigData1 = ByteArray(32) { 0x03 } + + val view2 = 124L + val blockId2 = FlowId.of(ByteArray(32) { 0x04 }) + val signerIndices2 = ByteArray(32) { 0x05 } + val sigData2 = ByteArray(32) { 0x06 } + + val flowQuorumCertificate1 = FlowQuorumCertificate(view1, blockId1, signerIndices1, sigData1) + val flowQuorumCertificate2 = FlowQuorumCertificate(view1, blockId1, signerIndices1, sigData1) + val flowQuorumCertificate3 = FlowQuorumCertificate(view2, blockId2, signerIndices2, sigData2) + + assertEquals(flowQuorumCertificate1, flowQuorumCertificate2) + assertNotEquals(flowQuorumCertificate1, flowQuorumCertificate3) + } + + @Test + fun `test hashCode() method`() { + val view1 = 123L + val blockId1 = FlowId.of(ByteArray(32) { 0x01 }) + val signerIndices1 = ByteArray(32) { 0x02 } + val sigData1 = ByteArray(32) { 0x03 } + + val view2 = 124L + val blockId2 = FlowId.of(ByteArray(32) { 0x04 }) + val signerIndices2 = ByteArray(32) { 0x05 } + val sigData2 = ByteArray(32) { 0x06 } + + val flowQuorumCertificate1 = FlowQuorumCertificate(view1, blockId1, signerIndices1, sigData1) + val flowQuorumCertificate2 = FlowQuorumCertificate(view1, blockId1, signerIndices1, sigData1) + val flowQuorumCertificate3 = FlowQuorumCertificate(view2, blockId2, signerIndices2, sigData2) + + assertEquals(flowQuorumCertificate1.hashCode(), flowQuorumCertificate2.hashCode()) + assertNotEquals(flowQuorumCertificate1.hashCode(), flowQuorumCertificate3.hashCode()) + } } + From 0df71b02ad46f81c249115afc2115b620b98eca1 Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Tue, 15 Oct 2024 19:43:17 +0900 Subject: [PATCH 11/16] Setup tests for FlowTimeoutCertificateTest --- .../sdk/models/FlowTimeoutCertificateTest.kt | 126 ++++++++++++++++++ 1 file changed, 126 insertions(+) diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt index 01a28ef..b0ae891 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt @@ -1,4 +1,130 @@ package org.onflow.flow.sdk.models +import com.google.protobuf.ByteString +import org.junit.jupiter.api.Assertions.* +import org.junit.jupiter.api.Test +import org.mockito.Mockito.* +import org.onflow.flow.sdk.FlowQuorumCertificate +import org.onflow.flow.sdk.FlowTimeoutCertificate +import org.onflow.protobuf.entities.BlockHeaderOuterClass + class FlowTimeoutCertificateTest { + @Test + fun `test of() method`() { + val view = 123L + val highQcViews = listOf(456L, 789L) + val signerIndicesBytes = ByteArray(32) { 0x01 } + val sigDataBytes = ByteArray(32) { 0x02 } + + // mock QuorumCertificate + val grpcHighestQc = mock(BlockHeaderOuterClass.QuorumCertificate::class.java) + `when`(grpcHighestQc.view).thenReturn(view) + `when`(grpcHighestQc.blockId).thenReturn(ByteString.copyFrom(ByteArray(32) { 0x01 })) + `when`(grpcHighestQc.signerIndices).thenReturn(ByteString.copyFrom(ByteArray(32) { 0x02 })) + `when`(grpcHighestQc.sigData).thenReturn(ByteString.copyFrom(ByteArray(32) { 0x03 })) + + val highestQc = FlowQuorumCertificate.of(grpcHighestQc) + + // mock TimeoutCertificate + val grpcTimeoutCertificate = mock(BlockHeaderOuterClass.TimeoutCertificate::class.java) + `when`(grpcTimeoutCertificate.view).thenReturn(view) + `when`(grpcTimeoutCertificate.highQcViewsList).thenReturn(highQcViews) + `when`(grpcTimeoutCertificate.highestQc).thenReturn(grpcHighestQc) // Return the mocked QuorumCertificate + `when`(grpcTimeoutCertificate.signerIndices).thenReturn(ByteString.copyFrom(signerIndicesBytes)) + `when`(grpcTimeoutCertificate.sigData).thenReturn(ByteString.copyFrom(sigDataBytes)) + + val flowTimeoutCertificate = FlowTimeoutCertificate.of(grpcTimeoutCertificate) + + assertEquals(view, flowTimeoutCertificate.view) + assertEquals(highQcViews, flowTimeoutCertificate.highQcViews) + assertEquals(highestQc, flowTimeoutCertificate.highestQc) + assertArrayEquals(signerIndicesBytes, flowTimeoutCertificate.signerIndices) + assertArrayEquals(sigDataBytes, flowTimeoutCertificate.sigData) + } + + @Test + fun `test builder() method`() { + + // mock FlowQuorumCertificate + val view = 123L + val blockIdBytes = ByteArray(32) { 0x01 } + val signerIndicesBytes = ByteArray(32) { 0x02 } + val sigDataBytes = ByteArray(32) { 0x03 } + + val grpcQuorumCertificate = mock(BlockHeaderOuterClass.QuorumCertificate::class.java) + `when`(grpcQuorumCertificate.view).thenReturn(view) + `when`(grpcQuorumCertificate.blockId).thenReturn(ByteString.copyFrom(blockIdBytes)) + `when`(grpcQuorumCertificate.signerIndices).thenReturn(ByteString.copyFrom(signerIndicesBytes)) + `when`(grpcQuorumCertificate.sigData).thenReturn(ByteString.copyFrom(sigDataBytes)) + + val flowQuorumCertificate = FlowQuorumCertificate.of(grpcQuorumCertificate) + + // mock FlowTimeoutCertificate + val highQcViews = listOf(1L, 2L) + val signerIndices = byteArrayOf(1, 2) + val sigData = byteArrayOf(3, 4) + + val timeoutCertificate = FlowTimeoutCertificate( + view = view, + highQcViews = highQcViews, + highestQc = flowQuorumCertificate, + signerIndices = signerIndices, + sigData = sigData + ) + + val result = timeoutCertificate.builder().build() + + assertEquals(view, result.view) + assertEquals(highQcViews, result.highQcViewsList) + assertEquals(ByteString.copyFrom(signerIndices), result.signerIndices) + assertEquals(ByteString.copyFrom(sigData), result.sigData) + assertEquals(flowQuorumCertificate.builder().build(), result.highestQc) + } + + @Test + fun `test equals() method`() { + val view1 = 123L + val highQcViews1 = listOf(456L, 789L) + val signerIndices1 = ByteArray(32) { 0x01 } + val sigData1 = ByteArray(32) { 0x02 } + + val view2 = 124L + val highQcViews2 = listOf(111L, 222L) + val signerIndices2 = ByteArray(32) { 0x03 } + val sigData2 = ByteArray(32) { 0x04 } + + val highestQc1 = mock(FlowQuorumCertificate::class.java) + val highestQc2 = mock(FlowQuorumCertificate::class.java) + + val flowTimeoutCertificate1 = FlowTimeoutCertificate(view1, highQcViews1, highestQc1, signerIndices1, sigData1) + val flowTimeoutCertificate2 = FlowTimeoutCertificate(view1, highQcViews1, highestQc1, signerIndices1, sigData1) + val flowTimeoutCertificate3 = FlowTimeoutCertificate(view2, highQcViews2, highestQc2, signerIndices2, sigData2) + + assertEquals(flowTimeoutCertificate1, flowTimeoutCertificate2) + assertNotEquals(flowTimeoutCertificate1, flowTimeoutCertificate3) + } + + @Test + fun `test hashCode() method`() { + val view1 = 123L + val highQcViews1 = listOf(456L, 789L) + val signerIndices1 = ByteArray(32) { 0x01 } + val sigData1 = ByteArray(32) { 0x02 } + + val view2 = 124L + val highQcViews2 = listOf(111L, 222L) + val signerIndices2 = ByteArray(32) { 0x03 } + val sigData2 = ByteArray(32) { 0x04 } + + val highestQc1 = mock(FlowQuorumCertificate::class.java) + val highestQc2 = mock(FlowQuorumCertificate::class.java) + + val flowTimeoutCertificate1 = FlowTimeoutCertificate(view1, highQcViews1, highestQc1, signerIndices1, sigData1) + val flowTimeoutCertificate2 = FlowTimeoutCertificate(view1, highQcViews1, highestQc1, signerIndices1, sigData1) + val flowTimeoutCertificate3 = FlowTimeoutCertificate(view2, highQcViews2, highestQc2, signerIndices2, sigData2) + + assertEquals(flowTimeoutCertificate1.hashCode(), flowTimeoutCertificate2.hashCode()) + assertNotEquals(flowTimeoutCertificate1.hashCode(), flowTimeoutCertificate3.hashCode()) + } } + From 008da9f7333a3b53806206672786cbcd9d0c64e2 Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Tue, 15 Oct 2024 23:49:55 +0900 Subject: [PATCH 12/16] Setup tests for FlowChunkTest --- .../onflow/flow/sdk/models/FlowChunkTest.kt | 199 ++++++++++++++++++ 1 file changed, 199 insertions(+) diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt index f3ad7f9..a990888 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt @@ -1,4 +1,203 @@ package org.onflow.flow.sdk.models +import com.google.protobuf.ByteString +import org.junit.jupiter.api.Assertions.* +import org.junit.jupiter.api.Test +import org.mockito.Mockito.* +import org.onflow.flow.sdk.FlowChunk +import org.onflow.flow.sdk.FlowId +import org.onflow.protobuf.entities.ExecutionResultOuterClass + class FlowChunkTest { + @Test + fun `test of() method`() { + val collectionIndex = 1 + val startStateBytes = ByteArray(32) { 0x01 } + val eventCollectionBytes = ByteArray(32) { 0x02 } + val blockIdBytes = ByteArray(32) { 0x03 } + val totalComputationUsed = 1000L + val numberOfTransactions = 10 + val index = 5L + val endStateBytes = ByteArray(32) { 0x04 } + val executionDataIdBytes = ByteArray(32) { 0x05 } + val stateDeltaCommitmentBytes = ByteArray(32) { 0x06 } + + val grpcChunk = mock(ExecutionResultOuterClass.Chunk::class.java) + `when`(grpcChunk.collectionIndex).thenReturn(collectionIndex) + `when`(grpcChunk.startState).thenReturn(ByteString.copyFrom(startStateBytes)) + `when`(grpcChunk.eventCollection).thenReturn(ByteString.copyFrom(eventCollectionBytes)) + `when`(grpcChunk.blockId).thenReturn(ByteString.copyFrom(blockIdBytes)) + `when`(grpcChunk.totalComputationUsed).thenReturn(totalComputationUsed) + `when`(grpcChunk.numberOfTransactions).thenReturn(numberOfTransactions) + `when`(grpcChunk.index).thenReturn(index) + `when`(grpcChunk.endState).thenReturn(ByteString.copyFrom(endStateBytes)) + `when`(grpcChunk.executionDataId).thenReturn(ByteString.copyFrom(executionDataIdBytes)) + `when`(grpcChunk.stateDeltaCommitment).thenReturn(ByteString.copyFrom(stateDeltaCommitmentBytes)) + + val flowChunk = FlowChunk.of(grpcChunk) + + assertEquals(collectionIndex, flowChunk.collectionIndex) + assertArrayEquals(startStateBytes, flowChunk.startState) + assertArrayEquals(eventCollectionBytes, flowChunk.eventCollection) + assertEquals(FlowId.of(blockIdBytes), flowChunk.blockId) + assertEquals(totalComputationUsed, flowChunk.totalComputationUsed) + assertEquals(numberOfTransactions, flowChunk.numberOfTransactions) + assertEquals(index, flowChunk.index) + assertArrayEquals(endStateBytes, flowChunk.endState) + assertEquals(FlowId.of(executionDataIdBytes), flowChunk.executionDataId) + assertArrayEquals(stateDeltaCommitmentBytes, flowChunk.stateDeltaCommitment) + } + + @Test + fun `test builder() method`() { + val collectionIndex = 1 + val startStateBytes = ByteArray(32) { 0x01 } + val eventCollectionBytes = ByteArray(32) { 0x02 } + val blockId = FlowId.of(ByteArray(32) { 0x03 }) + val totalComputationUsed = 1000L + val numberOfTransactions = 10 + val index = 5L + val endStateBytes = ByteArray(32) { 0x04 } + val executionDataId = FlowId.of(ByteArray(32) { 0x05 }) + val stateDeltaCommitmentBytes = ByteArray(32) { 0x06 } + + val flowChunk = FlowChunk( + collectionIndex = collectionIndex, + startState = startStateBytes, + eventCollection = eventCollectionBytes, + blockId = blockId, + totalComputationUsed = totalComputationUsed, + numberOfTransactions = numberOfTransactions, + index = index, + endState = endStateBytes, + executionDataId = executionDataId, + stateDeltaCommitment = stateDeltaCommitmentBytes + ) + + val result = flowChunk.builder().build() + + assertEquals(collectionIndex, result.collectionIndex) + assertEquals(ByteString.copyFrom(startStateBytes), result.startState) + assertEquals(ByteString.copyFrom(eventCollectionBytes), result.eventCollection) + assertEquals(blockId.byteStringValue, result.blockId) + assertEquals(totalComputationUsed, result.totalComputationUsed) + assertEquals(numberOfTransactions, result.numberOfTransactions) + assertEquals(index, result.index) + assertEquals(ByteString.copyFrom(endStateBytes), result.endState) + assertEquals(executionDataId.byteStringValue, result.executionDataId) + assertEquals(ByteString.copyFrom(stateDeltaCommitmentBytes), result.stateDeltaCommitment) + } + + @Test + fun `test equals() method`() { + val collectionIndex1 = 1 + val collectionIndex2 = 2 + val startState1 = ByteArray(32) { 0x01 } + val startState2 = ByteArray(32) { 0x02 } + val blockId1 = FlowId.of(ByteArray(32) { 0x03 }) + val blockId2 = FlowId.of(ByteArray(32) { 0x04 }) + val totalComputationUsed1 = 1000L + val totalComputationUsed2 = 2000L + val endState1 = ByteArray(32) { 0x04 } + val endState2 = ByteArray(32) { 0x05 } + + val flowChunk1 = FlowChunk( + collectionIndex = collectionIndex1, + startState = startState1, + eventCollection = startState1, + blockId = blockId1, + totalComputationUsed = totalComputationUsed1, + numberOfTransactions = 10, + index = 5L, + endState = endState1, + executionDataId = blockId1, + stateDeltaCommitment = startState1 + ) + + val flowChunk2 = FlowChunk( + collectionIndex = collectionIndex1, + startState = startState1, + eventCollection = startState1, + blockId = blockId1, + totalComputationUsed = totalComputationUsed1, + numberOfTransactions = 10, + index = 5L, + endState = endState1, + executionDataId = blockId1, + stateDeltaCommitment = startState1 + ) + + val flowChunk3 = FlowChunk( + collectionIndex = collectionIndex2, + startState = startState2, + eventCollection = startState2, + blockId = blockId2, + totalComputationUsed = totalComputationUsed2, + numberOfTransactions = 20, + index = 10L, + endState = endState2, + executionDataId = blockId2, + stateDeltaCommitment = startState2 + ) + + assertEquals(flowChunk1, flowChunk2) + assertNotEquals(flowChunk1, flowChunk3) + } + + @Test + fun `test hashCode() method`() { + val collectionIndex1 = 1 + val collectionIndex2 = 2 + val startState1 = ByteArray(32) { 0x01 } + val startState2 = ByteArray(32) { 0x02 } + val blockId1 = FlowId.of(ByteArray(32) { 0x03 }) + val blockId2 = FlowId.of(ByteArray(32) { 0x04 }) + val totalComputationUsed1 = 1000L + val totalComputationUsed2 = 2000L + val endState1 = ByteArray(32) { 0x04 } + val endState2 = ByteArray(32) { 0x05 } + + val flowChunk1 = FlowChunk( + collectionIndex = collectionIndex1, + startState = startState1, + eventCollection = startState1, + blockId = blockId1, + totalComputationUsed = totalComputationUsed1, + numberOfTransactions = 10, + index = 5L, + endState = endState1, + executionDataId = blockId1, + stateDeltaCommitment = startState1 + ) + + val flowChunk2 = FlowChunk( + collectionIndex = collectionIndex1, + startState = startState1, + eventCollection = startState1, + blockId = blockId1, + totalComputationUsed = totalComputationUsed1, + numberOfTransactions = 10, + index = 5L, + endState = endState1, + executionDataId = blockId1, + stateDeltaCommitment = startState1 + ) + + val flowChunk3 = FlowChunk( + collectionIndex = collectionIndex2, + startState = startState2, + eventCollection = startState2, + blockId = blockId2, + totalComputationUsed = totalComputationUsed2, + numberOfTransactions = 20, + index = 10L, + endState = endState2, + executionDataId = blockId2, + stateDeltaCommitment = startState2 + ) + + assertEquals(flowChunk1.hashCode(), flowChunk2.hashCode()) + assertNotEquals(flowChunk1.hashCode(), flowChunk3.hashCode()) + } } + From bfcd2cde792272d8fce99e7c22d873193ee29af3 Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Wed, 16 Oct 2024 00:53:48 +0900 Subject: [PATCH 13/16] Setup tests for FlowExecutionReceiptMetaTest, FlowExecutionResultTest --- .../flow/sdk/models/FlowBlockHeaderTest.kt | 244 ++++++++++++++++++ .../models/FlowExecutionReceiptMetaTest.kt | 88 +++++++ .../sdk/models/FlowExecutionResultTest.kt | 227 ++++++++++++++++ 3 files changed, 559 insertions(+) diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt index 42a9676..9481fee 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt @@ -1,4 +1,248 @@ package org.onflow.flow.sdk.models +import com.google.protobuf.ByteString +import com.google.protobuf.UnsafeByteOperations +import org.junit.jupiter.api.Assertions.* +import org.junit.jupiter.api.Test +import org.mockito.Mockito.* +import org.onflow.flow.sdk.* +import org.onflow.protobuf.entities.BlockHeaderOuterClass +import java.time.LocalDateTime + class FlowBlockHeaderTest { + companion object { + val idBytes = ByteArray(32) { 0x01 } + val parentIdBytes = ByteArray(32) { 0x02 } + val proposerIdBytes = ByteArray(32) { 0x03 } + val payloadHashBytes = ByteArray(32) { 0x04 } + val voterSigDataBytes = ByteArray(32) { 0x05 } + val proposerSigDataBytes = ByteArray(32) { 0x06 } + val voterIndicesBytes = ByteArray(32) { 0x07 } + + val id = FlowId.of(idBytes) + val parentId = FlowId.of(parentIdBytes) + val proposerId = FlowId.of(proposerIdBytes) + val chainId = FlowChainId.of("mainnet") + val timestamp = LocalDateTime.now() + val timeoutCertificate = mock(FlowTimeoutCertificate::class.java) + } + @Test + fun `test of() method`() { + val chainId = "mainnet" + + // Mock block header + val grpcBlockHeader = mock(BlockHeaderOuterClass.BlockHeader::class.java) + `when`(grpcBlockHeader.id).thenReturn(ByteString.copyFrom(idBytes)) + `when`(grpcBlockHeader.parentId).thenReturn(ByteString.copyFrom(parentIdBytes)) + `when`(grpcBlockHeader.height).thenReturn(123L) + `when`(grpcBlockHeader.timestamp).thenReturn(timestamp.asTimestamp()) + `when`(grpcBlockHeader.payloadHash).thenReturn(ByteString.copyFrom(payloadHashBytes)) + `when`(grpcBlockHeader.view).thenReturn(456L) + `when`(grpcBlockHeader.parentVoterSigData).thenReturn(ByteString.copyFrom(voterSigDataBytes)) + `when`(grpcBlockHeader.proposerId).thenReturn(ByteString.copyFrom(proposerIdBytes)) + `when`(grpcBlockHeader.proposerSigData).thenReturn(ByteString.copyFrom(proposerSigDataBytes)) + `when`(grpcBlockHeader.chainId).thenReturn(chainId) + `when`(grpcBlockHeader.parentVoterIndices).thenReturn(ByteString.copyFrom(voterIndicesBytes)) + + // Mock timeout certificate and quorum certificate + val timeoutCertificateMock = mock(BlockHeaderOuterClass.TimeoutCertificate::class.java) + `when`(timeoutCertificateMock.signerIndices).thenReturn(ByteString.copyFrom(idBytes)) + `when`(timeoutCertificateMock.sigData).thenReturn(ByteString.copyFrom(idBytes)) + + val highestQcMock = mock(BlockHeaderOuterClass.QuorumCertificate::class.java) + `when`(highestQcMock.blockId).thenReturn(ByteString.copyFrom(idBytes)) + `when`(highestQcMock.signerIndices).thenReturn(ByteString.copyFrom(idBytes)) + `when`(highestQcMock.sigData).thenReturn(ByteString.copyFrom(idBytes)) + + `when`(grpcBlockHeader.lastViewTc).thenReturn(timeoutCertificateMock) + `when`(timeoutCertificateMock.highestQc).thenReturn(highestQcMock) + `when`(grpcBlockHeader.parentView).thenReturn(789L) + + val flowBlockHeader = FlowBlockHeader.of(grpcBlockHeader) + + assertEquals(FlowId.of(idBytes), flowBlockHeader.id) + assertEquals(FlowId.of(parentIdBytes), flowBlockHeader.parentId) + assertEquals(123L, flowBlockHeader.height) + assertEquals(timestamp, flowBlockHeader.timestamp) + assertArrayEquals(payloadHashBytes, flowBlockHeader.payloadHash) + assertEquals(456L, flowBlockHeader.view) + assertArrayEquals(voterSigDataBytes, flowBlockHeader.parentVoterSigData) + assertEquals(FlowId.of(proposerIdBytes), flowBlockHeader.proposerId) + assertArrayEquals(proposerSigDataBytes, flowBlockHeader.proposerSigData) + assertEquals(FlowChainId.of(chainId), flowBlockHeader.chainId) + assertArrayEquals(voterIndicesBytes, flowBlockHeader.parentVoterIndices) + assertEquals(789L, flowBlockHeader.parentView) + } + + @Test + fun `test builder() method`() { + // mock FlowQuorumCertificate + val view = 123L + val blockIdBytes = ByteArray(32) { 0x01 } + val signerIndicesBytes = ByteArray(32) { 0x02 } + val sigDataBytes = ByteArray(32) { 0x03 } + + val grpcQuorumCertificate = mock(BlockHeaderOuterClass.QuorumCertificate::class.java) + `when`(grpcQuorumCertificate.view).thenReturn(view) + `when`(grpcQuorumCertificate.blockId).thenReturn(ByteString.copyFrom(blockIdBytes)) + `when`(grpcQuorumCertificate.signerIndices).thenReturn(ByteString.copyFrom(signerIndicesBytes)) + `when`(grpcQuorumCertificate.sigData).thenReturn(ByteString.copyFrom(sigDataBytes)) + + val flowQuorumCertificate = FlowQuorumCertificate.of(grpcQuorumCertificate) + + // mock FlowTimeoutCertificate + val highQcViews = listOf(1L, 2L) + val signerIndices = byteArrayOf(1, 2) + val sigData = byteArrayOf(3, 4) + + val timeoutCertificate = FlowTimeoutCertificate( + view = view, + highQcViews = highQcViews, + highestQc = flowQuorumCertificate, + signerIndices = signerIndices, + sigData = sigData + ) + + // Create FlowBlockHeader + val flowBlockHeader = FlowBlockHeader( + id = id, + parentId = parentId, + height = 123L, + timestamp = timestamp, + payloadHash = payloadHashBytes, + view = 456L, + parentVoterSigData = voterSigDataBytes, + proposerId = proposerId, + proposerSigData = proposerSigDataBytes, + chainId = chainId, + parentVoterIndices = voterIndicesBytes, + lastViewTc = timeoutCertificate, + parentView = 789L + ) + + val result = flowBlockHeader.builder().build() + + assertEquals(id.byteStringValue, result.id) + assertEquals(parentId.byteStringValue, result.parentId) + assertEquals(123L, result.height) + assertEquals(timestamp.asTimestamp(), result.timestamp) + assertEquals(UnsafeByteOperations.unsafeWrap(payloadHashBytes), result.payloadHash) + assertEquals(456L, result.view) + assertEquals(UnsafeByteOperations.unsafeWrap(voterSigDataBytes), result.parentVoterSigData) + assertEquals(proposerId.byteStringValue, result.proposerId) + assertEquals(UnsafeByteOperations.unsafeWrap(proposerSigDataBytes), result.proposerSigData) + assertEquals(chainId.id, result.chainId) + assertEquals(UnsafeByteOperations.unsafeWrap(voterIndicesBytes), result.parentVoterIndices) + assertEquals(789L, result.parentView) + } + + @Test + fun `test equals() method`() { + val flowBlockHeader1 = FlowBlockHeader( + id = id, + parentId = parentId, + height = 123L, + timestamp = timestamp, + payloadHash = payloadHashBytes, + view = 456L, + parentVoterSigData = voterSigDataBytes, + proposerId = proposerId, + proposerSigData = proposerSigDataBytes, + chainId = chainId, + parentVoterIndices = voterIndicesBytes, + lastViewTc = timeoutCertificate, + parentView = 789L + ) + + val flowBlockHeader2 = FlowBlockHeader( + id = id, + parentId = parentId, + height = 123L, + timestamp = timestamp, + payloadHash = payloadHashBytes, + view = 456L, + parentVoterSigData = voterSigDataBytes, + proposerId = proposerId, + proposerSigData = proposerSigDataBytes, + chainId = chainId, + parentVoterIndices = voterIndicesBytes, + lastViewTc = timeoutCertificate, + parentView = 789L + ) + + assertEquals(flowBlockHeader1, flowBlockHeader2) + + val flowBlockHeader3 = FlowBlockHeader( + id = FlowId.of(ByteArray(32) { 0x08 }), + parentId = parentId, + height = 124L, + timestamp = timestamp.plusDays(1), + payloadHash = ByteArray(32) { 0x09 }, + view = 789L, + parentVoterSigData = voterSigDataBytes, + proposerId = proposerId, + proposerSigData = proposerSigDataBytes, + chainId = chainId, + parentVoterIndices = voterIndicesBytes, + lastViewTc = timeoutCertificate, + parentView = 100L + ) + + assertNotEquals(flowBlockHeader1, flowBlockHeader3) + } + + @Test + fun `test hashCode() method`() { + val flowBlockHeader1 = FlowBlockHeader( + id = id, + parentId = parentId, + height = 123L, + timestamp = timestamp, + payloadHash = payloadHashBytes, + view = 456L, + parentVoterSigData = voterSigDataBytes, + proposerId = proposerId, + proposerSigData = proposerSigDataBytes, + chainId = chainId, + parentVoterIndices = voterIndicesBytes, + lastViewTc = timeoutCertificate, + parentView = 789L + ) + + val flowBlockHeader2 = FlowBlockHeader( + id = id, + parentId = parentId, + height = 123L, + timestamp = timestamp, + payloadHash = payloadHashBytes, + view = 456L, + parentVoterSigData = voterSigDataBytes, + proposerId = proposerId, + proposerSigData = proposerSigDataBytes, + chainId = chainId, + parentVoterIndices = voterIndicesBytes, + lastViewTc = timeoutCertificate, + parentView = 789L + ) + + assertEquals(flowBlockHeader1.hashCode(), flowBlockHeader2.hashCode()) + + val flowBlockHeader3 = FlowBlockHeader( + id = FlowId.of(ByteArray(32) { 0x08 }), + parentId = parentId, + height = 124L, + timestamp = timestamp.plusDays(1), + payloadHash = ByteArray(32) { 0x09 }, + view = 789L, + parentVoterSigData = voterSigDataBytes, + proposerId = proposerId, + proposerSigData = proposerSigDataBytes, + chainId = chainId, + parentVoterIndices = voterIndicesBytes, + lastViewTc = timeoutCertificate, + parentView = 100L + ) + + assertNotEquals(flowBlockHeader1.hashCode(), flowBlockHeader3.hashCode()) + } } diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionReceiptMetaTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionReceiptMetaTest.kt index 2a0731f..fc40ed6 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionReceiptMetaTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionReceiptMetaTest.kt @@ -1,4 +1,92 @@ package org.onflow.flow.sdk.models +import com.google.protobuf.ByteString +import org.junit.jupiter.api.Assertions.* +import org.junit.jupiter.api.Test +import org.mockito.Mockito.* +import org.onflow.flow.sdk.FlowExecutionReceiptMeta +import org.onflow.flow.sdk.FlowId +import org.onflow.flow.sdk.FlowSignature +import org.onflow.protobuf.entities.ExecutionResultOuterClass + class FlowExecutionReceiptMetaTest { + @Test + fun `test of() method`() { + val executorIdBytes = ByteArray(32) { 0x01 } + val resultIdBytes = ByteArray(32) { 0x02 } + val spocksBytes = listOf(ByteArray(32) { 0x03 }, ByteArray(32) { 0x04 }) + val executorSignatureBytes = ByteArray(32) { 0x05 } + + val grpcMeta = mock(ExecutionResultOuterClass.ExecutionReceiptMeta::class.java) + `when`(grpcMeta.executorId).thenReturn(ByteString.copyFrom(executorIdBytes)) + `when`(grpcMeta.resultId).thenReturn(ByteString.copyFrom(resultIdBytes)) + `when`(grpcMeta.spocksList).thenReturn(spocksBytes.map { ByteString.copyFrom(it) }) + `when`(grpcMeta.executorSignature).thenReturn(ByteString.copyFrom(executorSignatureBytes)) + + val flowMeta = FlowExecutionReceiptMeta.of(grpcMeta) + + assertEquals(FlowId.of(executorIdBytes), flowMeta.executorId) + assertEquals(FlowId.of(resultIdBytes), flowMeta.resultId) + assertArrayEquals(spocksBytes[0], flowMeta.spocks[0]) + assertArrayEquals(spocksBytes[1], flowMeta.spocks[1]) + assertEquals(FlowSignature(executorSignatureBytes), flowMeta.executorSignature) + } + + @Test + fun `test builder() method`() { + val executorId = FlowId.of(ByteArray(32) { 0x01 }) + val resultId = FlowId.of(ByteArray(32) { 0x02 }) + val spocks = listOf(ByteArray(32) { 0x03 }, ByteArray(32) { 0x04 }) + val executorSignature = FlowSignature(ByteArray(32) { 0x05 }) + + val flowMeta = FlowExecutionReceiptMeta(executorId, resultId, spocks, executorSignature) + val result = flowMeta.builder().build() + + assertEquals(executorId.byteStringValue, result.executorId) + assertEquals(resultId.byteStringValue, result.resultId) + assertEquals(spocks.size, result.spocksCount) + assertEquals(ByteString.copyFrom(spocks[0]), result.spocksList[0]) + assertEquals(ByteString.copyFrom(spocks[1]), result.spocksList[1]) + assertEquals(executorSignature.byteStringValue, result.executorSignature) + } + + @Test + fun `test equals() method`() { + val executorId1 = FlowId.of(ByteArray(32) { 0x01 }) + val resultId1 = FlowId.of(ByteArray(32) { 0x02 }) + val spocks1 = listOf(ByteArray(32) { 0x03 }) + val executorSignature1 = FlowSignature(ByteArray(32) { 0x04 }) + + val executorId2 = FlowId.of(ByteArray(32) { 0x05 }) + val resultId2 = FlowId.of(ByteArray(32) { 0x06 }) + val spocks2 = listOf(ByteArray(32) { 0x07 }) + val executorSignature2 = FlowSignature(ByteArray(32) { 0x08 }) + + val meta1 = FlowExecutionReceiptMeta(executorId1, resultId1, spocks1, executorSignature1) + val meta2 = FlowExecutionReceiptMeta(executorId1, resultId1, spocks1, executorSignature1) + val meta3 = FlowExecutionReceiptMeta(executorId2, resultId2, spocks2, executorSignature2) + + assertEquals(meta1, meta2) + assertNotEquals(meta1, meta3) + } + + @Test + fun `test hashCode() method`() { + val executorId1 = FlowId.of(ByteArray(32) { 0x01 }) + val resultId1 = FlowId.of(ByteArray(32) { 0x02 }) + val spocks1 = listOf(ByteArray(32) { 0x03 }) + val executorSignature1 = FlowSignature(ByteArray(32) { 0x04 }) + + val executorId2 = FlowId.of(ByteArray(32) { 0x05 }) + val resultId2 = FlowId.of(ByteArray(32) { 0x06 }) + val spocks2 = listOf(ByteArray(32) { 0x07 }) + val executorSignature2 = FlowSignature(ByteArray(32) { 0x08 }) + + val meta1 = FlowExecutionReceiptMeta(executorId1, resultId1, spocks1, executorSignature1) + val meta2 = FlowExecutionReceiptMeta(executorId1, resultId1, spocks1, executorSignature1) + val meta3 = FlowExecutionReceiptMeta(executorId2, resultId2, spocks2, executorSignature2) + + assertEquals(meta1.hashCode(), meta2.hashCode()) + assertNotEquals(meta1.hashCode(), meta3.hashCode()) + } } diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt index 6584548..d366b16 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt @@ -1,4 +1,231 @@ package org.onflow.flow.sdk.models +import com.google.protobuf.ByteString +import org.junit.jupiter.api.Assertions.* +import org.junit.jupiter.api.Test +import org.mockito.Mockito.* +import org.onflow.flow.sdk.* +import org.onflow.protobuf.access.Access +import org.onflow.protobuf.entities.ExecutionResultOuterClass + class FlowExecutionResultTest { + companion object { + val collectionIndex = 1 + val startStateBytes = ByteArray(32) { 0x01 } + val eventCollectionBytes = ByteArray(32) { 0x02 } + val totalComputationUsed = 1000L + val numberOfTransactions = 10 + val index = 5L + val endStateBytes = ByteArray(32) { 0x04 } + val executionDataIdBytes = ByteArray(32) { 0x05 } + val stateDeltaCommitmentBytes = ByteArray(32) { 0x06 } + val payloadBytes = ByteArray(32) { 0x07 } + + val blockIdBytes = ByteArray(32) { 0x01 } + val previousResultIdBytes = ByteArray(32) { 0x02 } + } + + @Test + fun `test of() method with ExecutionResultByIDResponse`() { + // Mock inner ExecutionResult + val grpcExecutionResultInner = mock(ExecutionResultOuterClass.ExecutionResult::class.java) + `when`(grpcExecutionResultInner.blockId).thenReturn(ByteString.copyFrom(blockIdBytes)) + `when`(grpcExecutionResultInner.previousResultId).thenReturn(ByteString.copyFrom(previousResultIdBytes)) + + // Mock flow chunk + val grpcChunk = mock(ExecutionResultOuterClass.Chunk::class.java) + `when`(grpcChunk.collectionIndex).thenReturn(collectionIndex) + `when`(grpcChunk.startState).thenReturn(ByteString.copyFrom(startStateBytes)) + `when`(grpcChunk.eventCollection).thenReturn(ByteString.copyFrom(eventCollectionBytes)) + `when`(grpcChunk.blockId).thenReturn(ByteString.copyFrom(blockIdBytes)) + `when`(grpcChunk.totalComputationUsed).thenReturn(totalComputationUsed) + `when`(grpcChunk.numberOfTransactions).thenReturn(numberOfTransactions) + `when`(grpcChunk.index).thenReturn(index) + `when`(grpcChunk.endState).thenReturn(ByteString.copyFrom(endStateBytes)) + `when`(grpcChunk.executionDataId).thenReturn(ByteString.copyFrom(executionDataIdBytes)) + `when`(grpcChunk.stateDeltaCommitment).thenReturn(ByteString.copyFrom(stateDeltaCommitmentBytes)) + + `when`(grpcExecutionResultInner.chunksList).thenReturn(listOf(grpcChunk)) + + // Mock service event + val serviceEvent = mock(ExecutionResultOuterClass.ServiceEvent::class.java) + `when`(serviceEvent.type).thenReturn("EventType") // Mock the getType() method + `when`(serviceEvent.payload).thenReturn(ByteString.copyFrom(payloadBytes)) + `when`(grpcExecutionResultInner.serviceEventsList).thenReturn(listOf(serviceEvent)) + + // Mock outer Access.ExecutionResultByIDResponse + val grpcExecutionResult = mock(Access.ExecutionResultByIDResponse::class.java) + `when`(grpcExecutionResult.executionResult).thenReturn(grpcExecutionResultInner) + + val flowExecutionResult = FlowExecutionResult.of(grpcExecutionResult) + + assertEquals(FlowId.of(blockIdBytes), flowExecutionResult.blockId) + assertEquals(FlowId.of(previousResultIdBytes), flowExecutionResult.previousResultId) + assertEquals(1, flowExecutionResult.chunks.size) + assertEquals(1, flowExecutionResult.serviceEvents.size) + assertEquals("EventType", flowExecutionResult.serviceEvents[0].type) + assertArrayEquals(payloadBytes, flowExecutionResult.serviceEvents[0].payload) + } + + @Test + fun `test of() method with ExecutionResult`() { + // Mock flow chunk + val grpcChunk = mock(ExecutionResultOuterClass.Chunk::class.java) + `when`(grpcChunk.collectionIndex).thenReturn(collectionIndex) + `when`(grpcChunk.startState).thenReturn(ByteString.copyFrom(startStateBytes)) + `when`(grpcChunk.eventCollection).thenReturn(ByteString.copyFrom(eventCollectionBytes)) + `when`(grpcChunk.blockId).thenReturn(ByteString.copyFrom(blockIdBytes)) + `when`(grpcChunk.totalComputationUsed).thenReturn(totalComputationUsed) + `when`(grpcChunk.numberOfTransactions).thenReturn(numberOfTransactions) + `when`(grpcChunk.index).thenReturn(index) + `when`(grpcChunk.endState).thenReturn(ByteString.copyFrom(endStateBytes)) + `when`(grpcChunk.executionDataId).thenReturn(ByteString.copyFrom(executionDataIdBytes)) + `when`(grpcChunk.stateDeltaCommitment).thenReturn(ByteString.copyFrom(stateDeltaCommitmentBytes)) + + // Mock service event + val serviceEvent = mock(ExecutionResultOuterClass.ServiceEvent::class.java) + `when`(serviceEvent.type).thenReturn("EventType") // Mock the getType() method + `when`(serviceEvent.payload).thenReturn(ByteString.copyFrom(payloadBytes)) // Mock the getPayload() method + + // Mock ExecutionResult + val grpcExecutionResult = mock(ExecutionResultOuterClass.ExecutionResult::class.java) + `when`(grpcExecutionResult.blockId).thenReturn(ByteString.copyFrom(blockIdBytes)) + `when`(grpcExecutionResult.previousResultId).thenReturn(ByteString.copyFrom(previousResultIdBytes)) + `when`(grpcExecutionResult.chunksList).thenReturn(listOf(grpcChunk)) + `when`(grpcExecutionResult.serviceEventsList).thenReturn(listOf(serviceEvent)) + + val flowExecutionResult = FlowExecutionResult.of(grpcExecutionResult) + + assertEquals(FlowId.of(blockIdBytes), flowExecutionResult.blockId) + assertEquals(FlowId.of(previousResultIdBytes), flowExecutionResult.previousResultId) + assertEquals(1, flowExecutionResult.chunks.size) + assertEquals(1, flowExecutionResult.serviceEvents.size) + assertEquals("EventType", flowExecutionResult.serviceEvents[0].type) // Validate the service event type + assertArrayEquals(payloadBytes, flowExecutionResult.serviceEvents[0].payload) // Validate the service event payload + } + + @Test + fun `test builder() method`() { + val blockIdBytes = ByteArray(32) { 0x01 } + val previousResultIdBytes = ByteArray(32) { 0x02 } + val blockId = FlowId.of(blockIdBytes) + val previousResultId = FlowId.of(previousResultIdBytes) + + // Mock FlowChunk and its builder + val chunk = mock(FlowChunk::class.java) + val chunkBuilder = mock(ExecutionResultOuterClass.Chunk.Builder::class.java) + + // Set up the behavior for the builder methods + `when`(chunk.builder()).thenReturn(chunkBuilder) + `when`(chunkBuilder.build()).thenReturn(ExecutionResultOuterClass.Chunk.newBuilder() + .setCollectionIndex(1) + .setBlockId(ByteString.copyFrom(blockIdBytes)) + .build()) // Return a valid Chunk object with required fields + + // Mock FlowServiceEvent and its builder + val serviceEvent = mock(FlowServiceEvent::class.java) + val serviceEventBuilder = mock(ExecutionResultOuterClass.ServiceEvent.Builder::class.java) + + // Set up the behavior for the service event builder + `when`(serviceEvent.builder()).thenReturn(serviceEventBuilder) + `when`(serviceEventBuilder.build()).thenReturn(ExecutionResultOuterClass.ServiceEvent.newBuilder() + .setType("EventType") + .setPayload(ByteString.copyFrom(ByteArray(32) { 0x07 })) // Use any valid ByteString payload + .build()) // Ensure this returns a valid built object + + // Create FlowExecutionResult instance with mocks + val flowExecutionResult = FlowExecutionResult( + blockId = blockId, + previousResultId = previousResultId, + chunks = listOf(chunk), + serviceEvents = listOf(serviceEvent) + ) + + // Call the builder() method and build the result + val result = flowExecutionResult.builder().build() + + // Assertions to verify the builder results + assertEquals(blockId.byteStringValue, result.blockId) + assertEquals(previousResultId.byteStringValue, result.previousResultId) + assertEquals(1, result.chunksCount) + assertEquals(1, result.serviceEventsCount) + + // Verify that the builders for the chunks and service events were invoked + verify(chunk).builder() + verify(serviceEvent).builder() + } + + @Test + fun `test equals() method`() { + val blockId1 = FlowId.of(ByteArray(32) { 0x01 }) + val previousResultId1 = FlowId.of(ByteArray(32) { 0x02 }) + val chunk1 = mock(FlowChunk::class.java) + val serviceEvent1 = mock(FlowServiceEvent::class.java) + + val blockId2 = FlowId.of(ByteArray(32) { 0x03 }) + val previousResultId2 = FlowId.of(ByteArray(32) { 0x04 }) + val chunk2 = mock(FlowChunk::class.java) + val serviceEvent2 = mock(FlowServiceEvent::class.java) + + val flowExecutionResult1 = FlowExecutionResult( + blockId = blockId1, + previousResultId = previousResultId1, + chunks = listOf(chunk1), + serviceEvents = listOf(serviceEvent1) + ) + + val flowExecutionResult2 = FlowExecutionResult( + blockId = blockId1, + previousResultId = previousResultId1, + chunks = listOf(chunk1), + serviceEvents = listOf(serviceEvent1) + ) + + val flowExecutionResult3 = FlowExecutionResult( + blockId = blockId2, + previousResultId = previousResultId2, + chunks = listOf(chunk2), + serviceEvents = listOf(serviceEvent2) + ) + + assertEquals(flowExecutionResult1, flowExecutionResult2) + assertNotEquals(flowExecutionResult1, flowExecutionResult3) + } + + @Test + fun `test hashCode() method`() { + val blockId1 = FlowId.of(ByteArray(32) { 0x01 }) + val previousResultId1 = FlowId.of(ByteArray(32) { 0x02 }) + val chunk1 = mock(FlowChunk::class.java) + val serviceEvent1 = mock(FlowServiceEvent::class.java) + + val blockId2 = FlowId.of(ByteArray(32) { 0x03 }) + val previousResultId2 = FlowId.of(ByteArray(32) { 0x04 }) + val chunk2 = mock(FlowChunk::class.java) + val serviceEvent2 = mock(FlowServiceEvent::class.java) + + val flowExecutionResult1 = FlowExecutionResult( + blockId = blockId1, + previousResultId = previousResultId1, + chunks = listOf(chunk1), + serviceEvents = listOf(serviceEvent1) + ) + + val flowExecutionResult2 = FlowExecutionResult( + blockId = blockId1, + previousResultId = previousResultId1, + chunks = listOf(chunk1), + serviceEvents = listOf(serviceEvent1) + ) + + val flowExecutionResult3 = FlowExecutionResult( + blockId = blockId2, + previousResultId = previousResultId2, + chunks = listOf(chunk2), + serviceEvents = listOf(serviceEvent2) + ) + + assertEquals(flowExecutionResult1.hashCode(), flowExecutionResult2.hashCode()) + assertNotEquals(flowExecutionResult1.hashCode(), flowExecutionResult3.hashCode()) + } } From 5988a580f7efa266487103e4b1c78e17a02c1b33 Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Wed, 16 Oct 2024 01:27:02 +0900 Subject: [PATCH 14/16] Lint --- .../flow/sdk/models/FlowBlockHeaderTest.kt | 4 +- .../sdk/models/FlowExecutionResultTest.kt | 71 +++++++------------ 2 files changed, 29 insertions(+), 46 deletions(-) diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt index 9481fee..dfc99bb 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt @@ -23,8 +23,8 @@ class FlowBlockHeaderTest { val parentId = FlowId.of(parentIdBytes) val proposerId = FlowId.of(proposerIdBytes) val chainId = FlowChainId.of("mainnet") - val timestamp = LocalDateTime.now() - val timeoutCertificate = mock(FlowTimeoutCertificate::class.java) + val timestamp: LocalDateTime = LocalDateTime.now() + val timeoutCertificate: FlowTimeoutCertificate = mock(FlowTimeoutCertificate::class.java) } @Test fun `test of() method`() { diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt index d366b16..f481c8d 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt @@ -49,7 +49,7 @@ class FlowExecutionResultTest { // Mock service event val serviceEvent = mock(ExecutionResultOuterClass.ServiceEvent::class.java) - `when`(serviceEvent.type).thenReturn("EventType") // Mock the getType() method + `when`(serviceEvent.type).thenReturn("EventType") `when`(serviceEvent.payload).thenReturn(ByteString.copyFrom(payloadBytes)) `when`(grpcExecutionResultInner.serviceEventsList).thenReturn(listOf(serviceEvent)) @@ -84,8 +84,8 @@ class FlowExecutionResultTest { // Mock service event val serviceEvent = mock(ExecutionResultOuterClass.ServiceEvent::class.java) - `when`(serviceEvent.type).thenReturn("EventType") // Mock the getType() method - `when`(serviceEvent.payload).thenReturn(ByteString.copyFrom(payloadBytes)) // Mock the getPayload() method + `when`(serviceEvent.type).thenReturn("EventType") + `when`(serviceEvent.payload).thenReturn(ByteString.copyFrom(payloadBytes)) // Mock ExecutionResult val grpcExecutionResult = mock(ExecutionResultOuterClass.ExecutionResult::class.java) @@ -100,59 +100,42 @@ class FlowExecutionResultTest { assertEquals(FlowId.of(previousResultIdBytes), flowExecutionResult.previousResultId) assertEquals(1, flowExecutionResult.chunks.size) assertEquals(1, flowExecutionResult.serviceEvents.size) - assertEquals("EventType", flowExecutionResult.serviceEvents[0].type) // Validate the service event type - assertArrayEquals(payloadBytes, flowExecutionResult.serviceEvents[0].payload) // Validate the service event payload + assertEquals("EventType", flowExecutionResult.serviceEvents[0].type) + assertArrayEquals(payloadBytes, flowExecutionResult.serviceEvents[0].payload) } @Test fun `test builder() method`() { - val blockIdBytes = ByteArray(32) { 0x01 } - val previousResultIdBytes = ByteArray(32) { 0x02 } - val blockId = FlowId.of(blockIdBytes) - val previousResultId = FlowId.of(previousResultIdBytes) - - // Mock FlowChunk and its builder - val chunk = mock(FlowChunk::class.java) - val chunkBuilder = mock(ExecutionResultOuterClass.Chunk.Builder::class.java) - - // Set up the behavior for the builder methods - `when`(chunk.builder()).thenReturn(chunkBuilder) - `when`(chunkBuilder.build()).thenReturn(ExecutionResultOuterClass.Chunk.newBuilder() - .setCollectionIndex(1) - .setBlockId(ByteString.copyFrom(blockIdBytes)) - .build()) // Return a valid Chunk object with required fields - - // Mock FlowServiceEvent and its builder - val serviceEvent = mock(FlowServiceEvent::class.java) - val serviceEventBuilder = mock(ExecutionResultOuterClass.ServiceEvent.Builder::class.java) - - // Set up the behavior for the service event builder - `when`(serviceEvent.builder()).thenReturn(serviceEventBuilder) - `when`(serviceEventBuilder.build()).thenReturn(ExecutionResultOuterClass.ServiceEvent.newBuilder() - .setType("EventType") - .setPayload(ByteString.copyFrom(ByteArray(32) { 0x07 })) // Use any valid ByteString payload - .build()) // Ensure this returns a valid built object - - // Create FlowExecutionResult instance with mocks + // Setup mock FlowServiceEvent + val serviceEvent = FlowServiceEvent("EventType", blockIdBytes) + + // Setup mock FlowChunk + val flowChunk = FlowChunk( + collectionIndex = collectionIndex, + startState = startStateBytes, + eventCollection = eventCollectionBytes, + blockId = FlowId.of(blockIdBytes), + totalComputationUsed = totalComputationUsed, + numberOfTransactions = numberOfTransactions, + index = index, + endState = endStateBytes, + executionDataId = FlowId.of(blockIdBytes), + stateDeltaCommitment = stateDeltaCommitmentBytes + ) + val flowExecutionResult = FlowExecutionResult( - blockId = blockId, - previousResultId = previousResultId, - chunks = listOf(chunk), + blockId = FlowId.of(blockIdBytes), + previousResultId = FlowId.of(previousResultIdBytes), + chunks = listOf(flowChunk), serviceEvents = listOf(serviceEvent) ) - // Call the builder() method and build the result val result = flowExecutionResult.builder().build() - // Assertions to verify the builder results - assertEquals(blockId.byteStringValue, result.blockId) - assertEquals(previousResultId.byteStringValue, result.previousResultId) + assertEquals(FlowId.of(blockIdBytes).byteStringValue, result.blockId) + assertEquals(FlowId.of(previousResultIdBytes).byteStringValue, result.previousResultId) assertEquals(1, result.chunksCount) assertEquals(1, result.serviceEventsCount) - - // Verify that the builders for the chunks and service events were invoked - verify(chunk).builder() - verify(serviceEvent).builder() } @Test From b2e9ff01a43bbe05bfac4a4a1090efcd27eb4a2a Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Wed, 16 Oct 2024 01:27:41 +0900 Subject: [PATCH 15/16] Lint --- sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt | 1 - .../flow/sdk/impl/AsyncFlowAccessApiImplTest.kt | 1 - .../org/onflow/flow/sdk/models/FlowBlockTest.kt | 14 +++++++------- .../org/onflow/flow/sdk/models/FlowChunkTest.kt | 1 - .../flow/sdk/models/FlowQuorumCertificateTest.kt | 2 -- .../flow/sdk/models/FlowTimeoutCertificateTest.kt | 2 -- 6 files changed, 7 insertions(+), 14 deletions(-) diff --git a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt index 83be771..1c22d0b 100644 --- a/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt +++ b/sdk/src/main/kotlin/org/onflow/flow/sdk/models.kt @@ -758,7 +758,6 @@ data class FlowBlock( .addAllExecutionResultList(executionResultList.map { it.builder().build() }) .setBlockHeader(blockHeader.builder().build()) .setProtocolStateId(protocolStateId.byteStringValue) - } data class FlowChunk( diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/impl/AsyncFlowAccessApiImplTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/impl/AsyncFlowAccessApiImplTest.kt index 11d6798..2a1da42 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/impl/AsyncFlowAccessApiImplTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/impl/AsyncFlowAccessApiImplTest.kt @@ -58,7 +58,6 @@ class AsyncFlowAccessApiImplTest { ), parentView = 1L ) - } private fun setupFutureMock(response: T): ListenableFuture { diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockTest.kt index 6728991..251d87f 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockTest.kt @@ -24,11 +24,12 @@ class FlowBlockTest { .setTimestamp(timestamp) .setBlockHeader( BlockHeaderOuterClass.BlockHeader.newBuilder() - .setId(ByteString.copyFromUtf8("header_id")) - .setParentId(ByteString.copyFromUtf8("header_parent_id")) - .setHeight(124) - .setTimestamp(timestamp) - .build()) + .setId(ByteString.copyFromUtf8("header_id")) + .setParentId(ByteString.copyFromUtf8("header_parent_id")) + .setHeight(124) + .setTimestamp(timestamp) + .build() + ) .setProtocolStateId(ByteString.copyFromUtf8("protocol_state_id")) val flowBlock = FlowBlock.of(blockBuilder.build()) @@ -91,7 +92,7 @@ class FlowBlockTest { assert(blockBuilder.id.toByteArray().contentEquals(fixedSize("id".toByteArray(), 32))) assert(blockBuilder.parentId.toByteArray().contentEquals(fixedSize("parent_id".toByteArray(), 32))) assert(blockBuilder.height == 123L) - assert(blockBuilder.timestamp == flowBlock.timestamp.asTimestamp()) + assert(blockBuilder.timestamp.equals(flowBlock.timestamp.asTimestamp())) assert(blockBuilder.collectionGuaranteesList.isEmpty()) assert(blockBuilder.blockSealsList.isEmpty()) assert(blockBuilder.signaturesList.isEmpty()) @@ -99,4 +100,3 @@ class FlowBlockTest { assert(blockBuilder.protocolStateId.toByteArray().contentEquals(fixedSize("protocol_state_id".toByteArray(), 32))) } } - diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt index a990888..228dd7e 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowChunkTest.kt @@ -200,4 +200,3 @@ class FlowChunkTest { assertNotEquals(flowChunk1.hashCode(), flowChunk3.hashCode()) } } - diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt index ab5cbb3..2fd4405 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowQuorumCertificateTest.kt @@ -9,7 +9,6 @@ import org.onflow.flow.sdk.FlowQuorumCertificate import org.onflow.protobuf.entities.BlockHeaderOuterClass class FlowQuorumCertificateTest { - @Test fun `test of() method`() { val view = 123L @@ -88,4 +87,3 @@ class FlowQuorumCertificateTest { assertNotEquals(flowQuorumCertificate1.hashCode(), flowQuorumCertificate3.hashCode()) } } - diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt index b0ae891..744f866 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt @@ -44,7 +44,6 @@ class FlowTimeoutCertificateTest { @Test fun `test builder() method`() { - // mock FlowQuorumCertificate val view = 123L val blockIdBytes = ByteArray(32) { 0x01 } @@ -127,4 +126,3 @@ class FlowTimeoutCertificateTest { assertNotEquals(flowTimeoutCertificate1.hashCode(), flowTimeoutCertificate3.hashCode()) } } - From 4636c3d5457782b6431438b8e9cd4aebbfca34ae Mon Sep 17 00:00:00 2001 From: Lea Lobanov Date: Wed, 16 Oct 2024 01:32:38 +0900 Subject: [PATCH 16/16] Lint --- .../kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt | 5 ++--- .../org/onflow/flow/sdk/models/FlowExecutionResultTest.kt | 2 -- .../org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt | 4 ++-- 3 files changed, 4 insertions(+), 7 deletions(-) diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt index dfc99bb..6c4314c 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowBlockHeaderTest.kt @@ -76,7 +76,7 @@ class FlowBlockHeaderTest { @Test fun `test builder() method`() { - // mock FlowQuorumCertificate + // Mock FlowQuorumCertificate val view = 123L val blockIdBytes = ByteArray(32) { 0x01 } val signerIndicesBytes = ByteArray(32) { 0x02 } @@ -90,7 +90,7 @@ class FlowBlockHeaderTest { val flowQuorumCertificate = FlowQuorumCertificate.of(grpcQuorumCertificate) - // mock FlowTimeoutCertificate + // Mock FlowTimeoutCertificate val highQcViews = listOf(1L, 2L) val signerIndices = byteArrayOf(1, 2) val sigData = byteArrayOf(3, 4) @@ -103,7 +103,6 @@ class FlowBlockHeaderTest { sigData = sigData ) - // Create FlowBlockHeader val flowBlockHeader = FlowBlockHeader( id = id, parentId = parentId, diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt index f481c8d..c19d954 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowExecutionResultTest.kt @@ -106,10 +106,8 @@ class FlowExecutionResultTest { @Test fun `test builder() method`() { - // Setup mock FlowServiceEvent val serviceEvent = FlowServiceEvent("EventType", blockIdBytes) - // Setup mock FlowChunk val flowChunk = FlowChunk( collectionIndex = collectionIndex, startState = startStateBytes, diff --git a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt index 744f866..8ffb319 100644 --- a/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt +++ b/sdk/src/test/kotlin/org/onflow/flow/sdk/models/FlowTimeoutCertificateTest.kt @@ -44,7 +44,7 @@ class FlowTimeoutCertificateTest { @Test fun `test builder() method`() { - // mock FlowQuorumCertificate + // Mock FlowQuorumCertificate val view = 123L val blockIdBytes = ByteArray(32) { 0x01 } val signerIndicesBytes = ByteArray(32) { 0x02 } @@ -58,7 +58,7 @@ class FlowTimeoutCertificateTest { val flowQuorumCertificate = FlowQuorumCertificate.of(grpcQuorumCertificate) - // mock FlowTimeoutCertificate + // Mock FlowTimeoutCertificate val highQcViews = listOf(1L, 2L) val signerIndices = byteArrayOf(1, 2) val sigData = byteArrayOf(3, 4)