From 4bb6fd7a2113b2f82347639703ca7ed9dbc19bf8 Mon Sep 17 00:00:00 2001 From: prakanth <50439067+prakanth97@users.noreply.github.com> Date: Wed, 8 May 2024 17:02:23 +0530 Subject: [PATCH 01/13] [Automated] Update the native jar versions --- ballerina/Ballerina.toml | 6 +++--- ballerina/CompilerPlugin.toml | 2 +- ballerina/Dependencies.toml | 2 +- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/ballerina/Ballerina.toml b/ballerina/Ballerina.toml index e3597f6..1577886 100644 --- a/ballerina/Ballerina.toml +++ b/ballerina/Ballerina.toml @@ -1,7 +1,7 @@ [package] org = "ballerina" name = "data.xmldata" -version = "0.1.0" +version = "0.1.1" authors = ["Ballerina"] keywords = ["xml"] repository = "https://github.com/ballerina-platform/module-ballerina-data-xmldata" @@ -12,5 +12,5 @@ export = ["data.xmldata"] [[platform.java17.dependency]] groupId = "io.ballerina.lib" artifactId = "data-native" -version = "0.1.0" -path = "../native/build/libs/data.xmldata-native-0.1.0.jar" +version = "0.1.1" +path = "../native/build/libs/data.xmldata-native-0.1.1-SNAPSHOT.jar" diff --git a/ballerina/CompilerPlugin.toml b/ballerina/CompilerPlugin.toml index 7eef6a8..98592f5 100644 --- a/ballerina/CompilerPlugin.toml +++ b/ballerina/CompilerPlugin.toml @@ -3,4 +3,4 @@ id = "constraint-compiler-plugin" class = "io.ballerina.lib.data.xmldata.compiler.XmldataCompilerPlugin" [[dependency]] -path = "../compiler-plugin/build/libs/data.xmldata-compiler-plugin-0.1.0.jar" +path = "../compiler-plugin/build/libs/data.xmldata-compiler-plugin-0.1.1-SNAPSHOT.jar" diff --git a/ballerina/Dependencies.toml b/ballerina/Dependencies.toml index 544d7c8..57bd1b1 100644 --- a/ballerina/Dependencies.toml +++ b/ballerina/Dependencies.toml @@ -10,7 +10,7 @@ distribution-version = "2201.9.0" [[package]] org = "ballerina" name = "data.xmldata" -version = "0.1.0" +version = "0.1.1" dependencies = [ {org = "ballerina", name = "io"}, {org = "ballerina", name = "jballerina.java"}, From ac2249bd538ebad48113f452d7ad3fd6d2fd11bf Mon Sep 17 00:00:00 2001 From: prakanth <50439067+prakanth97@users.noreply.github.com> Date: Mon, 20 May 2024 22:07:39 +0530 Subject: [PATCH 02/13] Support record default values and filler values --- ballerina/tests/fromXml_test.bal | 48 ++--- ballerina/tests/from_xml_with_options.bal | 2 +- .../lib/data/xmldata/utils/DataUtils.java | 62 +++---- .../data/xmldata/xml/QualifiedNameMap.java | 9 +- .../lib/data/xmldata/xml/XmlParser.java | 167 ++++++++++++------ .../lib/data/xmldata/xml/XmlTraversal.java | 135 +++++++++----- 6 files changed, 261 insertions(+), 162 deletions(-) diff --git a/ballerina/tests/fromXml_test.bal b/ballerina/tests/fromXml_test.bal index 1163c0e..128023a 100644 --- a/ballerina/tests/fromXml_test.bal +++ b/ballerina/tests/fromXml_test.bal @@ -1620,7 +1620,9 @@ function testCommentMiddleInContent2() returns error? { test:assertEquals(rec2.A, "John Doe"); } -@test:Config +@test:Config { + enable: false +} function testRegexAsFieldTypeWithParseString() returns error? { string xmlStr = string ` 1 @@ -2812,6 +2814,28 @@ function testProjectionWithXmlAttributeForParseAsType() returns error? { test:assertEquals(rec.A, "2"); } +type DataN3 record {| + int[4] A; +|}; + +@test:Config { + groups: ["fromXmlString"] +} +function testXmlStringToRecordWithArrayAsFieldType() returns error? { + string xmlStr1 = "123"; + DataN3 rec1 = check parseString(xmlStr1); + test:assertEquals(rec1.A, [1, 2, 3, 0]); +} + +@test:Config { + groups: ["fromXml"] +} +function testXmlToRecordWithArrayAsFieldType() returns error? { + xml xmlVal1 = xml `123`; + DataN3 rec1 = check parseAsType(xmlVal1); + test:assertEquals(rec1.A, [1, 2, 3, 0]); +} + // Negative cases type DataN1 record {| int A; @@ -2871,28 +2895,6 @@ function testXmlToRecordNegative4() { test:assertEquals((rec1).message(), "expected 'int' value for the field 'A' found 'array' value"); } -type DataN3 record {| - int[4] A; -|}; - -@test:Config { - groups: ["fromXmlString"] -} -function testXmlStringToRecordNegative5() { - string xmlStr1 = "123"; - DataN3|error rec1 = parseString(xmlStr1); - test:assertEquals((rec1).message(), "array size is not compatible with the expected size"); -} - -@test:Config { - groups: ["fromXml"] -} -function testXmlToRecordNegative5() { - xml xmlVal1 = xml `123`; - DataN3|error rec1 = parseAsType(xmlVal1); - test:assertEquals((rec1).message(), "array size is not compatible with the expected size"); -} - type DataN4 record {| string...; |}; diff --git a/ballerina/tests/from_xml_with_options.bal b/ballerina/tests/from_xml_with_options.bal index 93689fa..881bd41 100644 --- a/ballerina/tests/from_xml_with_options.bal +++ b/ballerina/tests/from_xml_with_options.bal @@ -294,7 +294,7 @@ function testDisableProjectionInArrayForParseStringNegative() { test:assertEquals((rec).message(), "array size is not compatible with the expected size"); } -@test:Config +@test:Config function testDisableProjectionInArrayForParseAsTypeNegative() { xml xmlVal1 = xml ` 1 diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java b/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java index dfe0d70..08952c7 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java @@ -205,10 +205,6 @@ private static String getModifiedName(Map fieldAnnotation, Stri return attributeName; } - public static BArray createNewAnydataList(Type type) { - return ValueCreator.createArrayValue(getArrayTypeFromElementType(type)); - } - public static QualifiedName getElementName(QName qName) { return new QualifiedName(qName.getNamespaceURI(), qName.getLocalPart(), qName.getPrefix()); } @@ -230,33 +226,18 @@ public static Object convertStringToExpType(BString value, Type expType) { return result; } - public static void validateRequiredFields(BMap currentMapValue, XmlAnalyzerData analyzerData) { - Map fields = analyzerData.fieldHierarchy.peek().getMembers(); - for (QualifiedName key : fields.keySet()) { - // Validate required array size - Field field = fields.get(key); - String fieldName = field.getFieldName(); - if (field.getFieldType().getTag() == TypeTags.ARRAY_TAG) { - ArrayType arrayType = (ArrayType) field.getFieldType(); - if (arrayType.getSize() != -1 - && arrayType.getSize() != ((BArray) currentMapValue.get( - StringUtils.fromString(fieldName))).getLength()) { - throw DiagnosticLog.error(DiagnosticErrorCode.ARRAY_SIZE_MISMATCH); - } - } - - if (!SymbolFlags.isFlagOn(field.getFlags(), SymbolFlags.OPTIONAL) - && !currentMapValue.containsKey(StringUtils.fromString(fieldName))) { - throw DiagnosticLog.error(DiagnosticErrorCode.REQUIRED_FIELD_NOT_PRESENT, fieldName); + public static void validateRequiredFields(XmlAnalyzerData analyzerData) { + Map remainingFields = analyzerData.fieldHierarchy.peek().getMembers(); + for (Field field : remainingFields.values()) { + if (SymbolFlags.isFlagOn(field.getFlags(), SymbolFlags.REQUIRED)) { + throw DiagnosticLog.error(DiagnosticErrorCode.REQUIRED_FIELD_NOT_PRESENT, field.getFieldName()); } } - Map attributes = analyzerData.attributeHierarchy.peek().getMembers(); - for (QualifiedName key : attributes.keySet()) { - Field field = attributes.get(key); - String fieldName = field.getFieldName(); - if (!SymbolFlags.isFlagOn(field.getFlags(), SymbolFlags.OPTIONAL)) { - throw DiagnosticLog.error(DiagnosticErrorCode.REQUIRED_ATTRIBUTE_NOT_PRESENT, fieldName); + Map remainingAttributes = analyzerData.attributeHierarchy.peek().getMembers(); + for (Field attribute : remainingAttributes.values()) { + if (!SymbolFlags.isFlagOn(attribute.getFlags(), SymbolFlags.OPTIONAL)) { + throw DiagnosticLog.error(DiagnosticErrorCode.REQUIRED_ATTRIBUTE_NOT_PRESENT, attribute.getFieldName()); } } } @@ -278,18 +259,15 @@ public static ArrayType getValidArrayType(Type type) { }; } - public static ArrayType getArrayTypeFromElementType(Type type) { + public static BArray createArrayValue(Type type) { return switch (type.getTag()) { - case TypeTags.ARRAY_TAG -> TypeCreator.createArrayType(((ArrayType) type).getElementType()); - case TypeTags.JSON_TAG -> PredefinedTypes.TYPE_JSON_ARRAY; - case TypeTags.INT_TAG, TypeTags.FLOAT_TAG, TypeTags.STRING_TAG, TypeTags.BOOLEAN_TAG, TypeTags.BYTE_TAG, - TypeTags.DECIMAL_TAG, TypeTags.RECORD_TYPE_TAG, TypeTags.MAP_TAG, TypeTags.OBJECT_TYPE_TAG, - TypeTags.XML_TAG, TypeTags.NULL_TAG -> TypeCreator.createArrayType(type); - case TypeTags.TYPE_REFERENCED_TYPE_TAG -> getArrayTypeFromElementType(TypeUtils.getReferredType(type)); - default -> PredefinedTypes.TYPE_ANYDATA_ARRAY; + case TypeTags.ARRAY_TAG -> ValueCreator.createArrayValue((ArrayType) type); + case TypeTags.JSON_TAG -> ValueCreator.createArrayValue(PredefinedTypes.TYPE_JSON_ARRAY); + case TypeTags.ANYDATA_TAG -> ValueCreator.createArrayValue(PredefinedTypes.TYPE_ANYDATA_ARRAY); + case TypeTags.TYPE_REFERENCED_TYPE_TAG -> createArrayValue(TypeUtils.getReferredType(type)); + default -> throw new IllegalStateException("Unexpected value: " + type.getTag()); }; } - public static MapType getMapTypeFromConstraintType(Type constraintType) { return switch (constraintType.getTag()) { case TypeTags.MAP_TAG -> (MapType) constraintType; @@ -304,15 +282,19 @@ public static MapType getMapTypeFromConstraintType(Type constraintType) { } public static void updateExpectedTypeStacks(RecordType recordType, XmlAnalyzerData analyzerData) { - analyzerData.attributeHierarchy.push(new QualifiedNameMap(getAllAttributesInRecordType(recordType))); - analyzerData.fieldHierarchy.push(new QualifiedNameMap(getAllFieldsInRecordType(recordType, analyzerData))); + analyzerData.attributeHierarchy.push(new QualifiedNameMap<>(getAllAttributesInRecordType(recordType))); + analyzerData.fieldHierarchy.push(new QualifiedNameMap<>(getAllFieldsInRecordType(recordType, analyzerData))); + analyzerData.visitedFieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); analyzerData.restTypes.push(recordType.getRestFieldType()); +// analyzerData.arrayIndexes.push(new HashMap<>()); } public static void removeExpectedTypeStacks(XmlAnalyzerData analyzerData) { analyzerData.attributeHierarchy.pop(); analyzerData.fieldHierarchy.pop(); + analyzerData.visitedFieldHierarchy.pop(); analyzerData.restTypes.pop(); + analyzerData.arrayIndexes.pop(); } public static boolean isAnydataOrJson(int typeTag) { @@ -790,8 +772,10 @@ private static String addAttributeToRecord(BString prefix, BString uri, String k public static class XmlAnalyzerData { public final Stack nodesStack = new Stack<>(); public final Stack> fieldHierarchy = new Stack<>(); + public final Stack> visitedFieldHierarchy = new Stack<>(); public final Stack> attributeHierarchy = new Stack<>(); public final Stack restTypes = new Stack<>(); + public final Stack> arrayIndexes = new Stack<>(); public RecordType rootRecord; public Field currentField; public QualifiedName rootElement; diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/QualifiedNameMap.java b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/QualifiedNameMap.java index 4fbc856..e0056e9 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/QualifiedNameMap.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/QualifiedNameMap.java @@ -29,12 +29,19 @@ public V remove(QualifiedName qName) { if (!fields.containsKey(localName)) { return null; } - for (QualifiedName qualifiedName : fields.get(localName)) { + + List qNames = fields.get(localName); + for (QualifiedName qualifiedName : qNames) { if (qualifiedName.getNamespaceURI().equals(Constants.NS_ANNOT_NOT_DEFINED)) { field = this.members.remove(qualifiedName); + qNames.remove(qualifiedName); break; } } + + if (qNames.isEmpty()) { + fields.remove(localName); + } } return field; } diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java index 4c16820..d20a3fa 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java @@ -129,6 +129,7 @@ public Object parse(Type type, XmlParserData xmlParserData) { private void reset(XmlParserData xmlParserData) { xmlParserData.fieldHierarchy.clear(); + xmlParserData.visitedFieldHierarchy.clear(); xmlParserData.attributeHierarchy.clear(); xmlParserData.restTypes.clear(); xmlParserData.nodesStack.clear(); @@ -136,6 +137,7 @@ private void reset(XmlParserData xmlParserData) { xmlParserData.siblings.clear(); xmlParserData.recordTypeStack.clear(); xmlParserData.restFieldsPoints.clear(); + xmlParserData.arrayIndexes.clear(); } public Object parse(XmlParserData xmlParserData) { @@ -198,8 +200,10 @@ public void parseRecordRest(String startElementName, XmlParserData xmlParserData if (endElement.getLocalPart().equals(startElementName)) { validateRequiredFields(xmlParserData); xmlParserData.fieldHierarchy.pop(); + xmlParserData.visitedFieldHierarchy.pop(); xmlParserData.restTypes.pop(); xmlParserData.attributeHierarchy.pop(); + xmlParserData.arrayIndexes.pop(); break; } } @@ -225,7 +229,7 @@ private void parseRootElement(XMLStreamReader xmlStreamReader, } RecordType rootRecord = xmlParserData.rootRecord; - xmlParserData.currentNode = ValueCreator.createRecordValue(rootRecord); + xmlParserData.currentNode = ValueCreator.createRecordValue(rootRecord.getPackage(), rootRecord.getName()); QualifiedName elementQName = getElementName(xmlStreamReader); xmlParserData.rootElement = DataUtils.validateAndGetXmlNameFromRecordAnnotation(rootRecord, rootRecord.getName(), elementQName); @@ -234,6 +238,7 @@ private void parseRootElement(XMLStreamReader xmlStreamReader, // Keep track of fields and attributes updateExpectedTypeStacks(rootRecord, xmlParserData); handleAttributes(xmlStreamReader, xmlParserData); + xmlParserData.arrayIndexes.push(new HashMap<>()); } @SuppressWarnings("unchecked") @@ -258,11 +263,14 @@ private void readText(XMLStreamReader xmlStreamReader, if (currentField == null) { QualifiedName contentQName = new QualifiedName("", textFieldName, ""); if (!xmlParserData.fieldHierarchy.peek().contains(contentQName)) { - if (!xmlParserData.allowDataProjection) { + if (xmlParserData.visitedFieldHierarchy.peek().contains(contentQName)) { + currentField = xmlParserData.visitedFieldHierarchy.peek().get(contentQName); + } else if (!xmlParserData.allowDataProjection) { throw DiagnosticLog.error(DiagnosticErrorCode.UNDEFINED_FIELD, textFieldName, xmlParserData.rootRecord); + } else { + return; } - return; } else { currentField = xmlParserData.fieldHierarchy.peek().remove(contentQName); } @@ -277,7 +285,8 @@ private void readText(XMLStreamReader xmlStreamReader, throw DiagnosticLog.error(DiagnosticErrorCode.INVALID_TYPE, fieldType, PredefinedTypes.TYPE_STRING); } - if (xmlParserData.currentNode.containsKey(bFieldName) && !DataUtils.isAnydataOrJson(fieldType.getTag())) { + Object temp = xmlParserData.currentNode.get(bFieldName); + if (temp instanceof BArray && !DataUtils.isAnydataOrJson(fieldType.getTag())) { if (fieldName.equals(textFieldName)) { xmlParserData.currentNode.put(bFieldName, convertStringToRestExpType(bText, fieldType)); return; @@ -287,13 +296,16 @@ private void readText(XMLStreamReader xmlStreamReader, throw DiagnosticLog.error(DiagnosticErrorCode.FOUND_ARRAY_FOR_NON_ARRAY_TYPE, fieldType, fieldName); } - int arraySize = ((ArrayType) fieldType).getSize(); - if (arraySize != -1 && arraySize <= ((BArray) xmlParserData.currentNode.get(bFieldName)).getLength()) { + ArrayType arrayType = (ArrayType) fieldType; + int currentIndex = xmlParserData.arrayIndexes.peek().get(fieldName); + if (arrayType.getState() == ArrayType.ArrayState.CLOSED + && arrayType.getSize() <= currentIndex) { DataUtils.logArrayMismatchErrorIfProjectionNotAllowed(xmlParserData.allowDataProjection); return; } - ((BArray) xmlParserData.currentNode.get(bFieldName)).append(convertStringToRestExpType(bText, fieldType)); + ((BArray) xmlParserData.currentNode.get(bFieldName)).add(currentIndex, + convertStringToRestExpType(bText, fieldType)); return; } @@ -328,10 +340,12 @@ private void convertTextAndUpdateCurrentNode(BMap currentNode, xmlParserData.currentNode = parent; xmlParserData.fieldHierarchy.pop(); + xmlParserData.visitedFieldHierarchy.pop(); xmlParserData.restTypes.pop(); xmlParserData.attributeHierarchy.pop(); xmlParserData.recordTypeStack.pop(); xmlParserData.siblings = xmlParserData.parents.pop(); + xmlParserData.arrayIndexes.pop(); } @SuppressWarnings("unchecked") @@ -343,11 +357,15 @@ private void addTextToCurrentNodeIfExpTypeIsArray(ArrayType fieldType, BString b bText, xmlParserData); case TypeTags.ANYDATA_TAG, TypeTags.JSON_TAG -> { BArray tempArr = (BArray) ((BMap) xmlParserData.nodesStack.peek()).get(bFieldName); - Object nextValue = tempArr.get(tempArr.getLength() - 1); - if (!(nextValue instanceof BMap)) { + // TODO: Using this -2 here seems strange for me. Find a better way to handle this. + HashMap indexes + = xmlParserData.arrayIndexes.get(xmlParserData.arrayIndexes.size() - 2); + int currentIndex = indexes.get(bFieldName.getValue()); + if (fieldType.getState() == ArrayType.ArrayState.CLOSED && currentIndex >= fieldType.getSize()) { + DataUtils.logArrayMismatchErrorIfProjectionNotAllowed(xmlParserData.allowDataProjection); return; } - tempArr.add(tempArr.getLength() - 1, convertStringToRestExpType(bText, fieldType)); + tempArr.add(currentIndex, convertStringToRestExpType(bText, fieldType)); } } } @@ -440,32 +458,17 @@ private void endElement(XMLStreamReader xmlStreamReader, XmlParserData xmlParser } private void validateRequiredFields(XmlParserData xmlParserData) { - BMap currentMapValue = xmlParserData.currentNode; - Map fields = xmlParserData.fieldHierarchy.peek().getMembers(); - for (QualifiedName key : fields.keySet()) { - // Validate required array size - Field field = fields.get(key); - String fieldName = field.getFieldName(); - if (field.getFieldType().getTag() == TypeTags.ARRAY_TAG) { - ArrayType arrayType = (ArrayType) field.getFieldType(); - if (arrayType.getSize() != -1 - && arrayType.getSize() != ((BArray) currentMapValue.get( - StringUtils.fromString(fieldName))).getLength()) { - throw DiagnosticLog.error(DiagnosticErrorCode.ARRAY_SIZE_MISMATCH); - } - } - - if (!SymbolFlags.isFlagOn(field.getFlags(), SymbolFlags.OPTIONAL) - && !currentMapValue.containsKey(StringUtils.fromString(fieldName))) { - throw DiagnosticLog.error(DiagnosticErrorCode.REQUIRED_FIELD_NOT_PRESENT, fieldName); + Map remainingFields = xmlParserData.fieldHierarchy.peek().getMembers(); + for (Field field : remainingFields.values()) { + if (SymbolFlags.isFlagOn(field.getFlags(), SymbolFlags.REQUIRED)) { + throw DiagnosticLog.error(DiagnosticErrorCode.REQUIRED_FIELD_NOT_PRESENT, field.getFieldName()); } } - Map attributes = xmlParserData.attributeHierarchy.peek().getMembers(); - for (QualifiedName key : attributes.keySet()) { - Field field = attributes.get(key); - if (!SymbolFlags.isFlagOn(field.getFlags(), SymbolFlags.OPTIONAL)) { - throw DiagnosticLog.error(DiagnosticErrorCode.REQUIRED_ATTRIBUTE_NOT_PRESENT, field.getFieldName()); + Map remainingAttributes = xmlParserData.attributeHierarchy.peek().getMembers(); + for (Field attribute : remainingAttributes.values()) { + if (!SymbolFlags.isFlagOn(attribute.getFlags(), SymbolFlags.OPTIONAL)) { + throw DiagnosticLog.error(DiagnosticErrorCode.REQUIRED_ATTRIBUTE_NOT_PRESENT, attribute.getFieldName()); } } } @@ -473,7 +476,13 @@ private void validateRequiredFields(XmlParserData xmlParserData) { private void readElement(XMLStreamReader xmlStreamReader, XmlParserData xmlParserData) { QualifiedName elemQName = getElementName(xmlStreamReader); QualifiedNameMap fieldMap = xmlParserData.fieldHierarchy.peek(); - Field currentField = fieldMap.get(elemQName); + Field currentField = null; + if (xmlParserData.visitedFieldHierarchy.peek().contains(elemQName)) { + currentField = xmlParserData.visitedFieldHierarchy.peek().get(elemQName); + } else if (fieldMap.contains(elemQName)) { + elemQName = fieldMap.getMatchedQualifiedName(elemQName); + currentField = fieldMap.remove(elemQName); + } xmlParserData.currentField = currentField; if (currentField == null) { String elemName = elemQName.getLocalPart(); @@ -490,32 +499,37 @@ private void readElement(XMLStreamReader xmlStreamReader, XmlParserData xmlParse return; } - elemQName = fieldMap.getMatchedQualifiedName(elemQName); + xmlParserData.visitedFieldHierarchy.peek().put(elemQName, currentField); BMap currentNode = xmlParserData.currentNode; String fieldName = currentField.getFieldName(); - Object temp = currentNode.get(StringUtils.fromString(fieldName)); BString bFieldName = StringUtils.fromString(fieldName); + Object temp = currentNode.get(bFieldName); Type fieldType = TypeUtils.getReferredType(currentField.getFieldType()); if (!xmlParserData.siblings.contains(elemQName)) { xmlParserData.siblings.put(elemQName, false); - currentNode.remove(bFieldName); // This handles attribute and element with same name. Removes attribute. - } else if (!(temp instanceof BArray)) { - BArray tempArray = ValueCreator.createArrayValue(DataUtils.getArrayTypeFromElementType(fieldType)); - tempArray.append(temp); - currentNode.put(bFieldName, tempArray); + } else { + if (DataUtils.isAnydataOrJson(fieldType.getTag()) && !(temp instanceof BArray)) { + BArray tempArray = DataUtils.createArrayValue(fieldType); + tempArray.append(temp); + currentNode.put(bFieldName, tempArray); + } else if (fieldType.getTag() != TypeTags.ARRAY_TAG) { + throw DiagnosticLog.error(DiagnosticErrorCode.FOUND_ARRAY_FOR_NON_ARRAY_TYPE, fieldType, fieldName); + } } switch (fieldType.getTag()) { case TypeTags.RECORD_TYPE_TAG -> updateNextRecord(xmlStreamReader, xmlParserData, fieldName, fieldType, (RecordType) fieldType); case TypeTags.ARRAY_TAG -> { - Type referredType = TypeUtils.getReferredType(((ArrayType) fieldType).getElementType()); - if (!currentNode.containsKey(bFieldName)) { - currentNode.put(StringUtils.fromString(fieldName), - ValueCreator.createArrayValue(DataUtils.getArrayTypeFromElementType(referredType))); + if (temp == null) { + xmlParserData.arrayIndexes.peek().put(fieldName, 0); + currentNode.put(bFieldName, ValueCreator.createArrayValue((ArrayType) fieldType)); + } else { + HashMap indexes = xmlParserData.arrayIndexes.peek(); + indexes.put(fieldName, indexes.get(fieldName) + 1); } - - updateNextArrayMember(xmlStreamReader, xmlParserData, fieldName, fieldType, referredType); + updateNextArrayMember(xmlStreamReader, xmlParserData, fieldName, fieldType, + ((ArrayType) fieldType).getElementType()); } case TypeTags.MAP_TAG, TypeTags.ANYDATA_TAG, TypeTags.JSON_TAG -> initializeAttributesForNextMappingValue(xmlParserData, fieldName, fieldType); @@ -529,6 +543,9 @@ private void updateNextArrayMember(XMLStreamReader xmlStreamReader, XmlParserDat fieldType, (RecordType) type); case TypeTags.MAP_TAG, TypeTags.ANYDATA_TAG, TypeTags.JSON_TAG -> initializeAttributesForNextMappingValue(xmlParserData, fieldName, fieldType); + case TypeTags.TYPE_REFERENCED_TYPE_TAG -> + updateNextArrayMember(xmlStreamReader, xmlParserData, fieldName, fieldType, + TypeUtils.getReferredType(type)); } } @@ -559,20 +576,27 @@ private BMap updateNextMappingValue(XmlParserData xmlParserData xmlParserData.attributeHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); xmlParserData.fieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); + xmlParserData.visitedFieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); xmlParserData.recordTypeStack.push(null); BMap currentNode = xmlParserData.currentNode; Object temp = currentNode.get(StringUtils.fromString(fieldName)); if (temp instanceof BArray) { - int arraySize = getArraySize(fieldType, temp); - if (arraySize > ((BArray) temp).getLength() || arraySize == -1) { + if (DataUtils.isAnydataOrJson(fieldType.getTag())) { ((BArray) temp).append(nextValue); } else { - DataUtils.logArrayMismatchErrorIfProjectionNotAllowed(xmlParserData.allowDataProjection); + int arraySize = getArraySize(fieldType, temp); + int currentIndex = xmlParserData.arrayIndexes.peek().get(fieldName); + if (currentIndex < arraySize || arraySize == -1) { + ((BArray) temp).add(currentIndex, nextValue); + } else { + DataUtils.logArrayMismatchErrorIfProjectionNotAllowed(xmlParserData.allowDataProjection); + } } } else { currentNode.put(StringUtils.fromString(fieldName), nextValue); } xmlParserData.nodesStack.push(currentNode); + xmlParserData.arrayIndexes.push(new HashMap<>()); return nextValue; } @@ -595,14 +619,15 @@ private void updateNextRecord(XMLStreamReader xmlStreamReader, XmlParserData xml private BMap updateNextValue(RecordType rootRecord, String fieldName, Type fieldType, XmlParserData xmlParserData) { - BMap nextValue = ValueCreator.createRecordValue(rootRecord); + BMap nextValue = ValueCreator.createRecordValue(rootRecord.getPackage(), rootRecord.getName()); updateExpectedTypeStacks(rootRecord, xmlParserData); BMap currentNode = xmlParserData.currentNode; Object temp = currentNode.get(StringUtils.fromString(fieldName)); if (temp instanceof BArray) { - int arraySize = ((ArrayType) fieldType).getSize(); - if (arraySize > ((BArray) temp).getLength() || arraySize == -1) { - ((BArray) temp).append(nextValue); + ArrayType arrayType = (ArrayType) fieldType; + int currentIndex = xmlParserData.arrayIndexes.peek().get(fieldName); + if (arrayType.getState() == ArrayType.ArrayState.OPEN || currentIndex < arrayType.getSize()) { + ((BArray) temp).add(currentIndex, nextValue); } else { DataUtils.logArrayMismatchErrorIfProjectionNotAllowed(xmlParserData.allowDataProjection); } @@ -610,21 +635,25 @@ private BMap updateNextValue(RecordType rootRecord, String fiel currentNode.put(StringUtils.fromString(fieldName), nextValue); } xmlParserData.nodesStack.push(currentNode); + xmlParserData.arrayIndexes.push(new HashMap<>()); return nextValue; } private void updateExpectedTypeStacks(RecordType recordType, XmlParserData xmlParserData) { xmlParserData.attributeHierarchy.push(new QualifiedNameMap<>(getAllAttributesInRecordType(recordType))); xmlParserData.fieldHierarchy.push(new QualifiedNameMap<>(getAllFieldsInRecordType(recordType, xmlParserData))); + xmlParserData.visitedFieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); xmlParserData.restTypes.push(recordType.getRestFieldType()); } private void popStacks(XmlParserData xmlParserData) { xmlParserData.fieldHierarchy.pop(); + xmlParserData.visitedFieldHierarchy.pop(); xmlParserData.restTypes.pop(); xmlParserData.attributeHierarchy.pop(); xmlParserData.siblings = xmlParserData.parents.pop(); xmlParserData.rootRecord = xmlParserData.recordTypeStack.pop(); + xmlParserData.arrayIndexes.pop(); } @SuppressWarnings("unchecked") @@ -685,7 +714,9 @@ private void updateExpectedTypeStacksOfRestType(Type restType, XmlParserData xml updateExpectedTypeStacksOfRestType(((ArrayType) restType).getElementType(), xmlParserData); } else if (restType.getTag() == TypeTags.ANYDATA_TAG || restType.getTag() == TypeTags.JSON_TAG) { xmlParserData.fieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); + xmlParserData.visitedFieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); xmlParserData.restTypes.push(restType); + xmlParserData.arrayIndexes.push(new HashMap<>()); } } @@ -725,7 +756,7 @@ private BString readElementRest(XMLStreamReader xmlStreamReader, XmlParserData x if (!xmlParserData.siblings.contains(elemQName)) { xmlParserData.siblings.put(elemQName, false); if (restType.getTag() == TypeTags.ARRAY_TAG) { - BArray tempArray = ValueCreator.createArrayValue(DataUtils.getArrayTypeFromElementType(restType)); + BArray tempArray = DataUtils.createArrayValue(restType); xmlParserData.currentNode.put(currentFieldName, tempArray); } else { BMap next = @@ -754,7 +785,7 @@ private BString readElementRest(XMLStreamReader xmlStreamReader, XmlParserData x throw DiagnosticLog.error( DiagnosticErrorCode.FOUND_ARRAY_FOR_NON_ARRAY_TYPE, restType, elemQName.getLocalPart()); } - BArray tempArray = ValueCreator.createArrayValue(DataUtils.getArrayTypeFromElementType(restType)); + BArray tempArray = DataUtils.createArrayValue(restType); tempArray.append(currentElement); xmlParserData.currentNode.put(currentFieldName, tempArray); @@ -791,8 +822,12 @@ private void endElementRest(XMLStreamReader xmlStreamReader, XmlParserData xmlPa xmlParserData.siblings = xmlParserData.parents.pop(); if (xmlParserData.siblings.contains(elemQName)) { // TODO: This is duplicated in several places. Remove the duplication. + // TODO: This place behaviour is strange need to check and fix it, Properly. xmlParserData.fieldHierarchy.pop(); + xmlParserData.visitedFieldHierarchy.pop(); xmlParserData.restTypes.pop(); +// xmlParserData.attributeHierarchy.pop(); + xmlParserData.arrayIndexes.pop(); } removeQNameFromRestFieldsPoints(elemQName, xmlParserData); xmlParserData.siblings.put(elemQName, true); @@ -968,7 +1003,14 @@ private void handleAttributes(XMLStreamReader xmlStreamReader, XmlParserData xml } private Optional getFieldFromFieldHierarchy(QualifiedName attQName, XmlParserData xmlParserData) { - Field field = xmlParserData.fieldHierarchy.peek().get(attQName); + Field field; + if (xmlParserData.visitedFieldHierarchy.peek().contains(attQName)) { + field = xmlParserData.visitedFieldHierarchy.peek().get(attQName); + } else { + field = xmlParserData.fieldHierarchy.peek().remove(attQName); + xmlParserData.visitedFieldHierarchy.peek().put(attQName, field); + } + if (field != null) { return Optional.of(field); } @@ -1013,12 +1055,19 @@ private Optional handleRecordRestType(XmlParserData xmlParserData, XMLSt case TypeTags.ARRAY_TAG -> { ArrayType arrayType = (ArrayType) restType; Type elemType = TypeUtils.getReferredType(arrayType.getElementType()); + HashMap indexes = xmlParserData.arrayIndexes.peek(); + if (indexes.containsKey(fieldName)) { + indexes.put(fieldName, indexes.get(fieldName) + 1); + } else { + indexes.put(fieldName, 0); + } + if (elemType.getTag() == TypeTags.RECORD_TYPE_TAG) { updateStacksWhenRecordAsRestType(elementQName, xmlParserData); // Create an array value since expected type is an array. if (!xmlParserData.currentNode.containsKey(StringUtils.fromString(fieldName))) { xmlParserData.currentNode.put(StringUtils.fromString(fieldName), - ValueCreator.createArrayValue(DataUtils.getArrayTypeFromElementType(elemType))); + DataUtils.createArrayValue(arrayType)); } xmlParserData.currentNode = updateNextValue((RecordType) elemType, fieldName, arrayType, xmlParserData); @@ -1064,10 +1113,12 @@ static class TextValue { public static class XmlParserData { private final Stack nodesStack = new Stack<>(); private final Stack> fieldHierarchy = new Stack<>(); + Stack> visitedFieldHierarchy = new Stack<>(); private final Stack> attributeHierarchy = new Stack<>(); private final Stack restTypes = new Stack<>(); private final Stack restFieldsPoints = new Stack<>(); private final Stack recordTypeStack = new Stack<>(); + Stack> arrayIndexes = new Stack<>(); private RecordType rootRecord; private Field currentField; private QualifiedName rootElement; diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java index 0e7d500..380e125 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java @@ -77,10 +77,10 @@ public Object traverseXml(BXml xml, BMap options, Type type) { XmlAnalyzerData analyzerData = new XmlAnalyzerData(); DataUtils.updateOptions(options, analyzerData); RecordType recordType = (RecordType) referredType; - currentNode = ValueCreator.createRecordValue(recordType); + currentNode = ValueCreator.createRecordValue(recordType.getPackage(), recordType.getName()); BXml nextXml = validateRootElement(xml, recordType, analyzerData); Object resultRecordValue = traverseXml(nextXml, recordType, analyzerData); - DataUtils.validateRequiredFields((BMap) resultRecordValue, analyzerData); + DataUtils.validateRequiredFields(analyzerData); return resultRecordValue; } case TypeTags.MAP_TAG -> { @@ -115,6 +115,8 @@ private void convertText(String text, XmlAnalyzerData analyzerData) { QualifiedName contentQName = new QualifiedName("", textFieldName, ""); if (analyzerData.fieldHierarchy.peek().contains(contentQName)) { currentField = analyzerData.fieldHierarchy.peek().remove(contentQName); + } else if (analyzerData.visitedFieldHierarchy.peek().contains(contentQName)) { + currentField = analyzerData.visitedFieldHierarchy.peek().get(contentQName); } else if (analyzerData.restTypes.peek() != null) { currentField = TypeCreator.createField(analyzerData.restTypes.peek(), analyzerData.textFieldName, SymbolFlags.REQUIRED); @@ -130,29 +132,29 @@ private void convertText(String text, XmlAnalyzerData analyzerData) { Type fieldType = TypeUtils.getReferredType(currentField.getFieldType()); Object convertedValue = DataUtils.convertStringToExpType(StringUtils.fromString(text), fieldType); - if (mapValue.containsKey(fieldName)) { + Object value = mapValue.get(fieldName); + if (value instanceof BArray) { if (fieldName.getValue().equals(textFieldName)) { mapValue.put(fieldName, convertedValue); return; } + // TODO: Check do we need this check. if (!DataUtils.isArrayValueAssignable(fieldType.getTag())) { throw DiagnosticLog.error(DiagnosticErrorCode.FOUND_ARRAY_FOR_NON_ARRAY_TYPE, fieldType, fieldName); } - Object value = mapValue.get(fieldName); - int arraySize = (DataUtils.getValidArrayType(fieldType)).getSize(); - if (value instanceof BArray) { - if (arraySize != -1 && arraySize <= ((BArray) value).getLength()) { - DataUtils.logArrayMismatchErrorIfProjectionNotAllowed(analyzerData.allowDataProjection); - return; - } + if (DataUtils.isAnydataOrJson(fieldType.getTag())) { ((BArray) value).append(convertedValue); - } else { - BArray array = DataUtils.createNewAnydataList(fieldType); - array.append(value); - array.append(convertedValue); - mapValue.put(fieldName, array); + return; + } + + ArrayType arrayType = (ArrayType) fieldType; + int currentIndex = analyzerData.arrayIndexes.peek().get(fieldName.getValue()); + if (arrayType.getState() == ArrayType.ArrayState.CLOSED && arrayType.getSize() <= currentIndex) { + DataUtils.logArrayMismatchErrorIfProjectionNotAllowed(analyzerData.allowDataProjection); + return; } + ((BArray) value).add(currentIndex, convertedValue); } else { mapValue.put(fieldName, convertedValue); } @@ -162,7 +164,17 @@ private void convertText(String text, XmlAnalyzerData analyzerData) { private void convertElement(BXmlItem xmlItem, XmlAnalyzerData analyzerData) { QualifiedName elementQName = DataUtils.getElementName(xmlItem.getQName()); QualifiedNameMap fieldsMap = analyzerData.fieldHierarchy.peek(); - Field currentField = fieldsMap.get(elementQName); + Field currentField; + if (analyzerData.visitedFieldHierarchy.peek().contains(elementQName)) { + currentField = analyzerData.visitedFieldHierarchy.peek().get(elementQName); + Type fieldType = TypeUtils.getReferredType(currentField.getFieldType()); + if (!DataUtils.isArrayValueAssignable(fieldType.getTag())) { + throw DiagnosticLog.error(DiagnosticErrorCode.FOUND_ARRAY_FOR_NON_ARRAY_TYPE, fieldType, + currentField.getFieldName()); + } + } else { + currentField = fieldsMap.remove(elementQName); + } analyzerData.currentField = currentField; if (currentField == null) { @@ -181,6 +193,7 @@ private void convertElement(BXmlItem xmlItem, XmlAnalyzerData analyzerData) { return; } + analyzerData.visitedFieldHierarchy.peek().put(elementQName, currentField); BMap mapValue = (BMap) currentNode; Type currentFieldType = TypeUtils.getReferredType(currentField.getFieldType()); if (!DataUtils.isSupportedType(currentFieldType)) { @@ -196,12 +209,22 @@ private void convertElement(BXmlItem xmlItem, XmlAnalyzerData analyzerData) { return; } case TypeTags.ARRAY_TAG -> { - if (!mapValue.containsKey(bCurrentFieldName)) { - BArray array = DataUtils.createNewAnydataList(currentFieldType); - mapValue.put(bCurrentFieldName, array); - } + Object temp = mapValue.get(bCurrentFieldName); Type elementType = TypeUtils.getReferredType(((ArrayType) currentFieldType).getElementType()); int elementTypeTag = elementType.getTag(); + if (elementTypeTag == TypeTags.JSON_TAG || elementTypeTag == TypeTags.ANYDATA_TAG) { + convertWithRestType(xmlItem, currentFieldType, analyzerData); + return; + } + + if (temp == null) { + analyzerData.arrayIndexes.peek().put(fieldName, 0); + mapValue.put(bCurrentFieldName, DataUtils.createArrayValue(currentFieldType)); + } else { + HashMap indexes = analyzerData.arrayIndexes.peek(); + indexes.put(fieldName, indexes.get(fieldName) + 1); + } + if (elementTypeTag == TypeTags.RECORD_TYPE_TAG) { handleRecordType(xmlItem, currentFieldType, fieldName, (RecordType) elementType, mapValue, analyzerData); @@ -211,15 +234,10 @@ private void convertElement(BXmlItem xmlItem, XmlAnalyzerData analyzerData) { currentNode = updateNextValue(elementType, fieldName, currentFieldType, mapValue, analyzerData); traverseXml(xmlItem.getChildrenSeq(), currentFieldType, analyzerData); - DataUtils.validateRequiredFields((BMap) currentNode, analyzerData); + DataUtils.validateRequiredFields(analyzerData); DataUtils.removeExpectedTypeStacks(analyzerData); currentNode = analyzerData.nodesStack.pop(); return; - } else if (elementTypeTag == TypeTags.JSON_TAG || elementTypeTag == TypeTags.ANYDATA_TAG) { - updateNextMap(elementType, analyzerData); - convertWithRestType(xmlItem, elementType, analyzerData); - DataUtils.removeExpectedTypeStacks(analyzerData); - return; } } case TypeTags.MAP_TAG -> { @@ -227,13 +245,14 @@ private void convertElement(BXmlItem xmlItem, XmlAnalyzerData analyzerData) { currentNode = updateNextValue(currentFieldType, fieldName, currentFieldType, mapValue, analyzerData); traverseXml(xmlItem.getChildrenSeq(), currentFieldType, analyzerData); - DataUtils.validateRequiredFields((BMap) currentNode, analyzerData); + DataUtils.validateRequiredFields(analyzerData); DataUtils.removeExpectedTypeStacks(analyzerData); currentNode = analyzerData.nodesStack.pop(); return; } case TypeTags.JSON_TAG, TypeTags.ANYDATA_TAG -> { updateNextMap(currentFieldType, analyzerData); + analyzerData.arrayIndexes.push(new HashMap<>()); convertWithRestType(xmlItem, currentFieldType, analyzerData); DataUtils.removeExpectedTypeStacks(analyzerData); return; @@ -249,7 +268,7 @@ private void handleRecordType(BXmlItem xmlItem, Type currentFieldType, String fi RecordType prevRecord = analyzerData.rootRecord; analyzerData.rootRecord = elementType; traverseXml(xmlItem.getChildrenSeq(), currentFieldType, analyzerData); - DataUtils.validateRequiredFields((BMap) currentNode, analyzerData); + DataUtils.validateRequiredFields(analyzerData); DataUtils.removeExpectedTypeStacks(analyzerData); analyzerData.rootRecord = prevRecord; currentNode = analyzerData.nodesStack.pop(); @@ -262,6 +281,7 @@ private void updateNextMap(Type fieldType, XmlAnalyzerData analyzerData) { analyzerData.restTypes.push(fieldType); } analyzerData.fieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); + analyzerData.visitedFieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); analyzerData.attributeHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); } @@ -283,7 +303,10 @@ private BMap updateNextValue(Type type, String fieldName, Type BMap nextValue; switch (type.getTag()) { - case TypeTags.RECORD_TYPE_TAG -> nextValue = ValueCreator.createRecordValue((RecordType) type); + case TypeTags.RECORD_TYPE_TAG -> { + RecordType recordType = (RecordType) type; + nextValue = ValueCreator.createRecordValue(recordType.getPackage(), recordType.getName()); + } case TypeTags.MAP_TAG -> nextValue = ValueCreator.createMapValue((MapType) type); case TypeTags.ANYDATA_TAG, TypeTags.JSON_TAG -> nextValue = ValueCreator.createMapValue(TypeCreator.createMapType(type)); @@ -292,9 +315,10 @@ private BMap updateNextValue(Type type, String fieldName, Type Object temp = currentMapValue.get(StringUtils.fromString(fieldName)); if (temp instanceof BArray) { - int arraySize = ((ArrayType) fieldType).getSize(); - if (arraySize > ((BArray) temp).getLength() || arraySize == -1) { - ((BArray) temp).append(nextValue); + ArrayType arrayType = (ArrayType) fieldType; + int currentIndex = analyzerData.arrayIndexes.peek().get(fieldName); + if (arrayType.getState() == ArrayType.ArrayState.OPEN || currentIndex < arrayType.getSize()) { + ((BArray) temp).add(currentIndex, nextValue); } else { DataUtils.logArrayMismatchErrorIfProjectionNotAllowed(analyzerData.allowDataProjection); } @@ -302,6 +326,7 @@ private BMap updateNextValue(Type type, String fieldName, Type currentMapValue.put(StringUtils.fromString(fieldName), nextValue); } analyzerData.nodesStack.push(currentMapValue); + analyzerData.arrayIndexes.push(new HashMap<>()); return nextValue; } @@ -317,11 +342,15 @@ private void convertWithRestType(BXmlItem xmlItem, Type restType, XmlAnalyzerDat return; } case TypeTags.ARRAY_TAG -> { - BString bElementName = StringUtils.fromString(elemName); - if (!mapValue.containsKey(bElementName)) { - BArray array = DataUtils.createNewAnydataList(restType); - mapValue.put(bElementName, array); + HashMap indexes = analyzerData.arrayIndexes.peek(); + if (!indexes.containsKey(elemName)) { + indexes.put(elemName, 0); + BArray array = DataUtils.createArrayValue(restType); + mapValue.put(StringUtils.fromString(elemName), array); + } else { + indexes.put(elemName, indexes.get(elemName) + 1); } + Type elementType = ((ArrayType) restType).getElementType(); if (elementType.getTag() == TypeTags.RECORD_TYPE_TAG) { handleRecordType(xmlItem, restType, elemName, (RecordType) elementType, mapValue, @@ -333,10 +362,14 @@ private void convertWithRestType(BXmlItem xmlItem, Type restType, XmlAnalyzerDat BString bElementName = StringUtils.fromString(elemName); BArray arrayValue; - if (mapValue.containsKey(bElementName)) { + if (mapValue.containsKey(bElementName) && mapValue.get(bElementName) != null) { Object currentElement = mapValue.get(bElementName); if (!(currentElement instanceof BArray)) { - arrayValue = DataUtils.createNewAnydataList(restType); + if (!DataUtils.isArrayValueAssignable(restType.getTag())) { + throw DiagnosticLog.error(DiagnosticErrorCode.FOUND_ARRAY_FOR_NON_ARRAY_TYPE, restType, + elemName); + } + arrayValue = DataUtils.createArrayValue(restType); arrayValue.append(currentElement); mapValue.put(bElementName, arrayValue); } else { @@ -365,12 +398,26 @@ private void convertWithRestType(BXmlItem xmlItem, Type restType, XmlAnalyzerDat } BMap nextValue = ValueCreator.createMapValue(DataUtils.getMapTypeFromConstraintType(restType)); - arrayValue.append(nextValue); + if (DataUtils.isAnydataOrJson(restType.getTag())) { + arrayValue.append(nextValue); + } else { + ArrayType arrayType = (ArrayType) restType; + int currentIndex = analyzerData.arrayIndexes.peek().get(elemName); + if (arrayType.getState() == ArrayType.ArrayState.CLOSED + && arrayType.getSize() <= currentIndex) { + DataUtils.logArrayMismatchErrorIfProjectionNotAllowed(analyzerData.allowDataProjection); + } else { + arrayValue.add(currentIndex, nextValue); + } + } analyzerData.nodesStack.push(currentNode); currentNode = nextValue; handleAttributesRest(xmlItem, nextValue, restType); analyzerData.fieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); + analyzerData.visitedFieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); + // TODO: Check and fix this. +// analyzerData.arrayIndexes.push(new HashMap<>()); if (restType.getTag() == TypeTags.ARRAY_TAG) { Type memberType = ((ArrayType) restType).getElementType(); analyzerData.restTypes.push(memberType); @@ -380,7 +427,9 @@ private void convertWithRestType(BXmlItem xmlItem, Type restType, XmlAnalyzerDat traverseXml(xmlItem.getChildrenSeq(), restType, analyzerData); } analyzerData.fieldHierarchy.pop(); + analyzerData.visitedFieldHierarchy.pop(); analyzerData.restTypes.pop(); +// analyzerData.arrayIndexes.pop(); currentNode = analyzerData.nodesStack.pop(); return; } @@ -524,6 +573,7 @@ private BXml validateRootElement(BXml xml, RecordType recordType, XmlAnalyzerDat // Keep track of fields and attributes DataUtils.updateExpectedTypeStacks(recordType, analyzerData); handleAttributes(xmlItem, (BMap) currentNode, analyzerData); + analyzerData.arrayIndexes.push(new HashMap<>()); return xmlItem.getChildrenSeq(); } @@ -545,7 +595,12 @@ private void handleAttributes(BXmlItem xmlItem, BMap currentNod // Element and Attribute have same name. Priority given to element. continue; } - field = analyzerData.fieldHierarchy.peek().get(attribute); + if (analyzerData.visitedFieldHierarchy.peek().contains(attribute)) { + field = analyzerData.visitedFieldHierarchy.peek().get(attribute); + } else { + field = analyzerData.fieldHierarchy.peek().remove(attribute); + analyzerData.visitedFieldHierarchy.peek().put(attribute, field); + } } if (field == null) { From a90910af81892b4d1718233bf1dd2e501a5da598 Mon Sep 17 00:00:00 2001 From: prakanth <50439067+prakanth97@users.noreply.github.com> Date: Tue, 21 May 2024 22:18:14 +0530 Subject: [PATCH 03/13] Restrict RegExp as expected type --- ballerina/tests/fromXml_test.bal | 130 +++++++++++------- .../lib/data/xmldata/utils/Constants.java | 2 + .../lib/data/xmldata/utils/DataUtils.java | 23 ++++ .../lib/data/xmldata/xml/XmlParser.java | 29 +++- .../lib/data/xmldata/xml/XmlTraversal.java | 109 +++++++++------ 5 files changed, 197 insertions(+), 96 deletions(-) diff --git a/ballerina/tests/fromXml_test.bal b/ballerina/tests/fromXml_test.bal index 128023a..6bda3d1 100644 --- a/ballerina/tests/fromXml_test.bal +++ b/ballerina/tests/fromXml_test.bal @@ -1620,54 +1620,6 @@ function testCommentMiddleInContent2() returns error? { test:assertEquals(rec2.A, "John Doe"); } -@test:Config { - enable: false -} -function testRegexAsFieldTypeWithParseString() returns error? { - string xmlStr = string ` - 1 - 2 - Code - - Kanth - - `; - record {| - string:RegExp[] A; - string B; - record {| - string name; - |} C; - |} rec1 = check parseString(xmlStr); - test:assertEquals(rec1.length(), 3); - test:assertEquals(rec1.A, [1, 2]); - test:assertEquals(rec1.B, "Code"); - test:assertEquals(rec1.C.name, "Kanth"); -} - -@test:Config -function testRegexAsFieldTypeWithParseAsType() returns error? { - xml xmlVal = xml ` - 1 - 2 - Code - - Kanth - - `; - record {| - string:RegExp[] A; - string B; - record {| - string name; - |} C; - |} rec1 = check parseAsType(xmlVal); - test:assertEquals(rec1.length(), 3); - test:assertEquals(rec1.A, [1, 2]); - test:assertEquals(rec1.B, "Code"); - test:assertEquals(rec1.C.name, "Kanth"); -} - @test:Config function testAnydataAsRestFieldWithParseString() returns error? { string xmlStr = string ` @@ -3242,3 +3194,85 @@ function testInvalidNamespaceInOpenRecordForParseAsType2() { test:assertTrue(err is error); test:assertEquals((err).message(), "undefined field 'name' in record 'data.xmldata:AuthorOpen'"); } + +@test:Config +function testRegexAsFieldTypeWithParseStringNegative1() { + string xmlStr = string ` + 1 + 2 + Code + + Kanth + + `; + record {| + string:RegExp[] A; + string B; + record {| + string name; + |} C; + |}|Error err = parseString(xmlStr); + test:assertTrue(err is error); + test:assertEquals((err).message(), "unsupported input type"); +} + +@test:Config +function testRegexAsFieldTypeWithParseStringNegative2() { + string xmlStr = string ` + 1 + Code + + Kanth + + `; + record {| + string:RegExp A; + string B; + record {| + string name; + |} C; + |}|Error err = parseString(xmlStr); + test:assertTrue(err is error); + test:assertEquals((err).message(), "unsupported input type"); +} + +@test:Config +function testRegexAsFieldTypeWithParseAsType() { + xml xmlVal = xml ` + 1 + 2 + Code + + Kanth + + `; + record {| + string:RegExp[] A; + string B; + record {| + string name; + |} C; + |}|error err = parseAsType(xmlVal); + test:assertTrue(err is error); + test:assertEquals((err).message(), "unsupported input type"); +} + +@test:Config +function testRegexAsFieldTypeWithParseAsType2() { + xml xmlVal = xml ` + 1 + Code + + Kanth + + `; + record {| + string:RegExp A; + string B; + record {| + string name; + |} C; + |}|error err = parseAsType(xmlVal); + test:assertTrue(err is error); + test:assertEquals((err).message(), "unsupported input type"); +} \ No newline at end of file diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/utils/Constants.java b/native/src/main/java/io/ballerina/lib/data/xmldata/utils/Constants.java index 2e3e477..b26969e 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/utils/Constants.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/utils/Constants.java @@ -60,4 +60,6 @@ private Constants() {} public static final BString ALLOW_DATA_PROJECTION = StringUtils.fromString("allowDataProjection"); public static final String NON_NUMERIC_STRING_REGEX = "[^a-zA-Z\\d\s]"; public static final String NS_ANNOT_NOT_DEFINED = "$$ns_annot_not_defined$$"; + public static final String REGEXP_MODULE_NAME = "lang.regexp"; + public static final String REGEXP_TYPE_NAME = "RegExp"; } diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java b/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java index 08952c7..ec2f911 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java @@ -21,6 +21,7 @@ import io.ballerina.lib.data.xmldata.FromString; import io.ballerina.lib.data.xmldata.xml.QualifiedName; import io.ballerina.lib.data.xmldata.xml.QualifiedNameMap; +import io.ballerina.runtime.api.Module; import io.ballerina.runtime.api.PredefinedTypes; import io.ballerina.runtime.api.TypeTags; import io.ballerina.runtime.api.creators.TypeCreator; @@ -30,6 +31,7 @@ import io.ballerina.runtime.api.types.Field; import io.ballerina.runtime.api.types.MapType; import io.ballerina.runtime.api.types.RecordType; +import io.ballerina.runtime.api.types.ReferenceType; import io.ballerina.runtime.api.types.Type; import io.ballerina.runtime.api.types.UnionType; import io.ballerina.runtime.api.utils.StringUtils; @@ -764,6 +766,27 @@ private static String addAttributeToRecord(BString prefix, BString uri, String k return prefix.getValue().concat(Constants.COLON).concat(key); } + public static boolean isRegExpType(Type type) { + Module module = type.getPackage(); + if (module == null) { + return false; + } + + String moduleName = module.getName(); + String typeName = type.getName(); + if (typeName == null || moduleName == null) { + return false; + } + if (moduleName.equals(Constants.REGEXP_MODULE_NAME) && typeName.equals(Constants.REGEXP_TYPE_NAME)) { + return true; + } + + if (type.getTag() == TypeTags.TYPE_REFERENCED_TYPE_TAG) { + return isRegExpType(((ReferenceType) type).getReferredType()); + } + return false; + } + /** * Holds data required for the traversing. * diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java index d20a3fa..0c24df0 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java @@ -504,26 +504,38 @@ private void readElement(XMLStreamReader xmlStreamReader, XmlParserData xmlParse String fieldName = currentField.getFieldName(); BString bFieldName = StringUtils.fromString(fieldName); Object temp = currentNode.get(bFieldName); - Type fieldType = TypeUtils.getReferredType(currentField.getFieldType()); + Type fieldType = currentField.getFieldType(); + Type referredFieldType = TypeUtils.getReferredType(fieldType); if (!xmlParserData.siblings.contains(elemQName)) { xmlParserData.siblings.put(elemQName, false); } else { - if (DataUtils.isAnydataOrJson(fieldType.getTag()) && !(temp instanceof BArray)) { - BArray tempArray = DataUtils.createArrayValue(fieldType); + if (DataUtils.isAnydataOrJson(referredFieldType.getTag()) && !(temp instanceof BArray)) { + BArray tempArray = DataUtils.createArrayValue(referredFieldType); tempArray.append(temp); currentNode.put(bFieldName, tempArray); - } else if (fieldType.getTag() != TypeTags.ARRAY_TAG) { + } else if (referredFieldType.getTag() != TypeTags.ARRAY_TAG) { throw DiagnosticLog.error(DiagnosticErrorCode.FOUND_ARRAY_FOR_NON_ARRAY_TYPE, fieldType, fieldName); } } + if (DataUtils.isRegExpType(fieldType)) { + throw DiagnosticLog.error(DiagnosticErrorCode.UNSUPPORTED_TYPE); + } + + initializeNextValueBasedOnExpectedType(fieldName, fieldType, temp, currentNode, xmlParserData); + } + + private void initializeNextValueBasedOnExpectedType(String fieldName, Type fieldType, Object temp, + BMap currentNode, + XmlParserData xmlParserData) { switch (fieldType.getTag()) { case TypeTags.RECORD_TYPE_TAG -> updateNextRecord(xmlStreamReader, xmlParserData, fieldName, fieldType, (RecordType) fieldType); case TypeTags.ARRAY_TAG -> { if (temp == null) { xmlParserData.arrayIndexes.peek().put(fieldName, 0); - currentNode.put(bFieldName, ValueCreator.createArrayValue((ArrayType) fieldType)); + currentNode.put(StringUtils.fromString(fieldName), + ValueCreator.createArrayValue((ArrayType) fieldType)); } else { HashMap indexes = xmlParserData.arrayIndexes.peek(); indexes.put(fieldName, indexes.get(fieldName) + 1); @@ -533,11 +545,18 @@ private void readElement(XMLStreamReader xmlStreamReader, XmlParserData xmlParse } case TypeTags.MAP_TAG, TypeTags.ANYDATA_TAG, TypeTags.JSON_TAG -> initializeAttributesForNextMappingValue(xmlParserData, fieldName, fieldType); + case TypeTags.TYPE_REFERENCED_TYPE_TAG -> + initializeNextValueBasedOnExpectedType(fieldName, TypeUtils.getReferredType(fieldType), temp, + currentNode, xmlParserData); } } private void updateNextArrayMember(XMLStreamReader xmlStreamReader, XmlParserData xmlParserData, String fieldName, Type fieldType, Type type) { + if (DataUtils.isRegExpType(type)) { + throw DiagnosticLog.error(DiagnosticErrorCode.UNSUPPORTED_TYPE); + } + switch (type.getTag()) { case TypeTags.RECORD_TYPE_TAG -> updateNextRecord(xmlStreamReader, xmlParserData, fieldName, fieldType, (RecordType) type); diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java index 380e125..06cd2b1 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java @@ -175,6 +175,7 @@ private void convertElement(BXmlItem xmlItem, XmlAnalyzerData analyzerData) { } else { currentField = fieldsMap.remove(elementQName); } + analyzerData.currentField = currentField; if (currentField == null) { @@ -194,52 +195,26 @@ private void convertElement(BXmlItem xmlItem, XmlAnalyzerData analyzerData) { } analyzerData.visitedFieldHierarchy.peek().put(elementQName, currentField); - BMap mapValue = (BMap) currentNode; - Type currentFieldType = TypeUtils.getReferredType(currentField.getFieldType()); + Type currentFieldType = currentField.getFieldType(); if (!DataUtils.isSupportedType(currentFieldType)) { throw DiagnosticLog.error(DiagnosticErrorCode.UNSUPPORTED_TYPE, currentFieldType); } - String fieldName = currentField.getFieldName(); - BString bCurrentFieldName = StringUtils.fromString(fieldName); - switch (currentFieldType.getTag()) { - case TypeTags.RECORD_TYPE_TAG -> { - handleRecordType(xmlItem, currentFieldType, fieldName, (RecordType) currentFieldType, mapValue, - analyzerData); - return; - } - case TypeTags.ARRAY_TAG -> { - Object temp = mapValue.get(bCurrentFieldName); - Type elementType = TypeUtils.getReferredType(((ArrayType) currentFieldType).getElementType()); - int elementTypeTag = elementType.getTag(); - if (elementTypeTag == TypeTags.JSON_TAG || elementTypeTag == TypeTags.ANYDATA_TAG) { - convertWithRestType(xmlItem, currentFieldType, analyzerData); - return; - } + if (DataUtils.isRegExpType(currentFieldType)) { + throw DiagnosticLog.error(DiagnosticErrorCode.UNSUPPORTED_TYPE); + } - if (temp == null) { - analyzerData.arrayIndexes.peek().put(fieldName, 0); - mapValue.put(bCurrentFieldName, DataUtils.createArrayValue(currentFieldType)); - } else { - HashMap indexes = analyzerData.arrayIndexes.peek(); - indexes.put(fieldName, indexes.get(fieldName) + 1); - } + convertToFieldType(xmlItem, currentField, currentField.getFieldName(), currentFieldType, + (BMap) currentNode, analyzerData); + } - if (elementTypeTag == TypeTags.RECORD_TYPE_TAG) { - handleRecordType(xmlItem, currentFieldType, fieldName, (RecordType) elementType, mapValue, - analyzerData); - return; - } else if (elementTypeTag == TypeTags.MAP_TAG) { - updateNextMap(elementType, analyzerData); - currentNode = updateNextValue(elementType, fieldName, currentFieldType, mapValue, - analyzerData); - traverseXml(xmlItem.getChildrenSeq(), currentFieldType, analyzerData); - DataUtils.validateRequiredFields(analyzerData); - DataUtils.removeExpectedTypeStacks(analyzerData); - currentNode = analyzerData.nodesStack.pop(); - return; - } - } + private void convertToFieldType(BXmlItem xmlItem, Field currentField, String fieldName, Type currentFieldType, + BMap mapValue, XmlAnalyzerData analyzerData) { + switch (currentFieldType.getTag()) { + case TypeTags.RECORD_TYPE_TAG -> handleRecordType(xmlItem, currentFieldType, fieldName, + (RecordType) currentFieldType, mapValue, analyzerData); + case TypeTags.ARRAY_TAG -> convertToArrayType(xmlItem, currentField, mapValue, + StringUtils.fromString(fieldName), (ArrayType) currentFieldType, analyzerData); case TypeTags.MAP_TAG -> { updateNextMap(currentFieldType, analyzerData); currentNode = updateNextValue(currentFieldType, fieldName, currentFieldType, mapValue, @@ -248,17 +223,65 @@ private void convertElement(BXmlItem xmlItem, XmlAnalyzerData analyzerData) { DataUtils.validateRequiredFields(analyzerData); DataUtils.removeExpectedTypeStacks(analyzerData); currentNode = analyzerData.nodesStack.pop(); - return; } case TypeTags.JSON_TAG, TypeTags.ANYDATA_TAG -> { updateNextMap(currentFieldType, analyzerData); analyzerData.arrayIndexes.push(new HashMap<>()); convertWithRestType(xmlItem, currentFieldType, analyzerData); DataUtils.removeExpectedTypeStacks(analyzerData); - return; } + case TypeTags.TYPE_REFERENCED_TYPE_TAG -> + convertToFieldType(xmlItem, currentField, fieldName, TypeUtils.getReferredType(currentFieldType), + mapValue, analyzerData); + default -> traverseXml(xmlItem.getChildrenSeq(), currentFieldType, analyzerData); + } + } + + private void convertToArrayType(BXmlItem xmlItem, Field field, BMap mapValue, + BString bCurrentFieldName, ArrayType arrayType, XmlAnalyzerData analyzerData) { + Object temp = mapValue.get(bCurrentFieldName); + Type elementType = arrayType.getElementType(); + + if (DataUtils.isRegExpType(elementType)) { + throw DiagnosticLog.error(DiagnosticErrorCode.UNSUPPORTED_TYPE); + } + + String fieldName = field.getFieldName(); + int referredElementTypeTag = TypeUtils.getReferredType(elementType).getTag(); + if (referredElementTypeTag == TypeTags.JSON_TAG || referredElementTypeTag == TypeTags.ANYDATA_TAG) { + convertWithRestType(xmlItem, arrayType, analyzerData); + return; + } + + if (temp == null) { + analyzerData.arrayIndexes.peek().put(fieldName, 0); + mapValue.put(bCurrentFieldName, DataUtils.createArrayValue(arrayType)); + } else { + HashMap indexes = analyzerData.arrayIndexes.peek(); + indexes.put(fieldName, indexes.get(fieldName) + 1); + } + + convertToArrayMemberType(xmlItem, fieldName, arrayType, elementType, mapValue, analyzerData); + } + + private void convertToArrayMemberType(BXmlItem xmlItem, String fieldName, ArrayType fieldType, Type elementType, + BMap mapValue, XmlAnalyzerData analyzerData) { + switch (elementType.getTag()) { + case TypeTags.RECORD_TYPE_TAG -> handleRecordType(xmlItem, fieldType, fieldName, + (RecordType) elementType, mapValue, analyzerData); + case TypeTags.MAP_TAG -> { + updateNextMap(elementType, analyzerData); + currentNode = updateNextValue(elementType, fieldName, fieldType, mapValue, analyzerData); + traverseXml(xmlItem.getChildrenSeq(), fieldType, analyzerData); + DataUtils.validateRequiredFields(analyzerData); + DataUtils.removeExpectedTypeStacks(analyzerData); + currentNode = analyzerData.nodesStack.pop(); + } + case TypeTags.TYPE_REFERENCED_TYPE_TAG -> + convertToArrayMemberType(xmlItem, fieldName, fieldType, TypeUtils.getReferredType(elementType), + mapValue, analyzerData); + default -> traverseXml(xmlItem.getChildrenSeq(), fieldType, analyzerData); } - traverseXml(xmlItem.getChildrenSeq(), currentFieldType, analyzerData); } private void handleRecordType(BXmlItem xmlItem, Type currentFieldType, String fieldName, RecordType elementType, From 0748528f3927f69dd0e74b4fc7936bb367252f16 Mon Sep 17 00:00:00 2001 From: prakanth <50439067+prakanth97@users.noreply.github.com> Date: Tue, 21 May 2024 23:51:12 +0530 Subject: [PATCH 04/13] Improve code quality --- .../lib/data/xmldata/xml/XmlTraversal.java | 240 ++++++++++-------- 1 file changed, 128 insertions(+), 112 deletions(-) diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java index 06cd2b1..455f38b 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java @@ -186,7 +186,7 @@ private void convertElement(BXmlItem xmlItem, XmlAnalyzerData analyzerData) { throw DiagnosticLog.error(DiagnosticErrorCode.UNDEFINED_FIELD, elementName, analyzerData.rootRecord); } - convertWithRestType(xmlItem, restType, analyzerData); + convertToRestType(xmlItem, restType, analyzerData); } else if (!analyzerData.allowDataProjection) { throw DiagnosticLog.error(DiagnosticErrorCode.UNDEFINED_FIELD, elementName, analyzerData.rootRecord); @@ -211,23 +211,16 @@ private void convertElement(BXmlItem xmlItem, XmlAnalyzerData analyzerData) { private void convertToFieldType(BXmlItem xmlItem, Field currentField, String fieldName, Type currentFieldType, BMap mapValue, XmlAnalyzerData analyzerData) { switch (currentFieldType.getTag()) { - case TypeTags.RECORD_TYPE_TAG -> handleRecordType(xmlItem, currentFieldType, fieldName, + case TypeTags.RECORD_TYPE_TAG -> convertToRecordType(xmlItem, currentFieldType, fieldName, (RecordType) currentFieldType, mapValue, analyzerData); case TypeTags.ARRAY_TAG -> convertToArrayType(xmlItem, currentField, mapValue, StringUtils.fromString(fieldName), (ArrayType) currentFieldType, analyzerData); - case TypeTags.MAP_TAG -> { - updateNextMap(currentFieldType, analyzerData); - currentNode = updateNextValue(currentFieldType, fieldName, currentFieldType, mapValue, - analyzerData); - traverseXml(xmlItem.getChildrenSeq(), currentFieldType, analyzerData); - DataUtils.validateRequiredFields(analyzerData); - DataUtils.removeExpectedTypeStacks(analyzerData); - currentNode = analyzerData.nodesStack.pop(); - } + case TypeTags.MAP_TAG -> convertToMapType(xmlItem, currentFieldType, currentFieldType, + fieldName, mapValue, analyzerData); case TypeTags.JSON_TAG, TypeTags.ANYDATA_TAG -> { updateNextMap(currentFieldType, analyzerData); analyzerData.arrayIndexes.push(new HashMap<>()); - convertWithRestType(xmlItem, currentFieldType, analyzerData); + convertToRestType(xmlItem, currentFieldType, analyzerData); DataUtils.removeExpectedTypeStacks(analyzerData); } case TypeTags.TYPE_REFERENCED_TYPE_TAG -> @@ -249,7 +242,7 @@ private void convertToArrayType(BXmlItem xmlItem, Field field, BMap mapValue, XmlAnalyzerData analyzerData) { switch (elementType.getTag()) { - case TypeTags.RECORD_TYPE_TAG -> handleRecordType(xmlItem, fieldType, fieldName, + case TypeTags.RECORD_TYPE_TAG -> convertToRecordType(xmlItem, fieldType, fieldName, (RecordType) elementType, mapValue, analyzerData); - case TypeTags.MAP_TAG -> { - updateNextMap(elementType, analyzerData); - currentNode = updateNextValue(elementType, fieldName, fieldType, mapValue, analyzerData); - traverseXml(xmlItem.getChildrenSeq(), fieldType, analyzerData); - DataUtils.validateRequiredFields(analyzerData); - DataUtils.removeExpectedTypeStacks(analyzerData); - currentNode = analyzerData.nodesStack.pop(); - } + case TypeTags.MAP_TAG -> + convertToMapType(xmlItem, fieldType, elementType, fieldName, mapValue, analyzerData); case TypeTags.TYPE_REFERENCED_TYPE_TAG -> convertToArrayMemberType(xmlItem, fieldName, fieldType, TypeUtils.getReferredType(elementType), mapValue, analyzerData); @@ -284,8 +271,9 @@ private void convertToArrayMemberType(BXmlItem xmlItem, String fieldName, ArrayT } } - private void handleRecordType(BXmlItem xmlItem, Type currentFieldType, String fieldName, RecordType elementType, - BMap mapValue, XmlAnalyzerData analyzerData) { + private void convertToRecordType(BXmlItem xmlItem, Type currentFieldType, String fieldName, + RecordType elementType, BMap mapValue, + XmlAnalyzerData analyzerData) { currentNode = updateNextRecord(xmlItem, elementType, fieldName, currentFieldType, mapValue, analyzerData); RecordType prevRecord = analyzerData.rootRecord; @@ -297,6 +285,16 @@ private void handleRecordType(BXmlItem xmlItem, Type currentFieldType, String fi currentNode = analyzerData.nodesStack.pop(); } + private void convertToMapType(BXmlItem xmlItem, Type fieldType, Type elementType, String fieldName, + BMap mapValue, XmlAnalyzerData analyzerData) { + updateNextMap(elementType, analyzerData); + currentNode = updateNextMappingValue(elementType, fieldName, fieldType, mapValue, analyzerData); + traverseXml(xmlItem.getChildrenSeq(), fieldType, analyzerData); + DataUtils.validateRequiredFields(analyzerData); + DataUtils.removeExpectedTypeStacks(analyzerData); + currentNode = analyzerData.nodesStack.pop(); + } + private void updateNextMap(Type fieldType, XmlAnalyzerData analyzerData) { if (fieldType.getTag() == TypeTags.MAP_TAG) { analyzerData.restTypes.push(((MapType) fieldType).getConstrainedType()); @@ -313,15 +311,16 @@ private BMap updateNextRecord(BXmlItem xmlItem, RecordType reco XmlAnalyzerData analyzerData) { DataUtils.updateExpectedTypeStacks(recordType, analyzerData); BMap nextValue = - updateNextValue(recordType, fieldName, fieldType, currentMapValue, analyzerData); + updateNextMappingValue(recordType, fieldName, fieldType, currentMapValue, analyzerData); QName qName = xmlItem.getQName(); DataUtils.validateTypeNamespace(qName.getPrefix(), qName.getNamespaceURI(), recordType); handleAttributes(xmlItem, nextValue, analyzerData); return nextValue; } - private BMap updateNextValue(Type type, String fieldName, Type fieldType, - BMap currentMapValue, XmlAnalyzerData analyzerData) { + private BMap updateNextMappingValue(Type type, String fieldName, Type fieldType, + BMap currentMapValue, + XmlAnalyzerData analyzerData) { analyzerData.currentField = null; BMap nextValue; @@ -354,107 +353,66 @@ private BMap updateNextValue(Type type, String fieldName, Type } @SuppressWarnings("unchecked") - private void convertWithRestType(BXmlItem xmlItem, Type restType, XmlAnalyzerData analyzerData) { + private void convertToRestType(BXmlItem xmlItem, Type restType, XmlAnalyzerData analyzerData) { String elemName = xmlItem.getQName().getLocalPart(); analyzerData.currentField = TypeCreator.createField(restType, elemName, SymbolFlags.PUBLIC); BMap mapValue = (BMap) currentNode; + checkRestTypeAndConvert(xmlItem, elemName, restType, restType, mapValue, analyzerData); + } - switch (restType.getTag()) { - case TypeTags.RECORD_TYPE_TAG -> { - handleRecordType(xmlItem, restType, elemName, (RecordType) restType, mapValue, analyzerData); - return; - } + private void checkRestTypeAndConvert(BXmlItem xmlItem, String elemName, Type restType, Type elementType, + BMap mapValue, XmlAnalyzerData analyzerData) { + switch (elementType.getTag()) { + case TypeTags.RECORD_TYPE_TAG -> + convertToRecordType(xmlItem, restType, elemName, (RecordType) elementType, mapValue, analyzerData); case TypeTags.ARRAY_TAG -> { HashMap indexes = analyzerData.arrayIndexes.peek(); if (!indexes.containsKey(elemName)) { indexes.put(elemName, 0); - BArray array = DataUtils.createArrayValue(restType); - mapValue.put(StringUtils.fromString(elemName), array); + mapValue.put(StringUtils.fromString(elemName), DataUtils.createArrayValue(restType)); } else { indexes.put(elemName, indexes.get(elemName) + 1); } - Type elementType = ((ArrayType) restType).getElementType(); - if (elementType.getTag() == TypeTags.RECORD_TYPE_TAG) { - handleRecordType(xmlItem, restType, elemName, (RecordType) elementType, mapValue, - analyzerData); + checkRestTypeAndConvert(xmlItem, elemName, restType, ((ArrayType) restType).getElementType(), + mapValue, analyzerData); + } + default -> { + BString bElementName = StringUtils.fromString(elemName); + if (mapValue.containsKey(bElementName) && mapValue.get(bElementName) != null) { + handleArrayValueForRestType(xmlItem, elemName, restType, mapValue, analyzerData); return; } - } - } - BString bElementName = StringUtils.fromString(elemName); - BArray arrayValue; - if (mapValue.containsKey(bElementName) && mapValue.get(bElementName) != null) { - Object currentElement = mapValue.get(bElementName); - if (!(currentElement instanceof BArray)) { - if (!DataUtils.isArrayValueAssignable(restType.getTag())) { - throw DiagnosticLog.error(DiagnosticErrorCode.FOUND_ARRAY_FOR_NON_ARRAY_TYPE, restType, - elemName); - } - arrayValue = DataUtils.createArrayValue(restType); - arrayValue.append(currentElement); - mapValue.put(bElementName, arrayValue); - } else { - arrayValue = (BArray) currentElement; - } - - if (isNextElementContent(xmlItem)) { - if (isElementHasAttributes(xmlItem)) { - BMap nextValue = - ValueCreator.createMapValue(DataUtils.getMapTypeFromConstraintType(restType)); - handleAttributesRest(xmlItem, nextValue, restType); - arrayValue.append(nextValue); - - if (!nextValue.isEmpty()) { - analyzerData.currentField = - TypeCreator.createField(restType, analyzerData.textFieldName, SymbolFlags.REQUIRED); - analyzerData.nodesStack.push(currentNode); - currentNode = nextValue; - traverseXml(xmlItem.getChildrenSeq(), restType, analyzerData); - currentNode = analyzerData.nodesStack.pop(); - return; - } + if (isNextElementContent(xmlItem)) { + convertContentToRestType(xmlItem, bElementName, restType, mapValue, analyzerData); + return; } - traverseXml(xmlItem.getChildrenSeq(), restType, analyzerData); - return; - } - BMap nextValue = - ValueCreator.createMapValue(DataUtils.getMapTypeFromConstraintType(restType)); - if (DataUtils.isAnydataOrJson(restType.getTag())) { - arrayValue.append(nextValue); - } else { - ArrayType arrayType = (ArrayType) restType; - int currentIndex = analyzerData.arrayIndexes.peek().get(elemName); - if (arrayType.getState() == ArrayType.ArrayState.CLOSED - && arrayType.getSize() <= currentIndex) { - DataUtils.logArrayMismatchErrorIfProjectionNotAllowed(analyzerData.allowDataProjection); - } else { - arrayValue.add(currentIndex, nextValue); + + if (restType.getTag() != TypeTags.ANYDATA_TAG && restType.getTag() != TypeTags.JSON_TAG) { + throw DiagnosticLog.error(DiagnosticErrorCode.EXPECTED_ANYDATA_OR_JSON); } - } - analyzerData.nodesStack.push(currentNode); - currentNode = nextValue; - handleAttributesRest(xmlItem, nextValue, restType); - analyzerData.fieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); - analyzerData.visitedFieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); - // TODO: Check and fix this. -// analyzerData.arrayIndexes.push(new HashMap<>()); - if (restType.getTag() == TypeTags.ARRAY_TAG) { - Type memberType = ((ArrayType) restType).getElementType(); - analyzerData.restTypes.push(memberType); - traverseXml(xmlItem.getChildrenSeq(), memberType, analyzerData); - } else { - analyzerData.restTypes.push(restType); - traverseXml(xmlItem.getChildrenSeq(), restType, analyzerData); + convertToJsonOrAnydataAsRestType(xmlItem, bElementName, restType, mapValue, analyzerData); } - analyzerData.fieldHierarchy.pop(); - analyzerData.visitedFieldHierarchy.pop(); - analyzerData.restTypes.pop(); -// analyzerData.arrayIndexes.pop(); - currentNode = analyzerData.nodesStack.pop(); - return; + } + } + + private void handleArrayValueForRestType(BXmlItem xmlItem, String elemName, Type restType, + BMap mapValue, XmlAnalyzerData analyzerData) { + BString bElementName = StringUtils.fromString(elemName); + Object currentElement = mapValue.get(bElementName); + BArray arrayValue; + if (!(currentElement instanceof BArray)) { + if (!DataUtils.isArrayValueAssignable(restType.getTag())) { + throw DiagnosticLog.error(DiagnosticErrorCode.FOUND_ARRAY_FOR_NON_ARRAY_TYPE, restType, + elemName); + } + arrayValue = DataUtils.createArrayValue(restType); + arrayValue.append(currentElement); + mapValue.put(bElementName, arrayValue); + } else { + arrayValue = (BArray) currentElement; } if (isNextElementContent(xmlItem)) { @@ -462,7 +420,7 @@ private void convertWithRestType(BXmlItem xmlItem, Type restType, XmlAnalyzerDat BMap nextValue = ValueCreator.createMapValue(DataUtils.getMapTypeFromConstraintType(restType)); handleAttributesRest(xmlItem, nextValue, restType); - mapValue.put(bElementName, nextValue); + arrayValue.append(nextValue); if (!nextValue.isEmpty()) { analyzerData.currentField = @@ -476,9 +434,67 @@ private void convertWithRestType(BXmlItem xmlItem, Type restType, XmlAnalyzerDat } traverseXml(xmlItem.getChildrenSeq(), restType, analyzerData); return; - } else if (restType.getTag() != TypeTags.ANYDATA_TAG && restType.getTag() != TypeTags.JSON_TAG) { - throw DiagnosticLog.error(DiagnosticErrorCode.EXPECTED_ANYDATA_OR_JSON); } + BMap nextValue = + ValueCreator.createMapValue(DataUtils.getMapTypeFromConstraintType(restType)); + if (DataUtils.isAnydataOrJson(restType.getTag())) { + arrayValue.append(nextValue); + } else { + ArrayType arrayType = (ArrayType) restType; + int currentIndex = analyzerData.arrayIndexes.peek().get(elemName); + if (arrayType.getState() == ArrayType.ArrayState.CLOSED + && arrayType.getSize() <= currentIndex) { + DataUtils.logArrayMismatchErrorIfProjectionNotAllowed(analyzerData.allowDataProjection); + } else { + arrayValue.add(currentIndex, nextValue); + } + } + analyzerData.nodesStack.push(currentNode); + currentNode = nextValue; + handleAttributesRest(xmlItem, nextValue, restType); + + analyzerData.fieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); + analyzerData.visitedFieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); + // TODO: Check and fix this. +// analyzerData.arrayIndexes.push(new HashMap<>()); + if (restType.getTag() == TypeTags.ARRAY_TAG) { + Type memberType = ((ArrayType) restType).getElementType(); + analyzerData.restTypes.push(memberType); + traverseXml(xmlItem.getChildrenSeq(), memberType, analyzerData); + } else { + analyzerData.restTypes.push(restType); + traverseXml(xmlItem.getChildrenSeq(), restType, analyzerData); + } + analyzerData.fieldHierarchy.pop(); + analyzerData.visitedFieldHierarchy.pop(); + analyzerData.restTypes.pop(); +// analyzerData.arrayIndexes.pop(); + currentNode = analyzerData.nodesStack.pop(); + } + + private void convertContentToRestType(BXmlItem xmlItem, BString bElementName, Type restType, + BMap mapValue, XmlAnalyzerData analyzerData) { + if (isElementHasAttributes(xmlItem)) { + BMap nextValue = + ValueCreator.createMapValue(DataUtils.getMapTypeFromConstraintType(restType)); + handleAttributesRest(xmlItem, nextValue, restType); + mapValue.put(bElementName, nextValue); + + if (!nextValue.isEmpty()) { + analyzerData.currentField = + TypeCreator.createField(restType, analyzerData.textFieldName, SymbolFlags.REQUIRED); + analyzerData.nodesStack.push(currentNode); + currentNode = nextValue; + traverseXml(xmlItem.getChildrenSeq(), restType, analyzerData); + currentNode = analyzerData.nodesStack.pop(); + return; + } + } + traverseXml(xmlItem.getChildrenSeq(), restType, analyzerData); + } + + private void convertToJsonOrAnydataAsRestType(BXmlItem xmlItem, BString bElementName, Type restType, + BMap mapValue, XmlAnalyzerData analyzerData) { BMap nextValue = ValueCreator.createMapValue(DataUtils.getMapTypeFromConstraintType(restType)); mapValue.put(bElementName, nextValue); From 1fa2f49ddf3cd0d20a12bce323c086850f3fefdb Mon Sep 17 00:00:00 2001 From: prakanth <50439067+prakanth97@users.noreply.github.com> Date: Wed, 22 May 2024 11:41:11 +0530 Subject: [PATCH 05/13] Add tests for record with default value --- ballerina/tests/fromXml_test.bal | 100 +++++++++++++++++++++++++++++++ 1 file changed, 100 insertions(+) diff --git a/ballerina/tests/fromXml_test.bal b/ballerina/tests/fromXml_test.bal index 6bda3d1..25c32ce 100644 --- a/ballerina/tests/fromXml_test.bal +++ b/ballerina/tests/fromXml_test.bal @@ -2788,6 +2788,106 @@ function testXmlToRecordWithArrayAsFieldType() returns error? { test:assertEquals(rec1.A, [1, 2, 3, 0]); } +type Student record {| + string name; + int age; + string code = "Admitted"; +|}; + +@test:Config +function testXmlToRecordWithDefaultValuesForParseString1() returns error? { + string xmlStr = string ` + + Walter + 27 + + `; + + Student student = check parseString(xmlStr); + test:assertEquals(student.name, "Walter"); + test:assertEquals(student.age, 27); + test:assertEquals(student.code, "Admitted"); +} + +@test:Config +function testXmlToRecordWithDefaultValuesForPasrseAsType1() returns error? { + xml xmlVal = xml ` + + Walter + 27 + + `; + + Student student = check parseAsType(xmlVal); + test:assertEquals(student.name, "Walter"); + test:assertEquals(student.age, 27); + test:assertEquals(student.code, "Admitted"); +} + +type University record {| + Student[] student; + string name; + string category = "State"; +|}; + +@test:Config +function testXmlToRecordWithDefaultValuesForParseString2() returns error? { + string xmlStr = string ` + + + Walter + 27 + + + Jessy + 18 + + Standford + + `; + + University university = check parseString(xmlStr); + test:assertEquals(university.student[0].name, "Walter"); + test:assertEquals(university.student[0].age, 27); + test:assertEquals(university.student[0].code, "Admitted"); + + test:assertEquals(university.student[1].name, "Jessy"); + test:assertEquals(university.student[1].age, 18); + test:assertEquals(university.student[1].code, "Admitted"); + + test:assertEquals(university.name, "Standford"); + test:assertEquals(university.category, "State"); +} + +@test:Config +function testXmlToRecordWithDefaultValuesForParseAsType2() returns error? { + xml xmlVal = xml ` + + + Walter + 27 + + + Jessy + 18 + + Standford + + `; + + University university = check parseAsType(xmlVal); + test:assertEquals(university.student[0].name, "Walter"); + test:assertEquals(university.student[0].age, 27); + test:assertEquals(university.student[0].code, "Admitted"); + + test:assertEquals(university.student[1].name, "Jessy"); + test:assertEquals(university.student[1].age, 18); + test:assertEquals(university.student[1].code, "Admitted"); + + test:assertEquals(university.name, "Standford"); + test:assertEquals(university.category, "State"); +} + // Negative cases type DataN1 record {| int A; From 862c54c199b7aad7ca263bba8711900c6d798561 Mon Sep 17 00:00:00 2001 From: prakanth <50439067+prakanth97@users.noreply.github.com> Date: Wed, 22 May 2024 12:29:52 +0530 Subject: [PATCH 06/13] Remove duplicate code and unreachable code --- .../lib/data/xmldata/utils/DataUtils.java | 5 +- .../lib/data/xmldata/xml/XmlParser.java | 47 ++++++++----------- .../lib/data/xmldata/xml/XmlTraversal.java | 15 ++---- 3 files changed, 27 insertions(+), 40 deletions(-) diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java b/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java index ec2f911..a0da945 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java @@ -288,14 +288,13 @@ public static void updateExpectedTypeStacks(RecordType recordType, XmlAnalyzerDa analyzerData.fieldHierarchy.push(new QualifiedNameMap<>(getAllFieldsInRecordType(recordType, analyzerData))); analyzerData.visitedFieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); analyzerData.restTypes.push(recordType.getRestFieldType()); -// analyzerData.arrayIndexes.push(new HashMap<>()); } - public static void removeExpectedTypeStacks(XmlAnalyzerData analyzerData) { - analyzerData.attributeHierarchy.pop(); + public static void popExpectedTypeStacks(XmlAnalyzerData analyzerData) { analyzerData.fieldHierarchy.pop(); analyzerData.visitedFieldHierarchy.pop(); analyzerData.restTypes.pop(); + analyzerData.attributeHierarchy.pop(); analyzerData.arrayIndexes.pop(); } diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java index 0c24df0..af209f2 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java @@ -199,11 +199,7 @@ public void parseRecordRest(String startElementName, XmlParserData xmlParserData QName endElement = xmlStreamReader.getName(); if (endElement.getLocalPart().equals(startElementName)) { validateRequiredFields(xmlParserData); - xmlParserData.fieldHierarchy.pop(); - xmlParserData.visitedFieldHierarchy.pop(); - xmlParserData.restTypes.pop(); - xmlParserData.attributeHierarchy.pop(); - xmlParserData.arrayIndexes.pop(); + popExpectedTypeStacks(xmlParserData); break; } } @@ -292,10 +288,6 @@ private void readText(XMLStreamReader xmlStreamReader, return; } - if (!DataUtils.isArrayValueAssignable(fieldType.getTag())) { - throw DiagnosticLog.error(DiagnosticErrorCode.FOUND_ARRAY_FOR_NON_ARRAY_TYPE, fieldType, fieldName); - } - ArrayType arrayType = (ArrayType) fieldType; int currentIndex = xmlParserData.arrayIndexes.peek().get(fieldName); if (arrayType.getState() == ArrayType.ArrayState.CLOSED @@ -330,22 +322,17 @@ private void convertTextAndUpdateCurrentNode(BMap currentNode, Object currentElement = currentNode.get(currentFieldName); Object result = convertStringToRestExpType(bText, restType); - if (currentElement == null && !currentNode.isEmpty()) { // Add text to the #content field + if (currentElement == null && !currentNode.isEmpty()) { // Add text to the #content field currentNode.put(StringUtils.fromString(Constants.CONTENT), result); } else if (parent.get(currentFieldName) instanceof BArray bArray) { bArray.add(bArray.getLength() - 1, result); - } else { + } else { parent.put(currentFieldName, result); } xmlParserData.currentNode = parent; - xmlParserData.fieldHierarchy.pop(); - xmlParserData.visitedFieldHierarchy.pop(); - xmlParserData.restTypes.pop(); - xmlParserData.attributeHierarchy.pop(); - xmlParserData.recordTypeStack.pop(); - xmlParserData.siblings = xmlParserData.parents.pop(); - xmlParserData.arrayIndexes.pop(); + popExpectedTypeStacks(xmlParserData); + updateSiblingAndRootRecord(xmlParserData); } @SuppressWarnings("unchecked") @@ -385,7 +372,8 @@ private void handleTruncatedCharacters(XMLStreamReader xmlStreamReader, TextValu @SuppressWarnings("unchecked") private void handleContentFieldInRecordType(RecordType recordType, BString text, XmlParserData xmlParserData) { - popStacks(xmlParserData); + popExpectedTypeStacks(xmlParserData); + updateSiblingAndRootRecord(xmlParserData); for (String key : recordType.getFields().keySet()) { if (key.contains(xmlParserData.textFieldName)) { xmlParserData.currentNode.put(StringUtils.fromString(key), @@ -454,7 +442,8 @@ private void endElement(XMLStreamReader xmlStreamReader, XmlParserData xmlParser validateRequiredFields(xmlParserData); xmlParserData.currentNode = (BMap) xmlParserData.nodesStack.pop(); - popStacks(xmlParserData); + popExpectedTypeStacks(xmlParserData); + updateSiblingAndRootRecord(xmlParserData); } private void validateRequiredFields(XmlParserData xmlParserData) { @@ -665,14 +654,21 @@ private void updateExpectedTypeStacks(RecordType recordType, XmlParserData xmlPa xmlParserData.restTypes.push(recordType.getRestFieldType()); } - private void popStacks(XmlParserData xmlParserData) { + private void popExpectedTypeStacks(XmlParserData xmlParserData) { + popMappingTypeStacks(xmlParserData); + xmlParserData.attributeHierarchy.pop(); + xmlParserData.arrayIndexes.pop(); + } + + private void popMappingTypeStacks(XmlParserData xmlParserData) { xmlParserData.fieldHierarchy.pop(); xmlParserData.visitedFieldHierarchy.pop(); xmlParserData.restTypes.pop(); - xmlParserData.attributeHierarchy.pop(); + } + + private void updateSiblingAndRootRecord(XmlParserData xmlParserData) { xmlParserData.siblings = xmlParserData.parents.pop(); xmlParserData.rootRecord = xmlParserData.recordTypeStack.pop(); - xmlParserData.arrayIndexes.pop(); } @SuppressWarnings("unchecked") @@ -840,11 +836,8 @@ private void endElementRest(XMLStreamReader xmlStreamReader, XmlParserData xmlPa xmlParserData.currentNode = (BMap) xmlParserData.nodesStack.pop(); xmlParserData.siblings = xmlParserData.parents.pop(); if (xmlParserData.siblings.contains(elemQName)) { - // TODO: This is duplicated in several places. Remove the duplication. // TODO: This place behaviour is strange need to check and fix it, Properly. - xmlParserData.fieldHierarchy.pop(); - xmlParserData.visitedFieldHierarchy.pop(); - xmlParserData.restTypes.pop(); + popMappingTypeStacks(xmlParserData); // xmlParserData.attributeHierarchy.pop(); xmlParserData.arrayIndexes.pop(); } diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java index 455f38b..7daca26 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlTraversal.java @@ -138,10 +138,6 @@ private void convertText(String text, XmlAnalyzerData analyzerData) { mapValue.put(fieldName, convertedValue); return; } - // TODO: Check do we need this check. - if (!DataUtils.isArrayValueAssignable(fieldType.getTag())) { - throw DiagnosticLog.error(DiagnosticErrorCode.FOUND_ARRAY_FOR_NON_ARRAY_TYPE, fieldType, fieldName); - } if (DataUtils.isAnydataOrJson(fieldType.getTag())) { ((BArray) value).append(convertedValue); @@ -221,7 +217,7 @@ private void convertToFieldType(BXmlItem xmlItem, Field currentField, String fie updateNextMap(currentFieldType, analyzerData); analyzerData.arrayIndexes.push(new HashMap<>()); convertToRestType(xmlItem, currentFieldType, analyzerData); - DataUtils.removeExpectedTypeStacks(analyzerData); + DataUtils.popExpectedTypeStacks(analyzerData); } case TypeTags.TYPE_REFERENCED_TYPE_TAG -> convertToFieldType(xmlItem, currentField, fieldName, TypeUtils.getReferredType(currentFieldType), @@ -280,7 +276,7 @@ private void convertToRecordType(BXmlItem xmlItem, Type currentFieldType, String analyzerData.rootRecord = elementType; traverseXml(xmlItem.getChildrenSeq(), currentFieldType, analyzerData); DataUtils.validateRequiredFields(analyzerData); - DataUtils.removeExpectedTypeStacks(analyzerData); + DataUtils.popExpectedTypeStacks(analyzerData); analyzerData.rootRecord = prevRecord; currentNode = analyzerData.nodesStack.pop(); } @@ -291,7 +287,7 @@ private void convertToMapType(BXmlItem xmlItem, Type fieldType, Type elementType currentNode = updateNextMappingValue(elementType, fieldName, fieldType, mapValue, analyzerData); traverseXml(xmlItem.getChildrenSeq(), fieldType, analyzerData); DataUtils.validateRequiredFields(analyzerData); - DataUtils.removeExpectedTypeStacks(analyzerData); + DataUtils.popExpectedTypeStacks(analyzerData); currentNode = analyzerData.nodesStack.pop(); } @@ -455,8 +451,7 @@ private void handleArrayValueForRestType(BXmlItem xmlItem, String elemName, Type analyzerData.fieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); analyzerData.visitedFieldHierarchy.push(new QualifiedNameMap<>(new HashMap<>())); - // TODO: Check and fix this. -// analyzerData.arrayIndexes.push(new HashMap<>()); + analyzerData.arrayIndexes.push(new HashMap<>()); if (restType.getTag() == TypeTags.ARRAY_TAG) { Type memberType = ((ArrayType) restType).getElementType(); analyzerData.restTypes.push(memberType); @@ -468,7 +463,7 @@ private void handleArrayValueForRestType(BXmlItem xmlItem, String elemName, Type analyzerData.fieldHierarchy.pop(); analyzerData.visitedFieldHierarchy.pop(); analyzerData.restTypes.pop(); -// analyzerData.arrayIndexes.pop(); + analyzerData.arrayIndexes.pop(); currentNode = analyzerData.nodesStack.pop(); } From 5c9d2b231f321f76ca4f2042742aa1663896eb65 Mon Sep 17 00:00:00 2001 From: prakanth <50439067+prakanth97@users.noreply.github.com> Date: Thu, 11 Jul 2024 14:58:18 +0530 Subject: [PATCH 07/13] Fix fail when attribute and element have same name --- ballerina/tests/fromXml_test.bal | 10 +++++----- .../io/ballerina/lib/data/xmldata/xml/XmlParser.java | 2 ++ 2 files changed, 7 insertions(+), 5 deletions(-) diff --git a/ballerina/tests/fromXml_test.bal b/ballerina/tests/fromXml_test.bal index 75701d1..ea445bc 100644 --- a/ballerina/tests/fromXml_test.bal +++ b/ballerina/tests/fromXml_test.bal @@ -994,18 +994,18 @@ type RecAtt5 record {| |}; @test:Config { - groups: ["fromXmlString", "testFail"] + groups: ["fromXmlString"] } function testXmlStringToRecord38() returns error? { string xmlStr = string `1`; RecAtt3 rec = check parseString(xmlStr); test:assertEquals(rec.A, "1"); - // RecAtt4 rec2 = check parseString(xmlStr); - // test:assertEquals(rec2.A.get("#content"), 1); + RecAtt4 rec2 = check parseString(xmlStr); + test:assertEquals(rec2.A.get("#content"), 1); - // RecAtt5 rec3 = check parseString(xmlStr); - // test:assertEquals(rec3.A, "name"); + RecAtt5 rec3 = check parseString(xmlStr); + test:assertEquals(rec3.A, "name"); } @test:Config { diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java index 3f7ec19..8d60671 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java @@ -1008,6 +1008,8 @@ private void handleAttributes(XMLStreamReader xmlStreamReader, XmlParserData xml ATTRIBUTE); Field field = xmlParserData.attributeHierarchy.peek().remove(attQName); if (field == null) { + // Here attQName state is set to NOT_DEFINED since it accessed from field hierarchy. + attQName.setAttributeState(NOT_DEFINED); Optional f = getFieldFromFieldHierarchy(attQName, xmlParserData); if (f.isEmpty()) { continue; From d01767f707c5f1df0db407c48aa910afcfee1174 Mon Sep 17 00:00:00 2001 From: prakanth <50439067+prakanth97@users.noreply.github.com> Date: Thu, 11 Jul 2024 15:13:50 +0530 Subject: [PATCH 08/13] Refactor code --- .../lib/data/xmldata/utils/DataUtils.java | 15 ++------------- .../ballerina/lib/data/xmldata/xml/XmlParser.java | 1 - 2 files changed, 2 insertions(+), 14 deletions(-) diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java b/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java index 9dd62de..c10ace8 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/utils/DataUtils.java @@ -237,15 +237,13 @@ public static Object convertStringToExpType(BString value, Type expType) { } public static void validateRequiredFields(XmlAnalyzerData analyzerData) { - Map remainingFields = analyzerData.fieldHierarchy.peek().getMembers(); - for (Field field : remainingFields.values()) { + for (Field field : analyzerData.fieldHierarchy.peek().getMembers().values()) { if (SymbolFlags.isFlagOn(field.getFlags(), SymbolFlags.REQUIRED)) { throw DiagnosticLog.error(DiagnosticErrorCode.REQUIRED_FIELD_NOT_PRESENT, field.getFieldName()); } } - Map remainingAttributes = analyzerData.attributeHierarchy.peek().getMembers(); - for (Field attribute : remainingAttributes.values()) { + for (Field attribute : analyzerData.attributeHierarchy.peek().getMembers().values()) { if (!SymbolFlags.isFlagOn(attribute.getFlags(), SymbolFlags.OPTIONAL)) { throw DiagnosticLog.error(DiagnosticErrorCode.REQUIRED_ATTRIBUTE_NOT_PRESENT, attribute.getFieldName()); } @@ -260,15 +258,6 @@ public static boolean isStringValueAssignable(int typeTag) { return typeTag == TypeTags.STRING_TAG || typeTag == TypeTags.ANYDATA_TAG || typeTag == TypeTags.JSON_TAG; } - public static ArrayType getValidArrayType(Type type) { - return switch (type.getTag()) { - case TypeTags.ARRAY_TAG -> (ArrayType) type; - case TypeTags.ANYDATA_TAG -> PredefinedTypes.TYPE_ANYDATA_ARRAY; - case TypeTags.JSON_TAG -> PredefinedTypes.TYPE_JSON_ARRAY; - default -> null; - }; - } - public static BArray createArrayValue(Type type) { return switch (type.getTag()) { case TypeTags.ARRAY_TAG -> ValueCreator.createArrayValue((ArrayType) type); diff --git a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java index 8d60671..22d8e54 100644 --- a/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java +++ b/native/src/main/java/io/ballerina/lib/data/xmldata/xml/XmlParser.java @@ -347,7 +347,6 @@ private void addTextToCurrentNodeIfExpTypeIsArray(ArrayType fieldType, BString b bText, xmlParserData); case TypeTags.ANYDATA_TAG, TypeTags.JSON_TAG -> { BArray tempArr = (BArray) ((BMap) xmlParserData.nodesStack.peek()).get(bFieldName); - // TODO: Using this -2 here seems strange for me. Find a better way to handle this. HashMap indexes = xmlParserData.arrayIndexes.get(xmlParserData.arrayIndexes.size() - 2); int currentIndex = indexes.get(bFieldName.getValue()); From e32a315f995864d55ac1ec6706bef969d28789fa Mon Sep 17 00:00:00 2001 From: prakanth <50439067+prakanth97@users.noreply.github.com> Date: Wed, 24 Jul 2024 16:57:16 +0530 Subject: [PATCH 09/13] Update to stable version --- ballerina/Ballerina.toml | 2 +- build-config/resources/Ballerina.toml | 2 +- gradle.properties | 4 ++-- 3 files changed, 4 insertions(+), 4 deletions(-) diff --git a/ballerina/Ballerina.toml b/ballerina/Ballerina.toml index 1f904e9..b3e1869 100644 --- a/ballerina/Ballerina.toml +++ b/ballerina/Ballerina.toml @@ -1,7 +1,7 @@ [package] org = "ballerina" name = "data.xmldata" -version = "0.1.4" +version = "1.0.0" authors = ["Ballerina"] keywords = ["xml"] repository = "https://github.com/ballerina-platform/module-ballerina-data-xmldata" diff --git a/build-config/resources/Ballerina.toml b/build-config/resources/Ballerina.toml index 741191d..5f14962 100644 --- a/build-config/resources/Ballerina.toml +++ b/build-config/resources/Ballerina.toml @@ -6,7 +6,7 @@ authors = ["Ballerina"] keywords = ["xml"] repository = "https://github.com/ballerina-platform/module-ballerina-data-xmldata" license = ["Apache-2.0"] -distribution = "2201.9.0" +distribution = "2201.10.0" export = ["data.xmldata"] [[platform.java17.dependency]] diff --git a/gradle.properties b/gradle.properties index 89e0012..89a09c9 100644 --- a/gradle.properties +++ b/gradle.properties @@ -1,7 +1,7 @@ org.gradle.caching=true group=io.ballerina.lib -version=0.1.4-SNAPSHOT -ballerinaLangVersion=2201.9.0 +version=1.0.0-SNAPSHOT +ballerinaLangVersion=2201.10.0-20240724-114000-40e856f7 checkstyleToolVersion=10.12.0 puppycrawlCheckstyleVersion=10.12.0 From e213d59e5473343af80e29b0542804fdf176e80a Mon Sep 17 00:00:00 2001 From: prakanth <50439067+prakanth97@users.noreply.github.com> Date: Wed, 24 Jul 2024 16:58:24 +0530 Subject: [PATCH 10/13] [Automated] Update the native jar versions --- ballerina/Ballerina.toml | 6 +++--- ballerina/CompilerPlugin.toml | 2 +- ballerina/Dependencies.toml | 4 ++-- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/ballerina/Ballerina.toml b/ballerina/Ballerina.toml index b3e1869..b424cf2 100644 --- a/ballerina/Ballerina.toml +++ b/ballerina/Ballerina.toml @@ -6,11 +6,11 @@ authors = ["Ballerina"] keywords = ["xml"] repository = "https://github.com/ballerina-platform/module-ballerina-data-xmldata" license = ["Apache-2.0"] -distribution = "2201.9.0" +distribution = "2201.10.0" export = ["data.xmldata"] [[platform.java17.dependency]] groupId = "io.ballerina.lib" artifactId = "data-native" -version = "0.1.4" -path = "../native/build/libs/data.xmldata-native-0.1.4-SNAPSHOT.jar" +version = "1.0.0" +path = "../native/build/libs/data.xmldata-native-1.0.0-SNAPSHOT.jar" diff --git a/ballerina/CompilerPlugin.toml b/ballerina/CompilerPlugin.toml index 2585887..e717227 100644 --- a/ballerina/CompilerPlugin.toml +++ b/ballerina/CompilerPlugin.toml @@ -3,4 +3,4 @@ id = "constraint-compiler-plugin" class = "io.ballerina.lib.data.xmldata.compiler.XmldataCompilerPlugin" [[dependency]] -path = "../compiler-plugin/build/libs/data.xmldata-compiler-plugin-0.1.4-SNAPSHOT.jar" +path = "../compiler-plugin/build/libs/data.xmldata-compiler-plugin-1.0.0-SNAPSHOT.jar" diff --git a/ballerina/Dependencies.toml b/ballerina/Dependencies.toml index e994f42..c3c6683 100644 --- a/ballerina/Dependencies.toml +++ b/ballerina/Dependencies.toml @@ -5,12 +5,12 @@ [ballerina] dependencies-toml-version = "2" -distribution-version = "2201.9.0" +distribution-version = "2201.10.0-20240724-114000-40e856f7" [[package]] org = "ballerina" name = "data.xmldata" -version = "0.1.4" +version = "1.0.0" dependencies = [ {org = "ballerina", name = "io"}, {org = "ballerina", name = "jballerina.java"}, From d0a26d9c685c3290541b954f9e389a52e37ff046 Mon Sep 17 00:00:00 2001 From: ballerina-bot Date: Thu, 25 Jul 2024 04:45:49 +0000 Subject: [PATCH 11/13] [Automated] Update the native jar versions --- ballerina/Ballerina.toml | 2 +- ballerina/CompilerPlugin.toml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/ballerina/Ballerina.toml b/ballerina/Ballerina.toml index 1f904e9..7896a5c 100644 --- a/ballerina/Ballerina.toml +++ b/ballerina/Ballerina.toml @@ -13,4 +13,4 @@ export = ["data.xmldata"] groupId = "io.ballerina.lib" artifactId = "data-native" version = "0.1.4" -path = "../native/build/libs/data.xmldata-native-0.1.4-SNAPSHOT.jar" +path = "../native/build/libs/data.xmldata-native-0.1.4.jar" diff --git a/ballerina/CompilerPlugin.toml b/ballerina/CompilerPlugin.toml index 2585887..133eede 100644 --- a/ballerina/CompilerPlugin.toml +++ b/ballerina/CompilerPlugin.toml @@ -3,4 +3,4 @@ id = "constraint-compiler-plugin" class = "io.ballerina.lib.data.xmldata.compiler.XmldataCompilerPlugin" [[dependency]] -path = "../compiler-plugin/build/libs/data.xmldata-compiler-plugin-0.1.4-SNAPSHOT.jar" +path = "../compiler-plugin/build/libs/data.xmldata-compiler-plugin-0.1.4.jar" From 332153c402518d6a41577bafa490173eda0db087 Mon Sep 17 00:00:00 2001 From: ballerina-bot Date: Thu, 25 Jul 2024 04:45:49 +0000 Subject: [PATCH 12/13] [Gradle Release Plugin] - pre tag commit: 'v0.1.4'. --- gradle.properties | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/gradle.properties b/gradle.properties index 89e0012..5c92179 100644 --- a/gradle.properties +++ b/gradle.properties @@ -1,6 +1,6 @@ org.gradle.caching=true group=io.ballerina.lib -version=0.1.4-SNAPSHOT +version=0.1.4 ballerinaLangVersion=2201.9.0 checkstyleToolVersion=10.12.0 From 3c60b2becd56df0852686061e3896a6932566bb5 Mon Sep 17 00:00:00 2001 From: ballerina-bot Date: Thu, 25 Jul 2024 04:45:51 +0000 Subject: [PATCH 13/13] [Gradle Release Plugin] - new version commit: 'v0.1.5-SNAPSHOT'. --- gradle.properties | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/gradle.properties b/gradle.properties index 5c92179..78a2cb2 100644 --- a/gradle.properties +++ b/gradle.properties @@ -1,6 +1,6 @@ org.gradle.caching=true group=io.ballerina.lib -version=0.1.4 +version=0.1.5-SNAPSHOT ballerinaLangVersion=2201.9.0 checkstyleToolVersion=10.12.0