From 44e9ff3ee0c857f451b65a1f841b3be6bb017439 Mon Sep 17 00:00:00 2001 From: Tyler Gregg Date: Mon, 16 Dec 2024 15:48:30 -0800 Subject: [PATCH] Fixes a bug that prevented nested struct macro invocations from being evaluated properly. --- .../ion/impl/macro/EExpressionArgsReader.java | 6 +- .../impl/macro/MacroEvaluatorAsIonReader.kt | 3 +- .../EncodingDirectiveCompilationTest.java | 61 +++++++++++++++++++ 3 files changed, 66 insertions(+), 4 deletions(-) diff --git a/src/main/java/com/amazon/ion/impl/macro/EExpressionArgsReader.java b/src/main/java/com/amazon/ion/impl/macro/EExpressionArgsReader.java index 0549b0ef9..05a9c7354 100644 --- a/src/main/java/com/amazon/ion/impl/macro/EExpressionArgsReader.java +++ b/src/main/java/com/amazon/ion/impl/macro/EExpressionArgsReader.java @@ -251,9 +251,6 @@ protected void readValueAsExpression(boolean isImplicitRest, List expressions) { - if (reader.isInStruct()) { - expressions.add(new Expression.FieldName(reader.getFieldNameSymbol())); - } Macro macro = loadMacro(); List signature = macro.getSignature(); PresenceBitmap presenceBitmap = loadPresenceBitmapIfNecessary(signature); @@ -281,6 +278,9 @@ public void beginEvaluatingMacroInvocation(MacroEvaluator macroEvaluator) { // TODO performance: use a pool of expression lists to avoid repetitive allocations. List expressions = new ArrayList<>(); // TODO performance: avoid fully materializing all expressions up-front. + if (reader.isInStruct()) { + expressions.add(new Expression.FieldName(reader.getFieldNameSymbol())); + } collectEExpressionArgs(expressions); macroEvaluator.initExpansion(expressions); } diff --git a/src/main/java/com/amazon/ion/impl/macro/MacroEvaluatorAsIonReader.kt b/src/main/java/com/amazon/ion/impl/macro/MacroEvaluatorAsIonReader.kt index 4945e5c47..926605407 100644 --- a/src/main/java/com/amazon/ion/impl/macro/MacroEvaluatorAsIonReader.kt +++ b/src/main/java/com/amazon/ion/impl/macro/MacroEvaluatorAsIonReader.kt @@ -70,6 +70,8 @@ class MacroEvaluatorAsIonReader( val arguments: List = evaluator.getArguments() val numberOfContainerEndsAtExpressionIndex = IntArray(arguments.size + 1) + currentFieldName = null // Field names are written only via FieldName expressions + while (index < arguments.size) { for (i in 0 until numberOfContainerEndsAtExpressionIndex[index]) { writer.stepOut() @@ -87,7 +89,6 @@ class MacroEvaluatorAsIonReader( writer.writeValue(this) } is Expression.FieldName -> { - queuedFieldName = argument writer.setFieldNameSymbol(argument.value) } is Expression.EExpression -> { diff --git a/src/test/java/com/amazon/ion/impl/EncodingDirectiveCompilationTest.java b/src/test/java/com/amazon/ion/impl/EncodingDirectiveCompilationTest.java index cf79062d5..f5bdc19e6 100644 --- a/src/test/java/com/amazon/ion/impl/EncodingDirectiveCompilationTest.java +++ b/src/test/java/com/amazon/ion/impl/EncodingDirectiveCompilationTest.java @@ -1010,6 +1010,67 @@ public void structAsParameter(InputType inputType, StreamType streamType) throws } } + private byte[] macroInvocationAsParameterToItself(StreamType outputFormat) { + ByteArrayOutputStream out = new ByteArrayOutputStream(); + IonRawWriter_1_1 writer = outputFormat.newWriter(out); + SortedMap expectedMacroTable = new TreeMap() {{ + put("SimonSays", writeSimonSaysMacro(writer)); + }}; + + outputFormat.startMacroInvocationByName(writer, "SimonSays", expectedMacroTable); + writer.stepInStruct(true); + writer.writeFieldName(FIRST_LOCAL_SYMBOL_ID); + outputFormat.startMacroInvocationByName(writer, "SimonSays", expectedMacroTable); + writer.stepInStruct(true); + writer.writeFieldName(FIRST_LOCAL_SYMBOL_ID); + writer.writeInt(123); + writer.stepOut(); + writer.stepOut(); + + writer.stepOut(); + writer.stepOut(); + + return getBytes(writer, out); + } + + @ParameterizedTest(name = "{0},{1}") + @MethodSource("allCombinations") + public void macroInvocationAsParameterToItself(InputType inputType, StreamType streamType) throws Exception { + byte[] data = macroInvocationAsParameterToItself(streamType); + + try (IonReader reader = inputType.newReader(data)) { + assertEquals(IonType.STRUCT, reader.next()); + reader.stepIn(); + assertEquals(IonType.STRUCT, reader.next()); + assertEquals("foo", reader.getFieldName()); + reader.stepIn(); + assertEquals(IonType.INT, reader.next()); + assertEquals("foo", reader.getFieldName()); + assertEquals(123, reader.intValue()); + assertNull(reader.next()); + reader.stepOut(); + reader.stepOut(); + assertNull(reader.next()); + } + } + + @ParameterizedTest(name = "{0},{1},{2}") + @MethodSource("allInputFormatsInputTypesAndOutputFormats") + public void macroInvocationAsParameterToItselfMacroAwareTranscode( + InputType inputType, + StreamType inputFormat, + StreamType outputFormat + ) throws Exception { + byte[] data = macroInvocationAsParameterToItself(outputFormat); + + verifyMacroAwareTranscode( + data, inputType, inputFormat, + substringCount("(:SimonSays", 2), + substringCount("foo:", 2), + substringCount("123", 1) + ); + } + @ParameterizedTest(name = "{0},{1}") @MethodSource("allCombinations") public void macroInvocationAsParameter(InputType inputType, StreamType streamType) throws Exception {