diff --git a/nifi-commons/nifi-record/src/main/java/org/apache/nifi/serialization/record/util/DataTypeUtils.java b/nifi-commons/nifi-record/src/main/java/org/apache/nifi/serialization/record/util/DataTypeUtils.java index d6db0b78fa..53eb8a68f9 100644 --- a/nifi-commons/nifi-record/src/main/java/org/apache/nifi/serialization/record/util/DataTypeUtils.java +++ b/nifi-commons/nifi-record/src/main/java/org/apache/nifi/serialization/record/util/DataTypeUtils.java @@ -1078,7 +1078,7 @@ public class DataTypeUtils { } public static boolean isStringTypeCompatible(final Object value) { - return value != null; + return !(value instanceof Record); } public static boolean isEnumTypeCompatible(final Object value, final EnumDataType enumType) { @@ -1998,6 +1998,26 @@ public class DataTypeUtils { final RecordFieldType thisFieldType = thisDataType.getFieldType(); final RecordFieldType otherFieldType = otherDataType.getFieldType(); + if (thisFieldType == RecordFieldType.ARRAY && otherFieldType == RecordFieldType.ARRAY) { + // Check for array and return the other (or empty if they are both array). This happens if at some point we inferred an element type of null which + // indicates an empty array, and then we inferred a non-null type for the same field in a different record. The non-null type should be used in that case. + ArrayDataType thisArrayType = (ArrayDataType) thisDataType; + ArrayDataType otherArrayType = (ArrayDataType) otherDataType; + if (thisArrayType.getElementType() == null) { + if (otherArrayType.getElementType() == null) { + return Optional.empty(); + } else { + return Optional.of(otherDataType); + } + } else { + if (otherArrayType.getElementType() == null) { + return Optional.of(thisDataType); + } else { + return Optional.empty(); + } + } + } + final int thisIntTypeValue = getIntegerTypeValue(thisFieldType); final int otherIntTypeValue = getIntegerTypeValue(otherFieldType); final boolean thisIsInt = thisIntTypeValue > -1; diff --git a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/pom.xml b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/pom.xml index 8cc3b56125..d307fbda4a 100755 --- a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/pom.xml +++ b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/pom.xml @@ -205,6 +205,8 @@ src/test/resources/json/bank-account-multiline.json src/test/resources/json/bank-account-oneline.json src/test/resources/json/similar-records.json + src/test/resources/json/choice-of-array-empty-or-array-record.json + src/test/resources/json/empty-arrays.json src/test/resources/json/choice-of-embedded-similar-records.json src/test/resources/json/choice-of-embedded-arrays-and-single-records.json src/test/resources/json/choice-of-merged-embedded-arrays-and-single-records.json diff --git a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/main/java/org/apache/nifi/json/JsonSchemaInference.java b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/main/java/org/apache/nifi/json/JsonSchemaInference.java index 630f6d9074..eeba8c7da3 100644 --- a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/main/java/org/apache/nifi/json/JsonSchemaInference.java +++ b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/main/java/org/apache/nifi/json/JsonSchemaInference.java @@ -96,6 +96,11 @@ public class JsonSchemaInference extends HierarchicalSchemaInference { return value.isArray(); } + @Override + protected boolean isEmptyArray(final JsonNode value) { + return value.isArray() && value.size() == 0; + } + @Override protected void forEachFieldInRecord(final JsonNode rawRecord, final BiConsumer fieldConsumer) { final Iterator> itr = rawRecord.fields(); diff --git a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/main/java/org/apache/nifi/schema/inference/HierarchicalSchemaInference.java b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/main/java/org/apache/nifi/schema/inference/HierarchicalSchemaInference.java index 78791bb503..98ee4490ee 100644 --- a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/main/java/org/apache/nifi/schema/inference/HierarchicalSchemaInference.java +++ b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/main/java/org/apache/nifi/schema/inference/HierarchicalSchemaInference.java @@ -21,6 +21,8 @@ import org.apache.nifi.serialization.record.DataType; import org.apache.nifi.serialization.record.RecordField; import org.apache.nifi.serialization.record.RecordFieldType; import org.apache.nifi.serialization.record.RecordSchema; +import org.apache.nifi.serialization.record.type.ArrayDataType; +import org.apache.nifi.serialization.record.type.RecordDataType; import java.io.IOException; import java.util.ArrayList; @@ -52,7 +54,13 @@ public abstract class HierarchicalSchemaInference implements SchemaInferenceE } } - return createSchema(typeMap, rootElementName); + RecordSchema inferredSchema = createSchema(typeMap, rootElementName); + // Replace array with array in the typeMap. We use array internally for empty arrays because for example if we encounter an empty array in the first record, + // we have no way of knowing the type of elements. If we just decide to use STRING as the type like was previously done, this can cause problems because anything can be coerced + // into a STRING, and if we later encounter an array of Records there, we end up inferring that as a STRING so we end up converting the Record objects into STRINGs. + // Instead, we create an Array where the element type is null, then consider ARRAY[x] wider than ARRAY[null] for any x (other than null). But to cover all cases we have to wait + // until the very end, after making inferences based on all data. At that point if the type is still inferred to be null we can just change it to a STRING. + return defaultArrayTypes(inferredSchema); } protected void inferSchema(final T rawRecord, final Map inferences) { @@ -78,29 +86,87 @@ public abstract class HierarchicalSchemaInference implements SchemaInferenceE final DataType fieldDataType = RecordFieldType.RECORD.getRecordDataType(schema); typeInference.addPossibleDataType(fieldDataType); } else if (isArray(value)) { - final FieldTypeInference arrayElementTypeInference = new FieldTypeInference(); - forEachRawRecordInArray(value, arrayElement -> inferType(arrayElement, arrayElementTypeInference)); + if (isEmptyArray(value)) { + // At this point we don't know the type of array elements as the array is empty, and it is too early to assume an array of strings. Use null as the + // element type for now, and call defaultArrayTypes() when all inferences are complete, to ensure that if there are any arrays with inferred element type + // of null, they default to string for the final schema. + final DataType arrayDataType = RecordFieldType.ARRAY.getArrayDataType(null); + typeInference.addPossibleDataType(arrayDataType); + } else { + final FieldTypeInference arrayElementTypeInference = new FieldTypeInference(); + forEachRawRecordInArray(value, arrayElement -> inferType(arrayElement, arrayElementTypeInference)); - final DataType elementDataType = arrayElementTypeInference.toDataType(); - final DataType arrayDataType = RecordFieldType.ARRAY.getArrayDataType(elementDataType); - typeInference.addPossibleDataType(arrayDataType); + DataType elementDataType = arrayElementTypeInference.toDataType(); + final DataType arrayDataType = RecordFieldType.ARRAY.getArrayDataType(elementDataType); + typeInference.addPossibleDataType(arrayDataType); + } } else { typeInference.addPossibleDataType(getDataType(value)); } } + /* + * This method checks a RecordSchema's child fields for array datatypes recursively and replaces them with the default array. This should be called + * after all inferences have been completed. + */ + private RecordSchema defaultArrayTypes(final RecordSchema recordSchema) { + List newRecordFields = new ArrayList<>(recordSchema.getFieldCount()); + for (RecordField childRecordField : recordSchema.getFields()) { + newRecordFields.add(defaultArrayTypes(childRecordField)); + } + return new SimpleRecordSchema(newRecordFields, recordSchema.getIdentifier()); + } + + /* + * This method checks a RecordField for array datatypes recursively and replaces them with the default array + */ + private RecordField defaultArrayTypes(final RecordField recordField) { + final DataType dataType = recordField.getDataType(); + if (dataType.getFieldType() == RecordFieldType.ARRAY) { + if (((ArrayDataType) dataType).getElementType() == null) { + return new RecordField(recordField.getFieldName(), RecordFieldType.ARRAY.getArrayDataType(RecordFieldType.STRING.getDataType()), + recordField.getDefaultValue(), recordField.getAliases(), recordField.isNullable()); + } else { + // Iterate over the array element type (using a synthesized temporary RecordField), defaulting any arrays as well + ArrayDataType arrayDataType = (ArrayDataType) dataType; + RecordField elementRecordField = new RecordField(recordField.getFieldName() + "_element", arrayDataType.getElementType(), recordField.isNullable()); + RecordField adjustedElementRecordField = defaultArrayTypes(elementRecordField); + + return new RecordField(recordField.getFieldName(), RecordFieldType.ARRAY.getArrayDataType(adjustedElementRecordField.getDataType()), + recordField.getDefaultValue(), recordField.getAliases(), recordField.isNullable()); + } + } + if (dataType.getFieldType() == RecordFieldType.RECORD) { + RecordDataType recordDataType = (RecordDataType) dataType; + RecordSchema childSchema = recordDataType.getChildSchema(); + RecordSchema adjustedRecordSchema = defaultArrayTypes(childSchema); + return new RecordField(recordField.getFieldName(), RecordFieldType.RECORD.getRecordDataType(adjustedRecordSchema), recordField.getDefaultValue(), + recordField.getAliases(), recordField.isNullable()); + } + + return recordField; + } + private void inferType(final T value, final FieldTypeInference typeInference) { if (isObject(value)) { final RecordSchema schema = createSchema(value); final DataType fieldDataType = RecordFieldType.RECORD.getRecordDataType(schema); typeInference.addPossibleDataType(fieldDataType); } else if (isArray(value)) { - final FieldTypeInference arrayElementTypeInference = new FieldTypeInference(); - forEachRawRecordInArray(value, arrayElement -> inferType(arrayElement, arrayElementTypeInference)); + if (isEmptyArray(value)) { + // At this point we don't know the type of array elements as the array is empty, and it is too early to assume an array of strings. Use null as the + // element type for now, and call defaultArrayTypes() when all inferences are complete, to ensure that if there are any arrays with inferred element type + // of null, they default to string for the final schema. + final DataType arrayDataType = RecordFieldType.ARRAY.getArrayDataType(null); + typeInference.addPossibleDataType(arrayDataType); + } else { + final FieldTypeInference arrayElementTypeInference = new FieldTypeInference(); + forEachRawRecordInArray(value, arrayElement -> inferType(arrayElement, arrayElementTypeInference)); - final DataType elementDataType = arrayElementTypeInference.toDataType(); - final DataType arrayDataType = RecordFieldType.ARRAY.getArrayDataType(elementDataType); - typeInference.addPossibleDataType(arrayDataType); + DataType elementDataType = arrayElementTypeInference.toDataType(); + final DataType arrayDataType = RecordFieldType.ARRAY.getArrayDataType(elementDataType); + typeInference.addPossibleDataType(arrayDataType); + } } else { typeInference.addPossibleDataType(getDataType(value)); } @@ -129,6 +195,8 @@ public abstract class HierarchicalSchemaInference implements SchemaInferenceE protected abstract boolean isArray(T value); + protected abstract boolean isEmptyArray(T value); + protected abstract void forEachFieldInRecord(T rawRecord, BiConsumer fieldConsumer); protected abstract void forEachRawRecordInArray(T arrayRecord, Consumer rawRecordConsumer); diff --git a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/main/java/org/apache/nifi/xml/inference/XmlSchemaInference.java b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/main/java/org/apache/nifi/xml/inference/XmlSchemaInference.java index 72cb2590fd..be94790253 100644 --- a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/main/java/org/apache/nifi/xml/inference/XmlSchemaInference.java +++ b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/main/java/org/apache/nifi/xml/inference/XmlSchemaInference.java @@ -60,6 +60,11 @@ public class XmlSchemaInference extends HierarchicalSchemaInference { return value.getNodeType() == XmlNodeType.ARRAY; } + @Override + protected boolean isEmptyArray(final XmlNode value) { + return value.getNodeType() == XmlNodeType.ARRAY && ((XmlArrayNode) value).getElements().isEmpty(); + } + @Override protected void forEachFieldInRecord(final XmlNode rawRecord, final BiConsumer fieldConsumer) { final XmlContainerNode container = (XmlContainerNode) rawRecord; diff --git a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/java/org/apache/nifi/json/TestJsonSchemaInference.java b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/java/org/apache/nifi/json/TestJsonSchemaInference.java index 1eb655c117..f7ec0b82aa 100644 --- a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/java/org/apache/nifi/json/TestJsonSchemaInference.java +++ b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/java/org/apache/nifi/json/TestJsonSchemaInference.java @@ -19,8 +19,12 @@ package org.apache.nifi.json; import org.apache.nifi.logging.ComponentLog; import org.apache.nifi.schema.inference.InferSchemaAccessStrategy; import org.apache.nifi.schema.inference.TimeValueInference; +import org.apache.nifi.serialization.record.DataType; +import org.apache.nifi.serialization.record.RecordField; import org.apache.nifi.serialization.record.RecordFieldType; import org.apache.nifi.serialization.record.RecordSchema; +import org.apache.nifi.serialization.record.type.ArrayDataType; +import org.apache.nifi.serialization.record.type.RecordDataType; import org.junit.jupiter.api.Test; import org.mockito.Mockito; @@ -41,17 +45,7 @@ class TestJsonSchemaInference { @Test void testInferenceIncludesAllRecords() throws IOException { - final File file = new File("src/test/resources/json/data-types.json"); - - final RecordSchema schema; - try (final InputStream in = new FileInputStream(file); - final InputStream bufferedIn = new BufferedInputStream(in)) { - - final InferSchemaAccessStrategy accessStrategy = new InferSchemaAccessStrategy<>( - (var, content) -> new JsonRecordSource(content), - new JsonSchemaInference(timestampInference), Mockito.mock(ComponentLog.class)); - schema = accessStrategy.getSchema(null, bufferedIn, null); - } + final RecordSchema schema = inferSchema(new File("src/test/resources/json/data-types.json")); assertSame(RecordFieldType.STRING, schema.getDataType("varcharc").get().getFieldType()); assertSame(RecordFieldType.INT, schema.getDataType("uuid").get().getFieldType()); @@ -82,4 +76,65 @@ class TestJsonSchemaInference { assertEquals(Arrays.asList("varcharc", "uuid", "tinyintc", "textc", "datec", "smallintc", "mediumintc", "longintc", "intc", "bigintc", "floatc", "doublec", "decimalc", "timestampc", "timec", "charc", "tinytextc", "blobc", "mediumtextc", "enumc", "setc", "boolc", "binaryc"), fieldNames); } + + @Test + public void testNestedArrayOfRecords() throws IOException { + final RecordSchema schema = inferSchema(new File("src/test/resources/json/choice-of-array-empty-or-array-record.json")); + final RecordField dataField = schema.getField("data").get(); + assertSame(RecordFieldType.RECORD, dataField.getDataType().getFieldType()); + + final RecordDataType dataFieldType = (RecordDataType) dataField.getDataType(); + final RecordSchema dataSchema = dataFieldType.getChildSchema(); + + final DataType itemsDataType = dataSchema.getDataType("items").get(); + assertSame(RecordFieldType.ARRAY, itemsDataType.getFieldType()); + + final ArrayDataType itemsArrayType = (ArrayDataType) itemsDataType; + final DataType itemsElementType = itemsArrayType.getElementType(); + assertEquals(RecordFieldType.RECORD, itemsElementType.getFieldType()); + + final RecordSchema itemsSchema = ((RecordDataType) itemsElementType).getChildSchema(); + final RecordField itemSchedulesField = itemsSchema.getField("itemSchedules").get(); + final DataType itemSchedulesDataType = itemSchedulesField.getDataType(); + assertEquals(RecordFieldType.ARRAY, itemSchedulesDataType.getFieldType()); + + final ArrayDataType schedulesArrayType = (ArrayDataType) itemSchedulesDataType; + final DataType schedulesElementType = schedulesArrayType.getElementType(); + assertEquals(RecordFieldType.RECORD, schedulesElementType.getFieldType()); + } + + @Test + public void testEmptyArrays() throws IOException { + final RecordSchema schema = inferSchema(new File("src/test/resources/json/empty-arrays.json")); + final DataType itemsDataType = schema.getDataType("items").get(); + assertSame(RecordFieldType.ARRAY, itemsDataType.getFieldType()); + + final ArrayDataType itemsArrayType = (ArrayDataType) itemsDataType; + final DataType itemsElementType = itemsArrayType.getElementType(); + assertEquals(RecordFieldType.RECORD, itemsElementType.getFieldType()); + + final RecordSchema itemsSchema = ((RecordDataType) itemsElementType).getChildSchema(); + final RecordField itemDataField = itemsSchema.getField("itemData").get(); + final DataType ItemDataDatatype = itemDataField.getDataType(); + assertEquals(RecordFieldType.ARRAY, ItemDataDatatype.getFieldType()); + + final ArrayDataType itemDataArrayType = (ArrayDataType) ItemDataDatatype; + final DataType itemDataElementType = itemDataArrayType.getElementType(); + // Empty arrays should be inferred as array + assertEquals(RecordFieldType.STRING, itemDataElementType.getFieldType()); + } + + private RecordSchema inferSchema(final File jsonFile) throws IOException { + try (final InputStream in = new FileInputStream(jsonFile); + final InputStream bufferedIn = new BufferedInputStream(in)) { + + final InferSchemaAccessStrategy accessStrategy = new InferSchemaAccessStrategy<>( + (var, content) -> new JsonRecordSource(content), + new JsonSchemaInference(timestampInference), Mockito.mock(ComponentLog.class)); + + final RecordSchema schema = accessStrategy.getSchema(null, bufferedIn, null); + return schema; + } + + } } \ No newline at end of file diff --git a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/java/org/apache/nifi/json/TestWriteJsonResult.java b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/java/org/apache/nifi/json/TestWriteJsonResult.java index 4cdef538c8..3929070e64 100644 --- a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/java/org/apache/nifi/json/TestWriteJsonResult.java +++ b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/java/org/apache/nifi/json/TestWriteJsonResult.java @@ -21,6 +21,7 @@ import org.apache.commons.lang3.StringUtils; import org.apache.nifi.logging.ComponentLog; import org.apache.nifi.record.NullSuppression; import org.apache.nifi.schema.access.SchemaNameAsAttribute; +import org.apache.nifi.schema.inference.TimeValueInference; import org.apache.nifi.serialization.SimpleRecordSchema; import org.apache.nifi.serialization.record.DataType; import org.apache.nifi.serialization.record.MapRecord; @@ -34,6 +35,7 @@ import org.junit.jupiter.api.Test; import org.mockito.Mockito; import java.io.ByteArrayOutputStream; +import java.io.FileInputStream; import java.io.IOException; import java.math.BigDecimal; import java.math.BigInteger; @@ -508,4 +510,46 @@ class TestWriteJsonResult { final String output = new String(data, StandardCharsets.UTF_8); assertEquals(expected, output); } + + @Test + void testChoiceArrayOfStringsOrArrayOfRecords() throws IOException { + final String FILE_LOCATION = "src/test/resources/json/choice-of-array-string-or-array-record.json"; + final JsonSchemaInference jsonSchemaInference = new JsonSchemaInference(new TimeValueInference(null, null, null)); + final RecordSchema schema = jsonSchemaInference.inferSchema(new JsonRecordSource(new FileInputStream(FILE_LOCATION))); + + final Map itemData1 = new HashMap<>(); + itemData1.put("itemData", new String[]{"test"}); + + final Map quantityMap = new HashMap<>(); + quantityMap.put("quantity", 10); + final List itemDataRecordFields = new ArrayList<>(1); + itemDataRecordFields.add(new RecordField("quantity", RecordFieldType.INT.getDataType(), true)); + final RecordSchema quantityRecordSchema = new SimpleRecordSchema(itemDataRecordFields); + final Record quantityRecord = new MapRecord(quantityRecordSchema, quantityMap); + + final Record[] quantityRecordArray = {quantityRecord}; + final Map itemData2 = new HashMap<>(); + + itemData2.put("itemData", quantityRecordArray); + + final Object[] itemDataArray = {itemData1, itemData2}; + + final Map values = new HashMap<>(); + values.put("items", itemDataArray); + Record topLevelRecord = new MapRecord(schema, values); + + final ByteArrayOutputStream baos = new ByteArrayOutputStream(); + try (final WriteJsonResult writer = new WriteJsonResult(Mockito.mock(ComponentLog.class), schema, new SchemaNameAsAttribute(), baos, true, + NullSuppression.NEVER_SUPPRESS, OutputGrouping.OUTPUT_ARRAY, null, null, null)) { + writer.beginRecordSet(); + writer.writeRecord(topLevelRecord); + writer.finishRecordSet(); + } + + final byte[] data = baos.toByteArray(); + + final String expected = new String(Files.readAllBytes(Paths.get(FILE_LOCATION)), StandardCharsets.UTF_8); + final String output = new String(data, StandardCharsets.UTF_8); + assertEquals(expected, output); + } } diff --git a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/resources/json/choice-of-array-empty-or-array-record.json b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/resources/json/choice-of-array-empty-or-array-record.json new file mode 100644 index 0000000000..5f028cec56 --- /dev/null +++ b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/resources/json/choice-of-array-empty-or-array-record.json @@ -0,0 +1,59 @@ +{ + "data": { + "headerWorklists": [], + "items": [ + { + "itemId": "10", + "itemStatus": {}, + "itemSchedules": [], + "itemWorklists": [ + { + "worklistItem": "364141264", + "reasonCode": "I09", + "description": "Incomplete" + } + ] + }, + { + "itemId": "20", + "itemStatus": { + "code": "1", + "text": "Planned" + }, + "itemSchedules": [ + { + "scheduleItem": "10", + "scheduledQuantity": { + "amount": 10, + "uom": "KX6" + }, + "scheduledVolume": { + "amount": 10000, + "uom": "BX6" + }, + "scheduledVolume2": { + "amount": 1589.873, + "uom": "N15" + }, + "scheduledWeight": { + "amount": 1211.642, + "uom": "TNZ" + } + } + ], + "itemWorklists": [ + { + "worklistItem": "364141264", + "reasonCode": "I09", + "description": "Line item is incomplete" + }, + { + "worklistItem": "364141265", + "reasonCode": "Z69", + "description": "Incorrect item" + } + ] + } + ] + } +} \ No newline at end of file diff --git a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/resources/json/choice-of-array-string-or-array-record.json b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/resources/json/choice-of-array-string-or-array-record.json new file mode 100644 index 0000000000..3e48dd0204 --- /dev/null +++ b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/resources/json/choice-of-array-string-or-array-record.json @@ -0,0 +1,9 @@ +[ { + "items" : [ { + "itemData" : [ "test" ] + }, { + "itemData" : [ { + "quantity" : 10 + } ] + } ] +} ] \ No newline at end of file diff --git a/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/resources/json/empty-arrays.json b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/resources/json/empty-arrays.json new file mode 100644 index 0000000000..c799d918f1 --- /dev/null +++ b/nifi-nar-bundles/nifi-standard-services/nifi-record-serialization-services-bundle/nifi-record-serialization-services/src/test/resources/json/empty-arrays.json @@ -0,0 +1,13 @@ +{ + "id": 6076549, + "items": [ + { + "itemData": [], + "test": 1 + }, + { + "itemData": [], + "test": 2 + } + ] +} \ No newline at end of file