From edbb6afb2c9162fb5c9be6ecdad1b778703bfe80 Mon Sep 17 00:00:00 2001 From: Alexander Kalankhodzhaev Date: Fri, 11 Feb 2022 21:14:20 +0300 Subject: [PATCH] Rpc contracts --- build.gradle | 2 + common/build.gradle | 2 +- rpc/rpc-codegen/build.gradle | 6 +- .../decoder/RpcDecoderAnnotatedClass.java | 20 ++--- .../encoder/RpcEncoderAnnotatedClass.java | 6 +- .../codegen/sections/RpcMethodProcessor.java | 2 +- .../sections/RpcSubscriptionProcessor.java | 3 +- .../decoder/RpcDecoderProcessorTests.java | 57 +++++++++---- .../sections/RpcSectionFactoryTest.java | 4 +- .../codegen/substitutes/TestDecodable.java | 3 +- rpc/rpc-core/build.gradle | 3 +- .../substrateclient/rpc/core/RpcDecoder.java | 4 +- .../rpc/core/decoders/AbstractDecoder.java | 10 +-- .../rpc/core/decoders/BooleanDecoder.java | 5 +- .../rpc/core/decoders/ByteDecoder.java | 5 +- .../rpc/core/decoders/DoubleDecoder.java | 5 +- .../rpc/core/decoders/FloatDecoder.java | 5 +- .../rpc/core/decoders/IntDecoder.java | 5 +- .../rpc/core/decoders/ListDecoder.java | 10 ++- .../rpc/core/decoders/LongDecoder.java | 5 +- .../rpc/core/decoders/MapDecoder.java | 19 +++-- .../rpc/core/decoders/ShortDecoder.java | 5 +- .../rpc/core/decoders/StringDecoder.java | 5 +- .../rpc/core/decoders/VoidDecoder.java | 3 +- .../rpc/core/encoders/ListEncoder.java | 3 +- .../rpc/core/encoders/MapEncoder.java | 2 +- .../rpc/core/decoders/KnownDecoderTests.java | 63 +++++++++------ rpc/rpc-sections/build.gradle | 8 +- rpc/rpc-types/build.gradle | 2 + .../rpc/types/ExtrinsicStatusRpcDecoder.java | 12 +-- .../rpc/types/NumberDecoder.java | 5 +- .../rpc/types/PairDecoder.java | 9 +-- .../rpc/types/StorageDataDecoder.java | 5 +- .../rpc/types/StorageKeyDecoder.java | 5 +- scale/scale-codegen/build.gradle | 4 +- tests/build.gradle | 2 +- transport/build.gradle | 12 +-- .../transport/ProviderInterface.java | 6 +- .../substrateclient/transport/RpcBoolean.java | 20 +++++ .../substrateclient/transport/RpcList.java | 22 ++++++ .../substrateclient/transport/RpcMap.java | 22 ++++++ .../substrateclient/transport/RpcNull.java | 61 ++++++++++++++ .../substrateclient/transport/RpcNumber.java | 20 +++++ .../substrateclient/transport/RpcObject.java | 79 +++++++++++++++++++ .../substrateclient/transport/RpcString.java | 20 +++++ .../transport/SubscriptionHandler.java | 2 +- ...sonRpcObject.java => JsonRpcContract.java} | 2 +- .../transport/coder/JsonRpcRequest.java | 2 +- .../transport/coder/JsonRpcResponse.java | 7 +- .../coder/JsonRpcResponseSingle.java | 9 ++- .../coder/JsonRpcResponseSubscription.java | 9 ++- .../transport/coder/RpcCoder.java | 6 +- .../coder/RpcObjectDeserializer.java | 55 +++++++++++++ .../transport/ws/WsProvider.java | 44 +++++------ .../transport/coder/RpcCoderTest.java | 65 +++++++++++++++ .../transport/ws/WsProviderTest.java | 4 +- 56 files changed, 612 insertions(+), 169 deletions(-) create mode 100644 transport/src/main/java/com/strategyobject/substrateclient/transport/RpcBoolean.java create mode 100644 transport/src/main/java/com/strategyobject/substrateclient/transport/RpcList.java create mode 100644 transport/src/main/java/com/strategyobject/substrateclient/transport/RpcMap.java create mode 100644 transport/src/main/java/com/strategyobject/substrateclient/transport/RpcNull.java create mode 100644 transport/src/main/java/com/strategyobject/substrateclient/transport/RpcNumber.java create mode 100644 transport/src/main/java/com/strategyobject/substrateclient/transport/RpcObject.java create mode 100644 transport/src/main/java/com/strategyobject/substrateclient/transport/RpcString.java rename transport/src/main/java/com/strategyobject/substrateclient/transport/coder/{JsonRpcObject.java => JsonRpcContract.java} (90%) create mode 100644 transport/src/main/java/com/strategyobject/substrateclient/transport/coder/RpcObjectDeserializer.java create mode 100644 transport/src/test/java/com/strategyobject/substrateclient/transport/coder/RpcCoderTest.java diff --git a/build.gradle b/build.gradle index db9e5fa4..282c0800 100644 --- a/build.gradle +++ b/build.gradle @@ -28,8 +28,10 @@ subprojects { testImplementation 'org.junit.jupiter:junit-jupiter-api:5.7.2' testImplementation 'org.junit.jupiter:junit-jupiter-params:5.7.2' + testImplementation 'org.assertj:assertj-core:3.22.0' testImplementation 'org.mockito:mockito-core:3.12.4' testImplementation 'org.mockito:mockito-inline:3.12.4' + testImplementation 'org.slf4j:slf4j-simple:1.7.32' testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.7.2' } diff --git a/common/build.gradle b/common/build.gradle index 0a99a26a..d989c557 100644 --- a/common/build.gradle +++ b/common/build.gradle @@ -1,4 +1,4 @@ dependencies { - implementation 'org.reflections:reflections:0.10.1' + implementation 'org.reflections:reflections:0.10.2' implementation 'com.squareup:javapoet:1.13.0' } \ No newline at end of file diff --git a/rpc/rpc-codegen/build.gradle b/rpc/rpc-codegen/build.gradle index bd43b14d..30ab8855 100644 --- a/rpc/rpc-codegen/build.gradle +++ b/rpc/rpc-codegen/build.gradle @@ -5,12 +5,12 @@ dependencies { implementation project(':scale:scale-codegen') implementation project(':transport') - compileOnly 'com.google.auto.service:auto-service-annotations:1.0' - annotationProcessor 'com.google.auto.service:auto-service:1.0' + compileOnly 'com.google.auto.service:auto-service-annotations:1.0.1' + annotationProcessor 'com.google.auto.service:auto-service:1.0.1' implementation 'com.squareup:javapoet:1.13.0' testImplementation 'com.google.testing.compile:compile-testing:0.19' - testImplementation 'com.google.code.gson:gson:2.8.8' + testImplementation 'com.google.code.gson:gson:2.8.9' testCompileOnly project(':rpc:rpc-codegen') testAnnotationProcessor project(':rpc:rpc-codegen') diff --git a/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/decoder/RpcDecoderAnnotatedClass.java b/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/decoder/RpcDecoderAnnotatedClass.java index 6574764c..3602af64 100644 --- a/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/decoder/RpcDecoderAnnotatedClass.java +++ b/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/decoder/RpcDecoderAnnotatedClass.java @@ -16,6 +16,7 @@ import com.strategyobject.substrateclient.scale.codegen.ScaleAnnotationParser; import com.strategyobject.substrateclient.scale.codegen.reader.ReaderCompositor; import com.strategyobject.substrateclient.scale.registries.ScaleReaderRegistry; +import com.strategyobject.substrateclient.transport.RpcObject; import lombok.NonNull; import lombok.val; import lombok.var; @@ -83,7 +84,7 @@ private MethodSpec generateDecodeMethod(ProcessorContext context, TypeName class .addAnnotation(suppressWarnings("unchecked")) .addModifiers(Modifier.PUBLIC) .returns(classWildcardTyped) - .addParameter(Object.class, VALUE_ARG) + .addParameter(RpcObject.class, VALUE_ARG) .addParameter(ArrayTypeName.of( ParameterizedTypeName.get( ClassName.get(DecoderPair.class), @@ -91,6 +92,7 @@ private MethodSpec generateDecodeMethod(ProcessorContext context, TypeName class DECODERS_ARG) .varargs(true); + shortcutIfNull(methodSpec); addValidationRules(methodSpec); addMethodBody(methodSpec, context); @@ -100,10 +102,9 @@ private MethodSpec generateDecodeMethod(ProcessorContext context, TypeName class private void addMethodBody(MethodSpec.Builder methodSpec, ProcessorContext context) throws ProcessingException { val resultType = JavaPoet.setEachGenericParameterAs(classElement, TypeName.OBJECT); methodSpec - .addStatement("if ($L == null) { return null; }", VALUE_ARG) .addStatement("$1T $2L = $1T.getInstance()", RpcDecoderRegistry.class, DECODER_REGISTRY) .addStatement("$1T $2L = $1T.getInstance()", ScaleReaderRegistry.class, SCALE_READER_REGISTRY) - .addStatement("$1T<$2T, ?> $3L = ($1T<$2T, ?>)$4L", Map.class, String.class, MAP_VAR, VALUE_ARG) + .addStatement("$T<$T, $T> $L = $L.asMap()", Map.class, String.class, RpcObject.class, MAP_VAR, VALUE_ARG) .addStatement("$1T $2L = new $1T()", resultType, RESULT_VAR) .beginControlFlow("try"); @@ -166,13 +167,8 @@ private void setScaleField(MethodSpec.Builder methodSpec, readerCompositor.traverse(fieldType); methodSpec .addStatement(code - .add("$T.$L(", - ScaleUtils.class, - FROM_HEX_STRING) - .add("($T)$L.get($S), ", - String.class, - MAP_VAR, - field) + .add("$T.$L(", ScaleUtils.class, FROM_HEX_STRING) + .add("$L.get($S).asString(), ", MAP_VAR, field) .add("($T)", ScaleReader.class) .add(readerCode) .add("))") @@ -225,4 +221,8 @@ private void addValidationRules(MethodSpec.Builder methodSpec) { } } } + + private void shortcutIfNull(MethodSpec.Builder methodSpec) { + methodSpec.addStatement("if ($L.isNull()) { return null; }", VALUE_ARG); + } } diff --git a/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/encoder/RpcEncoderAnnotatedClass.java b/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/encoder/RpcEncoderAnnotatedClass.java index d395ba41..5b1c9276 100644 --- a/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/encoder/RpcEncoderAnnotatedClass.java +++ b/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/encoder/RpcEncoderAnnotatedClass.java @@ -92,6 +92,7 @@ private MethodSpec generateEncodeMethod(ProcessorContext context, TypeName class ENCODERS_ARG) .varargs(true); + shortcutIfNull(methodSpec); addValidationRules(methodSpec, context); addMethodBody(methodSpec, context); @@ -100,7 +101,6 @@ private MethodSpec generateEncodeMethod(ProcessorContext context, TypeName class private void addMethodBody(MethodSpec.Builder methodSpec, ProcessorContext context) throws ProcessingException { methodSpec - .addStatement("if ($L == null) { return null; }", SOURCE_ARG) .addStatement("$1T $2L = $1T.getInstance()", RpcEncoderRegistry.class, ENCODER_REGISTRY) .addStatement("$1T $2L = $1T.getInstance()", ScaleWriterRegistry.class, SCALE_WRITER_REGISTRY) .addStatement("$1T<$2T, $3T> $4L = new $1T<>()", HashMap.class, String.class, Object.class, RESULT_VAR) @@ -223,4 +223,8 @@ private void addValidationRules(MethodSpec.Builder methodSpec, ProcessorContext } } } + + private void shortcutIfNull(MethodSpec.Builder methodSpec) { + methodSpec.addStatement("if ($L == null) { return null; }", SOURCE_ARG); + } } diff --git a/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/sections/RpcMethodProcessor.java b/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/sections/RpcMethodProcessor.java index e7bf2d2b..75c2ab1d 100644 --- a/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/sections/RpcMethodProcessor.java +++ b/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/sections/RpcMethodProcessor.java @@ -158,7 +158,7 @@ private CodeBlock getScaleReadCodeBlock(AnnotatedConstruct annotated, return CodeBlock.builder() .add("($T) (", resultType) .add("$T.$L(", ScaleUtils.class, FROM_HEX_STRING) - .add("($T) $L, ($T) ", String.class, arg, ScaleReader.class) + .add("$L.asString(), ($T) ", arg, ScaleReader.class) .add(readerCode) .add("))") .build(); diff --git a/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/sections/RpcSubscriptionProcessor.java b/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/sections/RpcSubscriptionProcessor.java index 55e08242..5bf0f3b1 100644 --- a/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/sections/RpcSubscriptionProcessor.java +++ b/rpc/rpc-codegen/src/main/java/com/strategyobject/substrateclient/rpc/codegen/sections/RpcSubscriptionProcessor.java @@ -6,6 +6,7 @@ import com.strategyobject.substrateclient.common.codegen.ProcessingException; import com.strategyobject.substrateclient.common.codegen.ProcessorContext; import com.strategyobject.substrateclient.rpc.core.annotations.RpcSubscription; +import com.strategyobject.substrateclient.transport.RpcObject; import lombok.NonNull; import lombok.val; @@ -58,7 +59,7 @@ protected void callProviderInterface(MethodSpec.Builder methodSpecBuilder, .add("$1T<$2T, $3T> $4L = ($5L, $6L) -> { $7N.$8L($5L, ", BiConsumer.class, Exception.class, - Object.class, + RpcObject.class, CALL_BACK_PROXY, CALL_BACK_EX_ARG, CALL_BACK_ARG, diff --git a/rpc/rpc-codegen/src/test/java/com/strategyobject/substrateclient/rpc/codegen/decoder/RpcDecoderProcessorTests.java b/rpc/rpc-codegen/src/test/java/com/strategyobject/substrateclient/rpc/codegen/decoder/RpcDecoderProcessorTests.java index cb7e1e4c..698277e0 100644 --- a/rpc/rpc-codegen/src/test/java/com/strategyobject/substrateclient/rpc/codegen/decoder/RpcDecoderProcessorTests.java +++ b/rpc/rpc-codegen/src/test/java/com/strategyobject/substrateclient/rpc/codegen/decoder/RpcDecoderProcessorTests.java @@ -5,14 +5,12 @@ import com.strategyobject.substrateclient.rpc.codegen.substitutes.TestDecodable; import com.strategyobject.substrateclient.rpc.core.DecoderPair; import com.strategyobject.substrateclient.rpc.core.registries.RpcDecoderRegistry; +import com.strategyobject.substrateclient.transport.RpcObject; import lombok.val; import org.junit.jupiter.api.Test; -import java.util.AbstractMap; import java.util.Arrays; -import java.util.Map; -import java.util.stream.Collectors; -import java.util.stream.Stream; +import java.util.HashMap; import static com.google.testing.compile.CompilationSubject.assertThat; import static com.google.testing.compile.Compiler.javac; @@ -62,23 +60,50 @@ void compilesAndDecodes() { // TODO move this test out of the project val decoder = registry.resolve(TestDecodable.class) .inject(DecoderPair.of(registry.resolve(String.class), null)); - Object source = gson.fromJson("{\"a\":4,\"b\":\"123\",\"c\":\"some\"," + - "\"d\":[\"1\",\"2\"],\"e\":{\"a\":1,\"b\":2},\"f\":\"0x04000000\"," + - "\"g\":\"0x0c0500000002000000fdffffff\"}", - Object.class); + /* + { + "a": 4, + "b": "123", + "c": "some", + "d": [ + "1", + "2" + ], + "e": { + "a": null, + "b": 2 + }, + "f": "0x04000000", + "g": "0x0c0500000002000000fdffffff", + "h": 1.5, + } + */ + val source = RpcObject.of(new HashMap() {{ + put("a", RpcObject.of(4)); + put("b", RpcObject.of("123")); + put("c", RpcObject.of("some")); + put("d", RpcObject.of(Arrays.asList(RpcObject.of("1"), RpcObject.of("2")))); + put("e", RpcObject.of(new HashMap() {{ + put("a", RpcObject.ofNull()); + put("b", RpcObject.of(2)); + }})); + put("f", RpcObject.of("0x04000000")); + put("g", RpcObject.of("0x0c0500000002000000fdffffff")); + put("h", RpcObject.of(1.5)); + }}); + val actual = decoder.decode(source); + val expected = new TestDecodable<>(4, "123", "some", Arrays.asList("1", "2"), - Stream.of( - new AbstractMap.SimpleEntry<>("a", 1), - new AbstractMap.SimpleEntry<>("b", 2)) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue)), + new HashMap() {{ + put("a", null); + put("b", 2f); + }}, 4, - Arrays.asList(5, 2, -3)); - - val actual = decoder.decode(source); - + Arrays.asList(5, 2, -3), + 1.5f); assertEquals(gson.toJson(expected), gson.toJson(actual)); } } diff --git a/rpc/rpc-codegen/src/test/java/com/strategyobject/substrateclient/rpc/codegen/sections/RpcSectionFactoryTest.java b/rpc/rpc-codegen/src/test/java/com/strategyobject/substrateclient/rpc/codegen/sections/RpcSectionFactoryTest.java index ff10fd4d..dad4d714 100644 --- a/rpc/rpc-codegen/src/test/java/com/strategyobject/substrateclient/rpc/codegen/sections/RpcSectionFactoryTest.java +++ b/rpc/rpc-codegen/src/test/java/com/strategyobject/substrateclient/rpc/codegen/sections/RpcSectionFactoryTest.java @@ -2,6 +2,8 @@ import com.strategyobject.substrateclient.rpc.codegen.substitutes.TestSection; import com.strategyobject.substrateclient.transport.ProviderInterface; +import com.strategyobject.substrateclient.transport.RpcBoolean; +import com.strategyobject.substrateclient.transport.RpcObject; import lombok.val; import org.junit.jupiter.api.Test; @@ -19,7 +21,7 @@ public class RpcSectionFactoryTest { @Test void createsRpcSectionAndCallsMethod() throws ExecutionException, InterruptedException, RpcInterfaceInitializationException { val expected = true; - val sendFuture = CompletableFuture.completedFuture((Object) Boolean.valueOf(expected)); + val sendFuture = CompletableFuture.completedFuture((RpcObject) new RpcBoolean(expected)); val provider = mock(ProviderInterface.class); when(provider.send(anyString(), anyList())) .thenReturn(sendFuture); diff --git a/rpc/rpc-codegen/src/test/java/com/strategyobject/substrateclient/rpc/codegen/substitutes/TestDecodable.java b/rpc/rpc-codegen/src/test/java/com/strategyobject/substrateclient/rpc/codegen/substitutes/TestDecodable.java index b8495c84..4b19d390 100644 --- a/rpc/rpc-codegen/src/test/java/com/strategyobject/substrateclient/rpc/codegen/substitutes/TestDecodable.java +++ b/rpc/rpc-codegen/src/test/java/com/strategyobject/substrateclient/rpc/codegen/substitutes/TestDecodable.java @@ -20,9 +20,10 @@ public class TestDecodable { private String b; private T c; private List d; - private Map e; + private Map e; @Scale private int f; @Scale private List g; + private float h; } diff --git a/rpc/rpc-core/build.gradle b/rpc/rpc-core/build.gradle index 8678c1d5..658861ac 100644 --- a/rpc/rpc-core/build.gradle +++ b/rpc/rpc-core/build.gradle @@ -1,6 +1,7 @@ dependencies { implementation project(':common') implementation project(':scale') + implementation project(':transport') - testImplementation 'com.google.code.gson:gson:2.8.8' + testImplementation 'com.google.code.gson:gson:2.8.9' } \ No newline at end of file diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/RpcDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/RpcDecoder.java index 83ba1916..5e6c6249 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/RpcDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/RpcDecoder.java @@ -1,7 +1,9 @@ package com.strategyobject.substrateclient.rpc.core; +import com.strategyobject.substrateclient.transport.RpcObject; + public interface RpcDecoder { - T decode(Object value, DecoderPair... decoders); + T decode(RpcObject value, DecoderPair... decoders); default RpcDecoder inject(DecoderPair... dependencies) { return (jsonToken, decoders) -> this.decode(jsonToken, dependencies); diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/AbstractDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/AbstractDecoder.java index 31972a31..f4920b6d 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/AbstractDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/AbstractDecoder.java @@ -3,23 +3,23 @@ import com.google.common.base.Preconditions; import com.strategyobject.substrateclient.rpc.core.DecoderPair; import com.strategyobject.substrateclient.rpc.core.RpcDecoder; +import com.strategyobject.substrateclient.transport.RpcObject; public abstract class AbstractDecoder implements RpcDecoder { @Override - public final T decode(Object value, DecoderPair... decoders) { - if (value == null) { + public final T decode(RpcObject value, DecoderPair... decoders) { + if (value.isNull()) { return null; } - // TODO I am not sure should it be the first instruction of the method or not checkArguments(value, decoders); return decodeNonNull(value, decoders); } - protected abstract T decodeNonNull(Object value, DecoderPair[] decoders); + protected abstract T decodeNonNull(RpcObject value, DecoderPair[] decoders); - protected void checkArguments(Object value, DecoderPair[] decoders) { + protected void checkArguments(RpcObject value, DecoderPair[] decoders) { Preconditions.checkArgument(decoders == null || decoders.length == 0); } } diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/BooleanDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/BooleanDecoder.java index 368a7ae7..9a970dc1 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/BooleanDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/BooleanDecoder.java @@ -1,10 +1,11 @@ package com.strategyobject.substrateclient.rpc.core.decoders; import com.strategyobject.substrateclient.rpc.core.DecoderPair; +import com.strategyobject.substrateclient.transport.RpcObject; public class BooleanDecoder extends AbstractDecoder { @Override - protected Boolean decodeNonNull(Object value, DecoderPair[] decoders) { - return (Boolean) value; + protected Boolean decodeNonNull(RpcObject value, DecoderPair[] decoders) { + return value.asBoolean(); } } diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/ByteDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/ByteDecoder.java index 396f838f..1e0fb28d 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/ByteDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/ByteDecoder.java @@ -1,10 +1,11 @@ package com.strategyobject.substrateclient.rpc.core.decoders; import com.strategyobject.substrateclient.rpc.core.DecoderPair; +import com.strategyobject.substrateclient.transport.RpcObject; public class ByteDecoder extends AbstractDecoder { @Override - protected Byte decodeNonNull(Object value, DecoderPair[] decoders) { - return ((Double) value).byteValue(); + protected Byte decodeNonNull(RpcObject value, DecoderPair[] decoders) { + return value.asNumber().byteValue(); } } diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/DoubleDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/DoubleDecoder.java index 5a353511..bbb4058b 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/DoubleDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/DoubleDecoder.java @@ -1,10 +1,11 @@ package com.strategyobject.substrateclient.rpc.core.decoders; import com.strategyobject.substrateclient.rpc.core.DecoderPair; +import com.strategyobject.substrateclient.transport.RpcObject; public class DoubleDecoder extends AbstractDecoder { @Override - protected Double decodeNonNull(Object value, DecoderPair[] decoders) { - return (Double) value; + protected Double decodeNonNull(RpcObject value, DecoderPair[] decoders) { + return value.asNumber().doubleValue(); } } diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/FloatDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/FloatDecoder.java index f5602d37..09b19b0d 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/FloatDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/FloatDecoder.java @@ -1,10 +1,11 @@ package com.strategyobject.substrateclient.rpc.core.decoders; import com.strategyobject.substrateclient.rpc.core.DecoderPair; +import com.strategyobject.substrateclient.transport.RpcObject; public class FloatDecoder extends AbstractDecoder { @Override - protected Float decodeNonNull(Object value, DecoderPair[] decoders) { - return ((Double) value).floatValue(); + protected Float decodeNonNull(RpcObject value, DecoderPair[] decoders) { + return value.asNumber().floatValue(); } } diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/IntDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/IntDecoder.java index f9fcff14..4c6aafce 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/IntDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/IntDecoder.java @@ -1,10 +1,11 @@ package com.strategyobject.substrateclient.rpc.core.decoders; import com.strategyobject.substrateclient.rpc.core.DecoderPair; +import com.strategyobject.substrateclient.transport.RpcObject; public class IntDecoder extends AbstractDecoder { @Override - protected Integer decodeNonNull(Object value, DecoderPair[] decoders) { - return ((Double) value).intValue(); + protected Integer decodeNonNull(RpcObject value, DecoderPair[] decoders) { + return value.asNumber().intValue(); } } diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/ListDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/ListDecoder.java index 91956490..97e297da 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/ListDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/ListDecoder.java @@ -2,6 +2,7 @@ import com.google.common.base.Preconditions; import com.strategyobject.substrateclient.rpc.core.DecoderPair; +import com.strategyobject.substrateclient.transport.RpcObject; import lombok.val; import java.util.List; @@ -9,14 +10,17 @@ public class ListDecoder extends AbstractDecoder> { @Override - protected List decodeNonNull(Object value, DecoderPair[] decoders) { + protected List decodeNonNull(RpcObject value, DecoderPair[] decoders) { val nestedDecoder = decoders[0].getDecoderOrThrow(); - return ((List) value).stream().map(nestedDecoder::decode).collect(Collectors.toList()); + return value.asList() + .stream() + .map(nestedDecoder::decode) + .collect(Collectors.toList()); } @Override - protected void checkArguments(Object value, DecoderPair[] decoders) { + protected void checkArguments(RpcObject value, DecoderPair[] decoders) { Preconditions.checkArgument(decoders != null && decoders.length == 1); Preconditions.checkNotNull(decoders[0]); } diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/LongDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/LongDecoder.java index eac5ed16..bd9efa1e 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/LongDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/LongDecoder.java @@ -1,10 +1,11 @@ package com.strategyobject.substrateclient.rpc.core.decoders; import com.strategyobject.substrateclient.rpc.core.DecoderPair; +import com.strategyobject.substrateclient.transport.RpcObject; public class LongDecoder extends AbstractDecoder { @Override - protected Long decodeNonNull(Object value, DecoderPair[] decoders) { - return ((Double) value).longValue(); + protected Long decodeNonNull(RpcObject value, DecoderPair[] decoders) { + return value.asNumber().longValue(); } } diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/MapDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/MapDecoder.java index 60888a4a..3507df60 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/MapDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/MapDecoder.java @@ -2,28 +2,31 @@ import com.google.common.base.Preconditions; import com.strategyobject.substrateclient.rpc.core.DecoderPair; +import com.strategyobject.substrateclient.transport.RpcObject; import lombok.val; +import java.util.HashMap; import java.util.Map; -import static java.util.stream.Collectors.toMap; - public class MapDecoder extends AbstractDecoder> { @Override - @SuppressWarnings("unchecked") - protected Map decodeNonNull(Object value, DecoderPair[] decoders) { + protected Map decodeNonNull(RpcObject value, DecoderPair[] decoders) { val keyDecoder = decoders[0].getDecoderOrThrow(); val valueDecoder = decoders[1].getDecoderOrThrow(); - return ((Map) value) + return value.asMap() .entrySet() .stream() - .collect(toMap(e -> keyDecoder.decode(e.getKey()), - e -> valueDecoder.decode(e.getValue()))); + .collect( + HashMap::new, + (map, x) -> map.put( + keyDecoder.decode(RpcObject.of(x.getKey())), + valueDecoder.decode(x.getValue())), + HashMap::putAll); } @Override - protected void checkArguments(Object value, DecoderPair[] decoders) { + protected void checkArguments(RpcObject value, DecoderPair[] decoders) { Preconditions.checkArgument(decoders != null && decoders.length == 2); Preconditions.checkNotNull(decoders[0]); Preconditions.checkNotNull(decoders[1]); diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/ShortDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/ShortDecoder.java index fc9b213a..51a7cb75 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/ShortDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/ShortDecoder.java @@ -1,10 +1,11 @@ package com.strategyobject.substrateclient.rpc.core.decoders; import com.strategyobject.substrateclient.rpc.core.DecoderPair; +import com.strategyobject.substrateclient.transport.RpcObject; public class ShortDecoder extends AbstractDecoder { @Override - protected Short decodeNonNull(Object value, DecoderPair[] decoders) { - return ((Double) value).shortValue(); + protected Short decodeNonNull(RpcObject value, DecoderPair[] decoders) { + return value.asNumber().shortValue(); } } diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/StringDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/StringDecoder.java index 7bb6f1d6..d435f5e8 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/StringDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/StringDecoder.java @@ -1,10 +1,11 @@ package com.strategyobject.substrateclient.rpc.core.decoders; import com.strategyobject.substrateclient.rpc.core.DecoderPair; +import com.strategyobject.substrateclient.transport.RpcObject; public class StringDecoder extends AbstractDecoder { @Override - protected String decodeNonNull(Object value, DecoderPair[] decoders) { - return (String) value; + protected String decodeNonNull(RpcObject value, DecoderPair[] decoders) { + return value.asString(); } } diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/VoidDecoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/VoidDecoder.java index 49e9af53..bb62bb0c 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/VoidDecoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/decoders/VoidDecoder.java @@ -3,10 +3,11 @@ import com.google.common.base.Preconditions; import com.strategyobject.substrateclient.rpc.core.DecoderPair; import com.strategyobject.substrateclient.rpc.core.RpcDecoder; +import com.strategyobject.substrateclient.transport.RpcObject; public class VoidDecoder implements RpcDecoder { @Override - public Void decode(Object value, DecoderPair... decoders) { + public Void decode(RpcObject value, DecoderPair... decoders) { Preconditions.checkArgument(decoders == null || decoders.length == 0); return null; diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/encoders/ListEncoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/encoders/ListEncoder.java index e1c3fbaf..0981a9f3 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/encoders/ListEncoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/encoders/ListEncoder.java @@ -12,12 +12,11 @@ public class ListEncoder implements RpcEncoder> { @Override @SuppressWarnings({"unchecked", "rawtypes"}) public Object encode(List source, EncoderPair... encoders) { - Preconditions.checkArgument(encoders != null && encoders.length == 1); - if (source == null) { return null; } + Preconditions.checkArgument(encoders != null && encoders.length == 1); Preconditions.checkNotNull(encoders[0]); val nestedEncoder = (RpcEncoder) encoders[0].getEncoderOrThrow(); diff --git a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/encoders/MapEncoder.java b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/encoders/MapEncoder.java index 862bfb1e..7879c41e 100644 --- a/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/encoders/MapEncoder.java +++ b/rpc/rpc-core/src/main/java/com/strategyobject/substrateclient/rpc/core/encoders/MapEncoder.java @@ -13,11 +13,11 @@ public class MapEncoder implements RpcEncoder> { @Override @SuppressWarnings({"unchecked", "rawtypes"}) public Object encode(Map source, EncoderPair... encoders) { - Preconditions.checkArgument(encoders != null && encoders.length == 2); if (source == null) { return null; } + Preconditions.checkArgument(encoders != null && encoders.length == 2); Preconditions.checkNotNull(encoders[0]); Preconditions.checkNotNull(encoders[1]); val keyEncoder = (RpcEncoder) encoders[0].getEncoderOrThrow(); diff --git a/rpc/rpc-core/src/test/java/com/strategyobject/substrateclient/rpc/core/decoders/KnownDecoderTests.java b/rpc/rpc-core/src/test/java/com/strategyobject/substrateclient/rpc/core/decoders/KnownDecoderTests.java index a0c58dfe..a0d9a2fe 100644 --- a/rpc/rpc-core/src/test/java/com/strategyobject/substrateclient/rpc/core/decoders/KnownDecoderTests.java +++ b/rpc/rpc-core/src/test/java/com/strategyobject/substrateclient/rpc/core/decoders/KnownDecoderTests.java @@ -1,9 +1,8 @@ package com.strategyobject.substrateclient.rpc.core.decoders; -import com.google.gson.Gson; import com.strategyobject.substrateclient.rpc.core.DecoderPair; import com.strategyobject.substrateclient.rpc.core.RpcDecoder; -import lombok.Getter; +import com.strategyobject.substrateclient.transport.RpcObject; import lombok.RequiredArgsConstructor; import org.junit.jupiter.api.Test; @@ -14,47 +13,63 @@ public class KnownDecoderTests { private final TestCase[] testCases = { - new TestCase<>(new BooleanDecoder(), "true", true), - new TestCase<>(new ByteDecoder(), "5", (byte) 5), - new TestCase<>(new DoubleDecoder(), "15.25", 15.25), - new TestCase<>(new FloatDecoder(), "19.5", 19.5f), - new TestCase<>(new IntDecoder(), "-5", -5), - new TestCase<>(new LongDecoder(), "2147483648", 2147483648L), - new TestCase<>(new ShortDecoder(), "290", (short) 290), - new TestCase<>(new StringDecoder(), "\"some\"", "some"), - new TestCase<>(new VoidDecoder(), "null", null), - + new TestCase<>(new BooleanDecoder(), RpcObject.of(true), true), + new TestCase<>(new BooleanDecoder(), RpcObject.ofNull(), null), + new TestCase<>(new ByteDecoder(), RpcObject.of(5), (byte) 5), + new TestCase<>(new ByteDecoder(), RpcObject.ofNull(), null), + new TestCase<>(new DoubleDecoder(), RpcObject.of(15.25), 15.25), + new TestCase<>(new DoubleDecoder(), RpcObject.ofNull(), null), + new TestCase<>(new FloatDecoder(), RpcObject.of(19.5), 19.5f), + new TestCase<>(new FloatDecoder(), RpcObject.ofNull(), null), + new TestCase<>(new IntDecoder(), RpcObject.of(-5), -5), + new TestCase<>(new IntDecoder(), RpcObject.ofNull(), null), + new TestCase<>(new LongDecoder(), RpcObject.of(2147483648L), 2147483648L), + new TestCase<>(new LongDecoder(), RpcObject.ofNull(), null), + new TestCase<>(new ShortDecoder(), RpcObject.of(290), (short) 290), + new TestCase<>(new ShortDecoder(), RpcObject.ofNull(), null), + new TestCase<>(new StringDecoder(), RpcObject.of("some"), "some"), + new TestCase<>(new StringDecoder(), RpcObject.ofNull(), null), + new TestCase<>(new VoidDecoder(), RpcObject.ofNull(), null), + new TestCase<>( + new ListDecoder().inject(DecoderPair.of(new IntDecoder(), null)), + RpcObject.of(Arrays.asList(RpcObject.of(1), RpcObject.ofNull(), RpcObject.of(3))), + Arrays.asList(1, null, 3)), new TestCase<>( new ListDecoder().inject(DecoderPair.of(new IntDecoder(), null)), - "[1, 2, 3]", - Arrays.asList(1, 2, 3)), + RpcObject.ofNull(), + null), new TestCase<>( new MapDecoder().inject( DecoderPair.of(new StringDecoder(), null), DecoderPair.of(new IntDecoder(), null)), - "{a: 1, b: 2, c: 3}", + RpcObject.of(new HashMap() {{ + put("a", RpcObject.of(1)); + put("b", RpcObject.ofNull()); + put("c", RpcObject.of(3)); + }}), new HashMap() {{ put("a", 1); - put("b", 2); + put("b", null); put("c", 3); }}), + new TestCase<>( + new MapDecoder().inject( + DecoderPair.of(new StringDecoder(), null), + DecoderPair.of(new IntDecoder(), null)), + RpcObject.ofNull(), + null) }; @Test public void decode() { - Gson gson = new Gson(); - Arrays.stream(testCases) - .forEach(c -> assertEquals( - c.decoder.decode(gson.fromJson(c.getJson(), Object.class)), - c.getResult())); + .forEach(c -> assertEquals(c.expected, c.decoder.decode(c.rpcObject))); } @RequiredArgsConstructor - @Getter static class TestCase { private final RpcDecoder decoder; - private final String json; - private final T result; + private final RpcObject rpcObject; + private final T expected; } } diff --git a/rpc/rpc-sections/build.gradle b/rpc/rpc-sections/build.gradle index 74959cee..9b27b11b 100644 --- a/rpc/rpc-sections/build.gradle +++ b/rpc/rpc-sections/build.gradle @@ -15,9 +15,9 @@ dependencies { testImplementation project(':crypto') testAnnotationProcessor project(':scale:scale-codegen') - testImplementation 'org.testcontainers:testcontainers:1.16.0' - testImplementation 'org.testcontainers:junit-jupiter:1.16.0' - testImplementation 'ch.qos.logback:logback-classic:1.2.6' - testImplementation 'org.awaitility:awaitility:4.1.0' + testImplementation 'org.testcontainers:testcontainers:1.16.3' + testImplementation 'org.testcontainers:junit-jupiter:1.16.3' + testImplementation 'ch.qos.logback:logback-classic:1.2.10' + testImplementation 'org.awaitility:awaitility:4.1.1' testImplementation 'org.bouncycastle:bcprov-jdk15on:1.69' } \ No newline at end of file diff --git a/rpc/rpc-types/build.gradle b/rpc/rpc-types/build.gradle index 1ccc775c..c2287a6c 100644 --- a/rpc/rpc-types/build.gradle +++ b/rpc/rpc-types/build.gradle @@ -4,6 +4,8 @@ dependencies { implementation project(':rpc:rpc-core') implementation project(':common') implementation project(':crypto') + implementation project(':transport') + annotationProcessor project(':rpc:rpc-codegen') annotationProcessor project(':scale:scale-codegen') } \ No newline at end of file diff --git a/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/ExtrinsicStatusRpcDecoder.java b/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/ExtrinsicStatusRpcDecoder.java index 4ca5a1d8..bccbb99c 100644 --- a/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/ExtrinsicStatusRpcDecoder.java +++ b/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/ExtrinsicStatusRpcDecoder.java @@ -4,6 +4,7 @@ import com.strategyobject.substrateclient.rpc.core.annotations.AutoRegister; import com.strategyobject.substrateclient.rpc.core.decoders.AbstractDecoder; import com.strategyobject.substrateclient.rpc.core.registries.RpcDecoderRegistry; +import com.strategyobject.substrateclient.transport.RpcObject; import java.util.HashMap; import java.util.Map; @@ -29,13 +30,12 @@ public class ExtrinsicStatusRpcDecoder extends AbstractDecoder } @Override - @SuppressWarnings("unchecked") - protected ExtrinsicStatus decodeNonNull(Object value, DecoderPair[] decoders) { + protected ExtrinsicStatus decodeNonNull(RpcObject value, DecoderPair[] decoders) { Optional decoded; - if (value instanceof String) { - decoded = Optional.ofNullable(STATUS_TO_VALUE.get((String) value)); - } else if (value instanceof Map) { - decoded = ((Map) value).entrySet().stream() + if (value.isString()) { + decoded = Optional.ofNullable(STATUS_TO_VALUE.get(value.asString())); + } else if (value.isMap()) { + decoded = (value.asMap()).entrySet().stream() .filter(e -> STATUS_TO_CLASS.containsKey(e.getKey())) .findFirst() .map(e -> diff --git a/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/NumberDecoder.java b/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/NumberDecoder.java index 6cc5a833..38cdaaf4 100644 --- a/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/NumberDecoder.java +++ b/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/NumberDecoder.java @@ -3,6 +3,7 @@ import com.strategyobject.substrateclient.rpc.core.DecoderPair; import com.strategyobject.substrateclient.rpc.core.annotations.AutoRegister; import com.strategyobject.substrateclient.rpc.core.decoders.AbstractDecoder; +import com.strategyobject.substrateclient.transport.RpcObject; import lombok.val; import java.math.BigInteger; @@ -10,8 +11,8 @@ @AutoRegister(types = Number.class) public class NumberDecoder extends AbstractDecoder { @Override - protected Number decodeNonNull(Object value, DecoderPair[] decoders) { - val stringValue = (String) value; + protected Number decodeNonNull(RpcObject value, DecoderPair[] decoders) { + val stringValue = value.asString(); val number = new BigInteger(stringValue.substring(2), 16); return Number.of(number); diff --git a/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/PairDecoder.java b/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/PairDecoder.java index 0475c3d2..d61ae91e 100644 --- a/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/PairDecoder.java +++ b/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/PairDecoder.java @@ -4,19 +4,18 @@ import com.strategyobject.substrateclient.rpc.core.DecoderPair; import com.strategyobject.substrateclient.rpc.core.annotations.AutoRegister; import com.strategyobject.substrateclient.rpc.core.decoders.AbstractDecoder; +import com.strategyobject.substrateclient.transport.RpcObject; import com.strategyobject.substrateclient.types.tuples.Pair; import lombok.val; -import java.util.List; - @AutoRegister(types = Pair.class) public class PairDecoder extends AbstractDecoder> { @Override - protected Pair decodeNonNull(Object value, DecoderPair[] decoders) { + protected Pair decodeNonNull(RpcObject value, DecoderPair[] decoders) { val firstDecoder = decoders[0].getDecoderOrThrow(); val secondDecoder = decoders[1].getDecoderOrThrow(); - val tuple = (List) value; + val tuple = value.asList(); return Pair.of( firstDecoder.decode(tuple.get(0)), @@ -25,7 +24,7 @@ public class PairDecoder extends AbstractDecoder> { } @Override - protected void checkArguments(Object value, DecoderPair[] decoders) { + protected void checkArguments(RpcObject value, DecoderPair[] decoders) { Preconditions.checkArgument(decoders != null && decoders.length == 2); Preconditions.checkNotNull(decoders[0]); Preconditions.checkNotNull(decoders[1]); diff --git a/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/StorageDataDecoder.java b/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/StorageDataDecoder.java index ad745b6f..d904cc0c 100644 --- a/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/StorageDataDecoder.java +++ b/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/StorageDataDecoder.java @@ -4,11 +4,12 @@ import com.strategyobject.substrateclient.rpc.core.DecoderPair; import com.strategyobject.substrateclient.rpc.core.annotations.AutoRegister; import com.strategyobject.substrateclient.rpc.core.decoders.AbstractDecoder; +import com.strategyobject.substrateclient.transport.RpcObject; @AutoRegister(types = StorageData.class) public class StorageDataDecoder extends AbstractDecoder { @Override - protected StorageData decodeNonNull(Object value, DecoderPair[] decoders) { - return StorageData.valueOf(HexConverter.toBytes((String) value)); + protected StorageData decodeNonNull(RpcObject value, DecoderPair[] decoders) { + return StorageData.valueOf(HexConverter.toBytes(value.asString())); } } \ No newline at end of file diff --git a/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/StorageKeyDecoder.java b/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/StorageKeyDecoder.java index 82fc7f78..6bee248f 100644 --- a/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/StorageKeyDecoder.java +++ b/rpc/rpc-types/src/main/java/com/strategyobject/substrateclient/rpc/types/StorageKeyDecoder.java @@ -4,11 +4,12 @@ import com.strategyobject.substrateclient.rpc.core.DecoderPair; import com.strategyobject.substrateclient.rpc.core.annotations.AutoRegister; import com.strategyobject.substrateclient.rpc.core.decoders.AbstractDecoder; +import com.strategyobject.substrateclient.transport.RpcObject; @AutoRegister(types = StorageKey.class) public class StorageKeyDecoder extends AbstractDecoder { @Override - protected StorageKey decodeNonNull(Object value, DecoderPair[] decoders) { - return StorageKey.valueOf(HexConverter.toBytes((String) value)); + protected StorageKey decodeNonNull(RpcObject value, DecoderPair[] decoders) { + return StorageKey.valueOf(HexConverter.toBytes(value.asString())); } } diff --git a/scale/scale-codegen/build.gradle b/scale/scale-codegen/build.gradle index 789b6aa1..d85b5311 100644 --- a/scale/scale-codegen/build.gradle +++ b/scale/scale-codegen/build.gradle @@ -2,8 +2,8 @@ dependencies { implementation project(':scale') implementation project(':common') - compileOnly 'com.google.auto.service:auto-service-annotations:1.0' - annotationProcessor 'com.google.auto.service:auto-service:1.0' + compileOnly 'com.google.auto.service:auto-service-annotations:1.0.1' + annotationProcessor 'com.google.auto.service:auto-service:1.0.1' implementation 'com.squareup:javapoet:1.13.0' testImplementation 'com.google.testing.compile:compile-testing:0.19' diff --git a/tests/build.gradle b/tests/build.gradle index c9930b86..18331c0e 100644 --- a/tests/build.gradle +++ b/tests/build.gradle @@ -1,3 +1,3 @@ dependencies { - implementation 'org.testcontainers:testcontainers:1.16.0' + implementation 'org.testcontainers:testcontainers:1.16.3' } \ No newline at end of file diff --git a/transport/build.gradle b/transport/build.gradle index 04fdfcd4..62f70bd3 100644 --- a/transport/build.gradle +++ b/transport/build.gradle @@ -2,12 +2,12 @@ dependencies { implementation project(':common') implementation 'org.java-websocket:Java-WebSocket:1.5.2' - implementation 'com.google.code.gson:gson:2.8.8' + implementation 'com.google.code.gson:gson:2.8.9' testImplementation project(':tests') - testImplementation 'ch.qos.logback:logback-classic:1.2.6' - testImplementation 'org.testcontainers:testcontainers:1.16.0' - testImplementation 'org.testcontainers:junit-jupiter:1.16.0' - testImplementation "org.testcontainers:toxiproxy:1.16.0" - testImplementation 'org.awaitility:awaitility:4.1.0' + testImplementation 'ch.qos.logback:logback-classic:1.2.10' + testImplementation 'org.testcontainers:testcontainers:1.16.3' + testImplementation 'org.testcontainers:junit-jupiter:1.16.3' + testImplementation "org.testcontainers:toxiproxy:1.16.3" + testImplementation 'org.awaitility:awaitility:4.1.1' } \ No newline at end of file diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/ProviderInterface.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/ProviderInterface.java index 521fc2e8..528a580d 100644 --- a/transport/src/main/java/com/strategyobject/substrateclient/transport/ProviderInterface.java +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/ProviderInterface.java @@ -54,7 +54,7 @@ public interface ProviderInterface { * @param params Encoded parameters as applicable for the method * @return future containing result */ - CompletableFuture send(String method, // TODO replace `Object` to something like `JObject` to have more strict contract + CompletableFuture send(String method, List params); /** @@ -63,7 +63,7 @@ CompletableFuture send(String method, // TODO replace `Object` to someth * @param method The RPC methods to execute * @return future containing result */ - CompletableFuture send(String method); + CompletableFuture send(String method); /** @@ -78,7 +78,7 @@ CompletableFuture send(String method, // TODO replace `Object` to someth CompletableFuture subscribe(String type, String method, List params, - BiConsumer callback); + BiConsumer callback); /** * Allows unsubscribing to subscriptions made with {@link #subscribe(String, String, List, BiConsumer)} diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcBoolean.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcBoolean.java new file mode 100644 index 00000000..208e9239 --- /dev/null +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcBoolean.java @@ -0,0 +1,20 @@ +package com.strategyobject.substrateclient.transport; + +import lombok.RequiredArgsConstructor; + +@RequiredArgsConstructor +public final class RpcBoolean extends RpcObject { + + private final boolean value; + + @Override + public boolean isBoolean() { + return true; + } + + @Override + public Boolean asBoolean() { + return value; + } + +} diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcList.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcList.java new file mode 100644 index 00000000..c0b44db1 --- /dev/null +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcList.java @@ -0,0 +1,22 @@ +package com.strategyobject.substrateclient.transport; + +import lombok.RequiredArgsConstructor; + +import java.util.List; + +@RequiredArgsConstructor +public final class RpcList extends RpcObject { + + private final List list; + + @Override + public boolean isList() { + return true; + } + + @Override + public List asList() { + return list; + } + +} diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcMap.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcMap.java new file mode 100644 index 00000000..64f80daa --- /dev/null +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcMap.java @@ -0,0 +1,22 @@ +package com.strategyobject.substrateclient.transport; + +import lombok.RequiredArgsConstructor; + +import java.util.Map; + +@RequiredArgsConstructor +public final class RpcMap extends RpcObject { + + private final Map map; + + @Override + public boolean isMap() { + return true; + } + + @Override + public Map asMap() { + return map; + } + +} diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcNull.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcNull.java new file mode 100644 index 00000000..a1da460d --- /dev/null +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcNull.java @@ -0,0 +1,61 @@ +package com.strategyobject.substrateclient.transport; + +import java.util.List; +import java.util.Map; + +public final class RpcNull extends RpcObject { + @Override + public boolean isNull() { + return true; + } + + @Override + public boolean isBoolean() { + return true; + } + + @Override + public boolean isList() { + return true; + } + + @Override + public boolean isMap() { + return true; + } + + @Override + public boolean isNumber() { + return true; + } + + @Override + public boolean isString() { + return true; + } + + @Override + public Boolean asBoolean() { + return null; + } + + @Override + public List asList() { + return null; + } + + @Override + public Map asMap() { + return null; + } + + @Override + public Number asNumber() { + return null; + } + + @Override + public String asString() { + return null; + } +} diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcNumber.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcNumber.java new file mode 100644 index 00000000..144054ea --- /dev/null +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcNumber.java @@ -0,0 +1,20 @@ +package com.strategyobject.substrateclient.transport; + +import lombok.RequiredArgsConstructor; + +@RequiredArgsConstructor +public final class RpcNumber extends RpcObject { + + private final Number value; + + @Override + public boolean isNumber() { + return true; + } + + @Override + public Number asNumber() { + return value; + } + +} diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcObject.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcObject.java new file mode 100644 index 00000000..65e95e21 --- /dev/null +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcObject.java @@ -0,0 +1,79 @@ +package com.strategyobject.substrateclient.transport; + +import java.util.List; +import java.util.Map; + +public abstract class RpcObject { + + RpcObject() { + } + + public boolean isNull() { + return false; + } + + public boolean isBoolean() { + return false; + } + + public boolean isList() { + return false; + } + + public boolean isMap() { + return false; + } + + public boolean isNumber() { + return false; + } + + public boolean isString() { + return false; + } + + public Boolean asBoolean() { + throw new IllegalStateException(); + } + + public List asList() { + throw new IllegalStateException(); + } + + public Map asMap() { + throw new IllegalStateException(); + } + + public Number asNumber() { + throw new IllegalStateException(); + } + + public String asString() { + throw new IllegalStateException(); + } + + public static RpcObject ofNull() { + return new RpcNull(); + } + + public static RpcObject of(boolean value) { + return new RpcBoolean(value); + } + + public static RpcObject of(List list) { + return list == null ? new RpcNull() : new RpcList(list); + } + + public static RpcObject of(Map map) { + return map == null ? new RpcNull() : new RpcMap(map); + } + + public static RpcObject of(Number value) { + return value == null ? new RpcNull() : new RpcNumber(value); + } + + public static RpcObject of(String value) { + return value == null ? new RpcNull() : new RpcString(value); + } + +} diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcString.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcString.java new file mode 100644 index 00000000..ffac54df --- /dev/null +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/RpcString.java @@ -0,0 +1,20 @@ +package com.strategyobject.substrateclient.transport; + +import lombok.RequiredArgsConstructor; + +@RequiredArgsConstructor +public final class RpcString extends RpcObject { + + private final String value; + + @Override + public boolean isString() { + return true; + } + + @Override + public String asString() { + return value; + } + +} diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/SubscriptionHandler.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/SubscriptionHandler.java index 6ecd601c..d0afb24c 100644 --- a/transport/src/main/java/com/strategyobject/substrateclient/transport/SubscriptionHandler.java +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/SubscriptionHandler.java @@ -10,6 +10,6 @@ @Getter @Setter public class SubscriptionHandler { - BiConsumer callBack; + BiConsumer callBack; String type; } diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcObject.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcContract.java similarity index 90% rename from transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcObject.java rename to transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcContract.java index 6400bd63..80751365 100644 --- a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcObject.java +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcContract.java @@ -4,7 +4,7 @@ import lombok.Getter; @Getter -public abstract class JsonRpcObject { +public abstract class JsonRpcContract { private static final String JSONRPC = "2.0"; protected int id; diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcRequest.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcRequest.java index c8fb4bfd..00297b84 100644 --- a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcRequest.java +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcRequest.java @@ -5,7 +5,7 @@ import java.util.List; @Getter -public class JsonRpcRequest extends JsonRpcObject { +public class JsonRpcRequest extends JsonRpcContract { protected final String method; protected final List params; diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcResponse.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcResponse.java index 3f00cdf7..e5eef12b 100644 --- a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcResponse.java +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcResponse.java @@ -1,18 +1,19 @@ package com.strategyobject.substrateclient.transport.coder; +import com.strategyobject.substrateclient.transport.RpcObject; import lombok.Getter; @Getter -public class JsonRpcResponse extends JsonRpcObject { +public class JsonRpcResponse extends JsonRpcContract { static class SubscriptionParam { JsonRpcResponseBaseError error; - Object result; + RpcObject result; String subscription; } // JsonRpcResponseSingle JsonRpcResponseBaseError error; - Object result; + RpcObject result; // JsonRpcResponseSubscription String method; diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcResponseSingle.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcResponseSingle.java index 9eb11f82..8ef7280c 100644 --- a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcResponseSingle.java +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcResponseSingle.java @@ -1,16 +1,17 @@ package com.strategyobject.substrateclient.transport.coder; +import com.strategyobject.substrateclient.transport.RpcObject; import lombok.Getter; @Getter -public class JsonRpcResponseSingle extends JsonRpcObject { +public class JsonRpcResponseSingle extends JsonRpcContract { private final JsonRpcResponseBaseError error; - private final Object result; + private final RpcObject result; private JsonRpcResponseSingle(int id, String jsonrpc, JsonRpcResponseBaseError error, - Object result) { + RpcObject result) { this.id = id; this.jsonrpc = jsonrpc; this.error = error; @@ -25,7 +26,7 @@ void validate() { } } - public Object getResult() { + public RpcObject getResult() { this.validate(); return this.result; } diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcResponseSubscription.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcResponseSubscription.java index c1f2c0b2..e9f3572a 100644 --- a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcResponseSubscription.java +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/JsonRpcResponseSubscription.java @@ -2,19 +2,20 @@ import com.google.common.base.Preconditions; import com.google.common.base.Strings; +import com.strategyobject.substrateclient.transport.RpcObject; import lombok.Getter; import lombok.val; @Getter -public class JsonRpcResponseSubscription extends JsonRpcObject { +public class JsonRpcResponseSubscription extends JsonRpcContract { @Getter public static class SubscriptionParam { private final JsonRpcResponseBaseError error; - private final Object result; + private final RpcObject result; private final String subscription; private SubscriptionParam(JsonRpcResponseBaseError error, - Object result, + RpcObject result, String subscription) { this.error = error; this.result = result; @@ -48,7 +49,7 @@ void validate() { } } - public Object getResult() { + public RpcObject getResult() { this.validate(); return this.params.result; } diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/RpcCoder.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/RpcCoder.java index 5bb72bd7..3e053035 100644 --- a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/RpcCoder.java +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/RpcCoder.java @@ -1,12 +1,16 @@ package com.strategyobject.substrateclient.transport.coder; import com.google.gson.Gson; +import com.google.gson.GsonBuilder; +import com.strategyobject.substrateclient.transport.RpcObject; import java.util.List; import java.util.concurrent.atomic.AtomicInteger; public class RpcCoder { - private static final Gson GSON = new Gson(); + private static final Gson GSON = new GsonBuilder() + .registerTypeAdapter(RpcObject.class, new RpcObjectDeserializer()) + .create(); private final AtomicInteger id = new AtomicInteger(0); diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/RpcObjectDeserializer.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/RpcObjectDeserializer.java new file mode 100644 index 00000000..322ef091 --- /dev/null +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/coder/RpcObjectDeserializer.java @@ -0,0 +1,55 @@ +package com.strategyobject.substrateclient.transport.coder; + +import com.google.gson.JsonDeserializationContext; +import com.google.gson.JsonDeserializer; +import com.google.gson.JsonElement; +import com.google.gson.JsonParseException; +import com.strategyobject.substrateclient.transport.*; +import lombok.val; + +import java.lang.reflect.Type; +import java.util.ArrayList; +import java.util.HashMap; + +public class RpcObjectDeserializer implements JsonDeserializer { + @Override + public RpcObject deserialize(JsonElement json, + Type typeOfT, + JsonDeserializationContext context) throws JsonParseException { + if (json.isJsonNull()) { + return new RpcNull(); + } + + if (json.isJsonPrimitive()) { + val primitive = json.getAsJsonPrimitive(); + + if (primitive.isBoolean()) { + return new RpcBoolean(primitive.getAsBoolean()); + } + + if (primitive.isNumber()) { + return new RpcNumber(primitive.getAsNumber()); + } + + return new RpcString(primitive.getAsString()); + } + + if (json.isJsonArray()) { + val jsonArray = json.getAsJsonArray(); + val list = new ArrayList(jsonArray.size()); + for (val item : jsonArray) { + list.add(context.deserialize(item, RpcObject.class)); + } + + return new RpcList(list); + } + + val jsonObject = json.getAsJsonObject(); + val map = new HashMap(jsonObject.size()); + for (val item : jsonObject.entrySet()) { + map.put(item.getKey(), context.deserialize(item.getValue(), RpcObject.class)); + } + + return new RpcMap(map); + } +} diff --git a/transport/src/main/java/com/strategyobject/substrateclient/transport/ws/WsProvider.java b/transport/src/main/java/com/strategyobject/substrateclient/transport/ws/WsProvider.java index e6d925b1..92edd0e0 100644 --- a/transport/src/main/java/com/strategyobject/substrateclient/transport/ws/WsProvider.java +++ b/transport/src/main/java/com/strategyobject/substrateclient/transport/ws/WsProvider.java @@ -4,10 +4,7 @@ import com.google.common.base.Strings; import com.strategyobject.substrateclient.common.eventemitter.EventEmitter; import com.strategyobject.substrateclient.common.eventemitter.EventListener; -import com.strategyobject.substrateclient.transport.ProviderInterface; -import com.strategyobject.substrateclient.transport.ProviderInterfaceEmitted; -import com.strategyobject.substrateclient.transport.ProviderStatus; -import com.strategyobject.substrateclient.transport.SubscriptionHandler; +import com.strategyobject.substrateclient.transport.*; import com.strategyobject.substrateclient.transport.coder.JsonRpcResponse; import com.strategyobject.substrateclient.transport.coder.JsonRpcResponseSingle; import com.strategyobject.substrateclient.transport.coder.JsonRpcResponseSubscription; @@ -29,7 +26,7 @@ class WsStateSubscription extends SubscriptionHandler { private String method; private List params; - public WsStateSubscription(BiConsumer callBack, + public WsStateSubscription(BiConsumer callBack, String type, String method, List params) { @@ -42,8 +39,8 @@ public WsStateSubscription(BiConsumer callBack, @Getter @Setter @AllArgsConstructor -class WsStateAwaiting { - private CompletableFuture callback; +class WsStateAwaiting { + private CompletableFuture callback; private String method; private List params; private SubscriptionHandler subscription; @@ -66,7 +63,7 @@ public class WsProvider implements ProviderInterface, AutoCloseable { private final URI endpoint; private final Map headers; private final EventEmitter eventEmitter = new EventEmitter(); - private final Map> handlers = new ConcurrentHashMap<>(); + private final Map handlers = new ConcurrentHashMap<>(); private final Map subscriptions = new ConcurrentHashMap<>(); private final Map waitingForId = new ConcurrentHashMap<>(); private final int heartbeatInterval; @@ -220,9 +217,9 @@ public Runnable on(ProviderInterfaceEmitted type, EventListener sub) { return () -> this.eventEmitter.removeListener(type, sub); } - private CompletableFuture send(String method, - List params, - SubscriptionHandler subscription) { + private CompletableFuture send(String method, + List params, + SubscriptionHandler subscription) { val ws = this.webSocket; Preconditions.checkState( ws != null && this.isConnected(), @@ -233,8 +230,8 @@ private CompletableFuture send(String method, val id = jsonRpcRequest.getId(); log.debug("Calling {} {}, {}, {}, {}", id, method, params, json, subscription); - val whenResponseReceived = new CompletableFuture(); - this.handlers.put(id, new WsStateAwaiting<>(whenResponseReceived, method, params, subscription)); + val whenResponseReceived = new CompletableFuture(); + this.handlers.put(id, new WsStateAwaiting(whenResponseReceived, method, params, subscription)); return CompletableFuture.runAsync(() -> ws.send(json)) .whenCompleteAsync((_res, ex) -> { @@ -256,7 +253,7 @@ private CompletableFuture send(String method, * @return future containing result */ @Override - public CompletableFuture send(String method, List params) { + public CompletableFuture send(String method, List params) { return send(method, params, null); } @@ -267,7 +264,7 @@ public CompletableFuture send(String method, List params) { * @return future containing result */ @Override - public CompletableFuture send(String method) { + public CompletableFuture send(String method) { return send(method, null, null); } @@ -283,8 +280,9 @@ public CompletableFuture send(String method) { public CompletableFuture subscribe(String type, String method, List params, - BiConsumer callback) { - return this.send(method, params, new SubscriptionHandler(callback, type)); + BiConsumer callback) { + return this.send(method, params, new SubscriptionHandler(callback, type)) + .thenApplyAsync(RpcObject::asString); } /** @@ -311,7 +309,8 @@ public CompletableFuture unsubscribe(String type, String method, String } else { this.subscriptions.remove(subscription); if (this.isConnected() && this.webSocket != null) { - return this.send(method, Collections.singletonList(id), null); + return this.send(method, Collections.singletonList(id), null) + .thenApplyAsync(RpcObject::asBoolean); } whenUnsubscribed.complete(true); @@ -399,24 +398,23 @@ private void onSocketMessage(String message) { } } - @SuppressWarnings("unchecked") - private void onSocketMessageResult(JsonRpcResponseSingle response) { + private void onSocketMessageResult(JsonRpcResponseSingle response) { val id = response.getId(); - val handler = (WsStateAwaiting) this.handlers.get(id); + val handler = (WsStateAwaiting) this.handlers.get(id); if (handler == null) { log.error("Unable to find handler for id={}", id); return; } try { - val result = (T) response.getResult(); + val result = response.getResult(); // first send the result - in case of subs, we may have an update // immediately if we have some queued results already handler.getCallback().complete(result); val subscription = handler.getSubscription(); if (subscription != null) { - val subId = subscription.getType() + "::" + result; + val subId = subscription.getType() + "::" + result.asString(); this.subscriptions.put( subId, new WsStateSubscription( diff --git a/transport/src/test/java/com/strategyobject/substrateclient/transport/coder/RpcCoderTest.java b/transport/src/test/java/com/strategyobject/substrateclient/transport/coder/RpcCoderTest.java new file mode 100644 index 00000000..06127663 --- /dev/null +++ b/transport/src/test/java/com/strategyobject/substrateclient/transport/coder/RpcCoderTest.java @@ -0,0 +1,65 @@ +package com.strategyobject.substrateclient.transport.coder; + +import com.strategyobject.substrateclient.transport.RpcObject; +import lombok.val; +import org.junit.jupiter.api.Test; + +import java.util.Arrays; +import java.util.Collections; +import java.util.HashMap; + +import static org.assertj.core.api.Assertions.assertThat; + +class RpcCoderTest { + + @Test + void decodeJson() { + val json = "{\n" + + " \"result\": {\n" + + " \"null\": null,\n" + + " \"bool\": true,\n" + + " \"list\": [\n" + + " true,\n" + + " null,\n" + + " false\n" + + " ],\n" + + " \"map\": {\n" + + " \"null\": null,\n" + + " \"bool\": false,\n" + + " \"list\": [],\n" + + " \"map\": {},\n" + + " \"num\": -1,\n" + + " \"str\": \"\"\n" + + " },\n" + + " \"num\": 123.456,\n" + + " \"str\": \"string\"\n" + + " },\n" + + " \"id\": 10,\n" + + " \"jsonrpc\": \"3.0\"\n" + + "}"; + val actual = RpcCoder.decodeJson(json); + + val expected = new JsonRpcResponse(); + expected.jsonrpc = "3.0"; + expected.id = 10; + expected.result = RpcObject.of(new HashMap() {{ + put("null", RpcObject.ofNull()); + put("bool", RpcObject.of(true)); + put("list", RpcObject.of(Arrays.asList(RpcObject.of(true), RpcObject.ofNull(), RpcObject.of(false)))); + put("map", RpcObject.of(new HashMap() {{ + put("null", RpcObject.ofNull()); + put("bool", RpcObject.of(true)); + put("list", RpcObject.of(Collections.emptyList())); + put("num", RpcObject.of(-1)); + put("map", RpcObject.of(new HashMap<>())); + put("str", RpcObject.of("")); + }})); + put("num", RpcObject.of(123.456)); + put("str", RpcObject.of("string")); + }}); + assertThat(actual) + .usingRecursiveComparison() + .isEqualTo(expected); + } + +} \ No newline at end of file diff --git a/transport/src/test/java/com/strategyobject/substrateclient/transport/ws/WsProviderTest.java b/transport/src/test/java/com/strategyobject/substrateclient/transport/ws/WsProviderTest.java index dcf25c27..2591a0ac 100644 --- a/transport/src/test/java/com/strategyobject/substrateclient/transport/ws/WsProviderTest.java +++ b/transport/src/test/java/com/strategyobject/substrateclient/transport/ws/WsProviderTest.java @@ -147,7 +147,9 @@ void canSend() { .build()) { wsProvider.connect().get(WAIT_TIMEOUT, TimeUnit.SECONDS); - val version = (String) wsProvider.send("system_version").get(WAIT_TIMEOUT, TimeUnit.SECONDS); + val version = wsProvider.send("system_version") + .get(WAIT_TIMEOUT, TimeUnit.SECONDS) + .asString(); assertFalse(Strings.isNullOrEmpty(version)); } }