From 95e903476db52a3312144bc62775fedfe887396f Mon Sep 17 00:00:00 2001 From: Sergio Mendoza Date: Mon, 9 Dec 2024 12:50:09 +0100 Subject: [PATCH 1/2] Counter on duplicated event names Signed-off-by: Sergio Mendoza --- .../codegen/SolidityFunctionWrapper.java | 59 ++++++- .../codegen/SolidityFunctionWrapperTest.java | 152 +++++++++++++++++- 2 files changed, 201 insertions(+), 10 deletions(-) diff --git a/codegen/src/main/java/org/web3j/codegen/SolidityFunctionWrapper.java b/codegen/src/main/java/org/web3j/codegen/SolidityFunctionWrapper.java index 674d24ac9..80435c5c4 100644 --- a/codegen/src/main/java/org/web3j/codegen/SolidityFunctionWrapper.java +++ b/codegen/src/main/java/org/web3j/codegen/SolidityFunctionWrapper.java @@ -418,10 +418,22 @@ FieldSpec createBinaryDefinition(String binary) { .build(); } - private FieldSpec createEventDefinition(String name, List parameters) { + private FieldSpec createEventDefinition( + String name, + List parameters, + Map eventsCount, + AbiDefinition event + ) { CodeBlock initializer = buildVariableLengthEventInitializer(name, parameters); + Integer occurrences = eventsCount.get(name); + if (occurrences > 1) { + event.setName(name + (occurrences - 1)); + eventsCount.replace(name, occurrences - 1); + name = event.getName(); + } + return FieldSpec.builder(Event.class, buildEventDefinitionName(name)) .addModifiers(Modifier.PUBLIC, Modifier.STATIC, Modifier.FINAL) .initializer(initializer) @@ -432,13 +444,14 @@ private String buildEventDefinitionName(String eventName) { return eventName.toUpperCase() + "_EVENT"; } - private List buildFunctionDefinitions( + List buildFunctionDefinitions( String className, TypeSpec.Builder classBuilder, List functionDefinitions) throws ClassNotFoundException { Set duplicateFunctionNames = getDuplicateFunctionNames(functionDefinitions); + Map eventsCount = getDuplicatedEventNames(functionDefinitions); List methodSpecs = new ArrayList<>(); for (AbiDefinition functionDefinition : functionDefinitions) { if (functionDefinition.getType().equals(TYPE_FUNCTION)) { @@ -446,12 +459,35 @@ private List buildFunctionDefinitions( boolean useUpperCase = !duplicateFunctionNames.contains(functionName); methodSpecs.addAll(buildFunctions(functionDefinition, useUpperCase)); } else if (functionDefinition.getType().equals(TYPE_EVENT)) { - methodSpecs.addAll(buildEventFunctions(functionDefinition, classBuilder)); + methodSpecs.addAll( + buildEventFunctions(functionDefinition, classBuilder, eventsCount) + ); } } return methodSpecs; } + Map getDuplicatedEventNames(List functionDefinitions) { + Map countMap = new HashMap<>(); + + functionDefinitions.stream() + .filter( + function -> + TYPE_EVENT.equals(function.getType()) && function.getName() != null) + .forEach( + function -> { + String functionName = function.getName(); + if (countMap.containsKey(functionName)) { + int count = countMap.get(functionName); + countMap.put(functionName, count + 1); + } else { + countMap.put(functionName, 1); + } + }); + + return countMap; + } + private List buildStructTypes(final List functionDefinitions) throws ClassNotFoundException { final List orderedKeys = extractStructs(functionDefinitions); @@ -1917,11 +1953,12 @@ private static String getEventFromLogFunctionName(String functionName) { } List buildEventFunctions( - AbiDefinition functionDefinition, TypeSpec.Builder classBuilder) + AbiDefinition functionDefinition, + TypeSpec.Builder classBuilder, + Map eventsCount + ) throws ClassNotFoundException { - String functionName = functionDefinition.getName(); List inputs = functionDefinition.getInputs(); - String responseClassName = Strings.capitaliseFirstLetter(functionName) + "EventResponse"; List parameters = new ArrayList<>(); List indexedParameters = new ArrayList<>(); @@ -1946,7 +1983,15 @@ List buildEventFunctions( parameters.add(parameter); } - classBuilder.addField(createEventDefinition(functionName, parameters)); + String functionName = functionDefinition.getName(); + + classBuilder.addField(createEventDefinition( + functionName, parameters, eventsCount, functionDefinition + )); + + functionName = functionDefinition.getName(); + + String responseClassName = Strings.capitaliseFirstLetter(functionName) + "EventResponse"; classBuilder.addType( buildEventResponseObject( diff --git a/codegen/src/test/java/org/web3j/codegen/SolidityFunctionWrapperTest.java b/codegen/src/test/java/org/web3j/codegen/SolidityFunctionWrapperTest.java index 156641638..3624bd5b4 100644 --- a/codegen/src/test/java/org/web3j/codegen/SolidityFunctionWrapperTest.java +++ b/codegen/src/test/java/org/web3j/codegen/SolidityFunctionWrapperTest.java @@ -685,7 +685,11 @@ public void testBuildEventConstantMultipleValueReturn() throws Exception { TypeSpec.Builder builder = TypeSpec.classBuilder("testClass"); builder.addMethods( - solidityFunctionWrapper.buildEventFunctions(functionDefinition, builder)); + solidityFunctionWrapper.buildEventFunctions( + functionDefinition, + builder, + solidityFunctionWrapper.getDuplicatedEventNames( + Collections.singletonList(functionDefinition)))); String expected = "class testClass {\n" @@ -773,7 +777,11 @@ public void testBuildEventWithNamedAndNoNamedParameters() throws Exception { TypeSpec.Builder builder = TypeSpec.classBuilder("testClass"); builder.addMethods( - solidityFunctionWrapper.buildEventFunctions(functionDefinition, builder)); + solidityFunctionWrapper.buildEventFunctions( + functionDefinition, + builder, + solidityFunctionWrapper.getDuplicatedEventNames( + Collections.singletonList(functionDefinition)))); String expected = "class testClass {\n" @@ -853,7 +861,11 @@ public void testBuildEventWithNativeList() throws Exception { TypeSpec.Builder builder = TypeSpec.classBuilder("testClass"); builder.addMethods( - solidityFunctionWrapper.buildEventFunctions(functionDefinition, builder)); + solidityFunctionWrapper.buildEventFunctions( + functionDefinition, + builder, + solidityFunctionWrapper.getDuplicatedEventNames( + Collections.singletonList(functionDefinition)))); String expected = "class testClass {\n" @@ -926,6 +938,140 @@ public void testBuildFuncNameConstants() throws Exception { assertEquals(builder.build().toString(), (expected)); } + @Test + public void testBuildFunctionDuplicatedEventNames() throws Exception { + AbiDefinition firstEventDefinition = + new AbiDefinition( + false, + Arrays.asList( + new NamedType("action", "string", false), + new NamedType("pauseState", "bool", false)), + "eventName", + Collections.emptyList(), + "event", + false); + AbiDefinition secondEventDefinition = + new AbiDefinition( + false, + Arrays.asList( + new NamedType("cToken", "address", false), + new NamedType("action", "string", false), + new NamedType("pauseState", "bool", false)), + "eventName", + Collections.emptyList(), + "event", + false); + TypeSpec.Builder builder = TypeSpec.classBuilder("testClass"); + builder.addMethods( + solidityFunctionWrapper.buildFunctionDefinitions( + "testClass", + builder, + Arrays.asList(firstEventDefinition, secondEventDefinition))); + + String expected = + "class testClass {\n" + + " public static final org.web3j.abi.datatypes.Event EVENTNAME1_EVENT = new org.web3j.abi.datatypes.Event(\"eventName\", \n" + + " java.util.Arrays.>asList(new org.web3j.abi.TypeReference() {}, new org.web3j.abi.TypeReference() {}));\n" + + " ;\n" + + "\n" + + " public static final org.web3j.abi.datatypes.Event EVENTNAME_EVENT = new org.web3j.abi.datatypes.Event(\"eventName\", \n" + + " java.util.Arrays.>asList(new org.web3j.abi.TypeReference() {}, new org.web3j.abi.TypeReference() {}, new org.web3j.abi.TypeReference() {}));\n" + + " ;\n" + + "\n" + + " public static java.util.List getEventName1Events(\n" + + " org.web3j.protocol.core.methods.response.TransactionReceipt transactionReceipt) {\n" + + " java.util.List valueList = staticExtractEventParametersWithLog(EVENTNAME1_EVENT, transactionReceipt);\n" + + " java.util.ArrayList responses = new java.util.ArrayList(valueList.size());\n" + + " for (org.web3j.tx.Contract.EventValuesWithLog eventValues : valueList) {\n" + + " EventName1EventResponse typedResponse = new EventName1EventResponse();\n" + + " typedResponse.log = eventValues.getLog();\n" + + " typedResponse.action = (java.lang.String) eventValues.getNonIndexedValues().get(0).getValue();\n" + + " typedResponse.pauseState = (java.lang.Boolean) eventValues.getNonIndexedValues().get(1).getValue();\n" + + " responses.add(typedResponse);\n" + + " }\n" + + " return responses;\n" + + " }\n" + + "\n" + + " public static EventName1EventResponse getEventName1EventFromLog(\n" + + " org.web3j.protocol.core.methods.response.Log log) {\n" + + " org.web3j.tx.Contract.EventValuesWithLog eventValues = staticExtractEventParametersWithLog(EVENTNAME1_EVENT, log);\n" + + " EventName1EventResponse typedResponse = new EventName1EventResponse();\n" + + " typedResponse.log = log;\n" + + " typedResponse.action = (java.lang.String) eventValues.getNonIndexedValues().get(0).getValue();\n" + + " typedResponse.pauseState = (java.lang.Boolean) eventValues.getNonIndexedValues().get(1).getValue();\n" + + " return typedResponse;\n" + + " }\n" + + "\n" + + " public io.reactivex.Flowable eventName1EventFlowable(\n" + + " org.web3j.protocol.core.methods.request.EthFilter filter) {\n" + + " return web3j.ethLogFlowable(filter).map(log -> getEventName1EventFromLog(log));\n" + + " }\n" + + "\n" + + " public io.reactivex.Flowable eventName1EventFlowable(\n" + + " org.web3j.protocol.core.DefaultBlockParameter startBlock,\n" + + " org.web3j.protocol.core.DefaultBlockParameter endBlock) {\n" + + " org.web3j.protocol.core.methods.request.EthFilter filter = new org.web3j.protocol.core.methods.request.EthFilter(startBlock, endBlock, getContractAddress());\n" + + " filter.addSingleTopic(org.web3j.abi.EventEncoder.encode(EVENTNAME1_EVENT));\n" + + " return eventName1EventFlowable(filter);\n" + + " }\n" + + "\n" + + " public static java.util.List getEventNameEvents(\n" + + " org.web3j.protocol.core.methods.response.TransactionReceipt transactionReceipt) {\n" + + " java.util.List valueList = staticExtractEventParametersWithLog(EVENTNAME_EVENT, transactionReceipt);\n" + + " java.util.ArrayList responses = new java.util.ArrayList(valueList.size());\n" + + " for (org.web3j.tx.Contract.EventValuesWithLog eventValues : valueList) {\n" + + " EventNameEventResponse typedResponse = new EventNameEventResponse();\n" + + " typedResponse.log = eventValues.getLog();\n" + + " typedResponse.cToken = (java.lang.String) eventValues.getNonIndexedValues().get(0).getValue();\n" + + " typedResponse.action = (java.lang.String) eventValues.getNonIndexedValues().get(1).getValue();\n" + + " typedResponse.pauseState = (java.lang.Boolean) eventValues.getNonIndexedValues().get(2).getValue();\n" + + " responses.add(typedResponse);\n" + + " }\n" + + " return responses;\n" + + " }\n" + + "\n" + + " public static EventNameEventResponse getEventNameEventFromLog(\n" + + " org.web3j.protocol.core.methods.response.Log log) {\n" + + " org.web3j.tx.Contract.EventValuesWithLog eventValues = staticExtractEventParametersWithLog(EVENTNAME_EVENT, log);\n" + + " EventNameEventResponse typedResponse = new EventNameEventResponse();\n" + + " typedResponse.log = log;\n" + + " typedResponse.cToken = (java.lang.String) eventValues.getNonIndexedValues().get(0).getValue();\n" + + " typedResponse.action = (java.lang.String) eventValues.getNonIndexedValues().get(1).getValue();\n" + + " typedResponse.pauseState = (java.lang.Boolean) eventValues.getNonIndexedValues().get(2).getValue();\n" + + " return typedResponse;\n" + + " }\n" + + "\n" + + " public io.reactivex.Flowable eventNameEventFlowable(\n" + + " org.web3j.protocol.core.methods.request.EthFilter filter) {\n" + + " return web3j.ethLogFlowable(filter).map(log -> getEventNameEventFromLog(log));\n" + + " }\n" + + "\n" + + " public io.reactivex.Flowable eventNameEventFlowable(\n" + + " org.web3j.protocol.core.DefaultBlockParameter startBlock,\n" + + " org.web3j.protocol.core.DefaultBlockParameter endBlock) {\n" + + " org.web3j.protocol.core.methods.request.EthFilter filter = new org.web3j.protocol.core.methods.request.EthFilter(startBlock, endBlock, getContractAddress());\n" + + " filter.addSingleTopic(org.web3j.abi.EventEncoder.encode(EVENTNAME_EVENT));\n" + + " return eventNameEventFlowable(filter);\n" + + " }\n" + + "\n" + + " public static class EventName1EventResponse extends org.web3j.protocol.core.methods.response.BaseEventResponse {\n" + + " public java.lang.String action;\n" + + "\n" + + " public java.lang.Boolean pauseState;\n" + + " }\n" + + "\n" + + " public static class EventNameEventResponse extends org.web3j.protocol.core.methods.response.BaseEventResponse {\n" + + " public java.lang.String cToken;\n" + + "\n" + + " public java.lang.String action;\n" + + "\n" + + " public java.lang.Boolean pauseState;\n" + + " }\n" + + "}\n"; + + assertEquals(builder.build().toString(), (expected)); + } + @Test public void testBuildFunctionTransactionAndCall() throws Exception { AbiDefinition functionDefinition = From ae3edebfa98fb39679f84df641a09ab67fa721f8 Mon Sep 17 00:00:00 2001 From: Dmitry Shohov Date: Mon, 16 Dec 2024 14:20:22 +0200 Subject: [PATCH 2/2] #2121 - clear subscriptionForId map after websocket disconnects Signed-off-by: Dmitry Shohov --- CHANGELOG.md | 1 + .../java/org/web3j/protocol/websocket/WebSocketService.java | 2 ++ 2 files changed, 3 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 48a967283..16f9b2547 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,6 +9,7 @@ See [Conventional Commits](https://conventionalcommits.org) for commit guideline * fixed several code issues found by sonar [#2113](https://github.com/hyperledger/web3j/pull/2113) * update GitHub actions versions [#2114](https://github.com/hyperledger/web3j/pull/2114) +* fixed subscription object leaking after disconnect [#2121](https://github.com/hyperledger/web3j/pull/2121) ### Features diff --git a/core/src/main/java/org/web3j/protocol/websocket/WebSocketService.java b/core/src/main/java/org/web3j/protocol/websocket/WebSocketService.java index 8012cca72..107ba4a0d 100644 --- a/core/src/main/java/org/web3j/protocol/websocket/WebSocketService.java +++ b/core/src/main/java/org/web3j/protocol/websocket/WebSocketService.java @@ -557,6 +557,7 @@ private void closeOutstandingRequests() { request.getOnReply() .completeExceptionally( new IOException("Connection was closed"))); + requestForId.clear(); } private void closeOutstandingSubscriptions() { @@ -567,6 +568,7 @@ private void closeOutstandingSubscriptions() { subscription .getSubject() .onError(new IOException("Connection was closed"))); + subscriptionForId.clear(); } // Method visible for unit-tests