From 2b00ec8e70ad5a9e27b121c9f6d3e9abde5d2d8a Mon Sep 17 00:00:00 2001
From: ebensza <bence.janos.szabo@ericsson.com>
Date: Mon, 20 Jun 2016 12:19:30 +0200
Subject: [PATCH] xsd2ttcn: list length restriction corrected (Bug 495990)

Signed-off-by: ebensza <bence.janos.szabo@ericsson.com>
---
 .../www_example_org_list_complextype_e.ttcn   | 26 +++++++++
 .../www_example_org_list_simpletype_e.ttcn    | 14 +++++
 .../XML/XmlWorkflow/xsd/list_complextype.xsd  | 56 +++++++++++++++++++
 .../XML/XmlWorkflow/xsd/list_simpletype.xsd   | 26 +++++++++
 xsdconvert/ComplexType.cc                     | 35 +++++++-----
 xsdconvert/ComplexType.hh                     |  3 -
 xsdconvert/SimpleType.cc                      | 17 ++++--
 xsdconvert/SimpleType.hh                      | 14 ++++-
 xsdconvert/XMLParser.cc                       | 11 +++-
 9 files changed, 177 insertions(+), 25 deletions(-)

diff --git a/regression_test/XML/XmlWorkflow/XmlTest_expectedTtcns/www_example_org_list_complextype_e.ttcn b/regression_test/XML/XmlWorkflow/XmlTest_expectedTtcns/www_example_org_list_complextype_e.ttcn
index 1bf29018a..7f075be61 100644
--- a/regression_test/XML/XmlWorkflow/XmlTest_expectedTtcns/www_example_org_list_complextype_e.ttcn
+++ b/regression_test/XML/XmlWorkflow/XmlTest_expectedTtcns/www_example_org_list_complextype_e.ttcn
@@ -60,6 +60,13 @@ with {
 };
 
 
+type record length(3) of XSD.String ElementWithList2 length(4)
+with {
+  variant "list";
+  variant "element";
+};
+
+
 type record ChoiceWithList
 {
 	union {
@@ -76,6 +83,25 @@ with {
 };
 
 
+type record ChoiceWithList2
+{
+	union {
+		record length(3) of XSD.Float element1,
+		record of XSD.String element2,
+		record length(3) of XSD.String element3 length(4)
+	} choice
+}
+with {
+  variant (choice) "untagged";
+  variant (choice.element1) "name as capitalized";
+  variant (choice.element1) "list";
+  variant (choice.element2) "name as capitalized";
+  variant (choice.element2) "list";
+  variant (choice.element3) "name as capitalized";
+  variant (choice.element3) "list";
+};
+
+
 type XSD.Token Ipv4 (pattern "IPV4")
 with {
   variant "name as uncapitalized";
diff --git a/regression_test/XML/XmlWorkflow/XmlTest_expectedTtcns/www_example_org_list_simpletype_e.ttcn b/regression_test/XML/XmlWorkflow/XmlTest_expectedTtcns/www_example_org_list_simpletype_e.ttcn
index 768f00dd2..e9ab5da14 100644
--- a/regression_test/XML/XmlWorkflow/XmlTest_expectedTtcns/www_example_org_list_simpletype_e.ttcn
+++ b/regression_test/XML/XmlWorkflow/XmlTest_expectedTtcns/www_example_org_list_simpletype_e.ttcn
@@ -44,6 +44,13 @@ module www_example_org_list_simpletype {
 import from XSD all;
 
 
+type record length(3) of XSD.String ListLength length(4)
+with {
+  variant "name as uncapitalized";
+  variant "list";
+};
+
+
 type record of enumerated
 {
 	orange,
@@ -114,6 +121,13 @@ with {
 };
 
 
+type record length(3) of XSD.Float ElementWithList
+with {
+  variant "list";
+  variant "element";
+};
+
+
 }
 with {
   encode "XML";
diff --git a/regression_test/XML/XmlWorkflow/xsd/list_complextype.xsd b/regression_test/XML/XmlWorkflow/xsd/list_complextype.xsd
index 1889d039d..f69db00d8 100644
--- a/regression_test/XML/XmlWorkflow/xsd/list_complextype.xsd
+++ b/regression_test/XML/XmlWorkflow/xsd/list_complextype.xsd
@@ -43,6 +43,24 @@
 	</xs:simpleType>
 </xs:element>
 
+<xs:element name="ElementWithList2">
+	<xs:simpleType>
+		<xs:restriction>
+			<xs:simpleType>
+				<xs:list>
+					<xs:simpleType>
+						<xs:restriction base="xs:string">
+							<xs:length value="4" />
+						</xs:restriction>
+					</xs:simpleType>
+				</xs:list>    
+			</xs:simpleType>
+			<xs:minLength value="3" />
+			<xs:maxLength value="3" />
+		</xs:restriction>
+	</xs:simpleType>
+</xs:element>
+
 <xs:complexType name="ChoiceWithList">
 	<xs:choice>
 		<xs:element name="Element1">
@@ -64,6 +82,44 @@
 	</xs:choice>
 </xs:complexType>
 
+<xs:complexType name="ChoiceWithList2">
+	<xs:choice>
+		<xs:element name="Element1">
+			<xs:simpleType>
+				<xs:restriction>
+					<xs:simpleType>
+						<xs:list itemType="xs:float" />
+					</xs:simpleType>
+					<xs:minLength value="3" />
+					<xs:maxLength value="3" />
+				</xs:restriction>
+			</xs:simpleType>
+		</xs:element>
+		<xs:element name="Element2">
+			<xs:simpleType>
+				<xs:list itemType="xs:string" />
+			</xs:simpleType>
+		</xs:element>
+		<xs:element name="Element3">
+			<xs:simpleType>
+				<xs:restriction>
+					<xs:simpleType>
+						<xs:list>
+							<xs:simpleType>
+								<xs:restriction base="string">
+									<xs:length value="4" />
+								</xs:restriction>
+							</xs:simpleType>
+						</xs:list> 
+					</xs:simpleType>
+					<xs:minLength value="3" />
+					<xs:maxLength value="3" />
+				</xs:restriction>
+			</xs:simpleType>
+		</xs:element>
+	</xs:choice>
+</xs:complexType>
+
 <xs:simpleType name="ipv4">
 	<xs:restriction base="xs:token">
 		<xs:pattern value="IPV4"/>
diff --git a/regression_test/XML/XmlWorkflow/xsd/list_simpletype.xsd b/regression_test/XML/XmlWorkflow/xsd/list_simpletype.xsd
index 6cd1129b5..b67ea6c12 100644
--- a/regression_test/XML/XmlWorkflow/xsd/list_simpletype.xsd
+++ b/regression_test/XML/XmlWorkflow/xsd/list_simpletype.xsd
@@ -4,6 +4,21 @@
  xmlns:this="www.example.org/list/simpletype"
  targetNamespace="www.example.org/list/simpletype">
 
+<xsd:simpleType name="listLength">
+	<xsd:restriction>
+		<xsd:simpleType>
+			<xsd:list>
+				<xsd:simpleType>
+					<xsd:restriction base="xsd:string">
+						<xsd:length value="4" />
+					</xsd:restriction>
+				</xsd:simpleType>
+			</xsd:list>    
+		</xsd:simpleType>
+		<xsd:length value="3"/>
+	</xsd:restriction>
+</xsd:simpleType>
+
 <xsd:element name="SimpleListEnumeration">
 	<xsd:simpleType>
 		<xsd:list>
@@ -84,6 +99,17 @@
 	</xsd:complexType>
 </xsd:element>
 
+<xsd:element name="ElementWithList">
+	<xsd:simpleType>
+		<xsd:restriction>
+			<xsd:simpleType>
+				<xsd:list itemType="xsd:float" />
+			</xsd:simpleType>
+			<xsd:minLength value="3" />
+			<xsd:maxLength value="3" />
+		</xsd:restriction>
+	</xsd:simpleType>
+</xsd:element>
 
 
 
diff --git a/xsdconvert/ComplexType.cc b/xsdconvert/ComplexType.cc
index 7c56c8955..130189a24 100644
--- a/xsdconvert/ComplexType.cc
+++ b/xsdconvert/ComplexType.cc
@@ -35,7 +35,6 @@ ComplexType::ComplexType(XMLParser * a_parser, TTCN3Module * a_module, Construct
 , fromAll(false)
 , max_alt(0)
 , skipback(0)
-, list(false)
 , lastType()
 , actualPath(empty_string)
 , actfield(this)
@@ -63,7 +62,6 @@ ComplexType::ComplexType(ComplexType & other)
 , fromAll(other.fromAll)
 , max_alt(other.max_alt)
 , skipback(other.skipback)
-, list(other.list)
 , lastType(other.lastType)
 , actualPath(other.actualPath)
 , actfield(this)
@@ -111,7 +109,6 @@ ComplexType::ComplexType(ComplexType * other)
 , fromAll(false)
 , max_alt(0)
 , skipback(0)
-, list(false)
 , lastType()
 , actualPath(empty_string)
 , actfield(this)
@@ -141,7 +138,6 @@ ComplexType::ComplexType(const SimpleType & other, CT_fromST c)
 , fromAll(false)
 , max_alt(0)
 , skipback(0)
-, list(false)
 , lastType()
 , actualPath(empty_string)
 , actfield(this)
@@ -301,6 +297,12 @@ void ComplexType::loadWithValues() {
         complexfields.push_back(f);
         basefield = f;
         actfield = f;
+        
+        // If it is a restriction of a list, then no new basefield will be 
+        // present, to we apply the references to the parent.
+        if(parent != NULL && parent->inList) {
+          parent->applyReference(*f, true);
+        }
       } else if (cmode == CT_complextype_mode) {
         setReference(atts.base);
         xsdtype = n_restriction;
@@ -585,12 +587,14 @@ void ComplexType::loadWithValues() {
       break;
     case n_list:
       if (parent != NULL && parent->basefield == this) {
-        parent->list = true;
+        parent->inList = true;
         parent->SimpleType::loadWithValues();
         parent->basefield = NULL;
         setInvisible();
-        break;
+      } else if(parent != NULL) {
+        SimpleType::loadWithValues();
       }
+      break;
     case n_length:
     case n_minLength:
     case n_maxLength:
@@ -626,12 +630,6 @@ void ComplexType::modifyValues() {
     skipback = skipback - 1;
   }
   
-  //embedded simpletype with list in element
-  if(xsdtype == n_simpleType && list) {
-    list = false;
-    return;
-  }
-  
 
   if ( parent != NULL && 
       (xsdtype == n_element || 
@@ -645,7 +643,7 @@ void ComplexType::modifyValues() {
        xsdtype == n_attributeGroup || 
        xsdtype == n_extension || 
        xsdtype == n_restriction || 
-       (xsdtype == n_simpleType && !list) || 
+       (xsdtype == n_simpleType && !inList) || 
        xsdtype == n_simpleContent ||
        (xsdtype == n_sequence && skipback < 0)
       )) {
@@ -659,6 +657,9 @@ void ComplexType::modifyValues() {
       parent->lastType = xsdtype;
     }
   }
+  if (xsdtype == n_simpleType) {
+    inList = false;
+  }
 }
 
 void ComplexType::referenceResolving() {
@@ -1869,7 +1870,13 @@ void ComplexType::resolveSimpleTypeRestriction() {
       addNameSpaceAsVariant(this, ct);
     }
     if(!basefield->parent->top){
-      applyReference(*basefield, true);
+      // This is the case of restriction -> list -> simpletype -> restriction
+      // we have to apply the reference to the parent's parent.
+      if(basefield->parent->parent != NULL && !basefield->parent->isVisible()) {
+        basefield->parent->parent->applyReference(*basefield, true);
+      } else {
+        applyReference(*basefield, true);
+      }
       basefield->setInvisible();
     }
   }
diff --git a/xsdconvert/ComplexType.hh b/xsdconvert/ComplexType.hh
index 08ad6568b..7ce650dd1 100644
--- a/xsdconvert/ComplexType.hh
+++ b/xsdconvert/ComplexType.hh
@@ -71,9 +71,6 @@ private:
   bool fromAll;
   unsigned max_alt;
   int skipback;
-  //Special use: true if an element of a complextype is restricted to a list
-  // in a simpletype
-  bool list;
   TagName lastType;
   Mstring actualPath;
   RootType * actfield;
diff --git a/xsdconvert/SimpleType.cc b/xsdconvert/SimpleType.cc
index 73a45fd5c..5021f7e2e 100644
--- a/xsdconvert/SimpleType.cc
+++ b/xsdconvert/SimpleType.cc
@@ -46,6 +46,7 @@ SimpleType::SimpleType(XMLParser * a_parser, TTCN3Module * a_module, ConstructTy
 , typeSubsGroup(NULL)
 , addedToTypeSubstitution(false)
 , block(not_set)
+, inList(false)
 , parent(NULL) {
 }
 
@@ -70,6 +71,7 @@ SimpleType::SimpleType(const SimpleType& other)
 , typeSubsGroup(other.typeSubsGroup)
 , addedToTypeSubstitution(other.addedToTypeSubstitution)
 , block(other.block)
+, inList(other.inList)
 , parent(NULL) {
   length.parent = this;
   pattern.parent = this;
@@ -93,6 +95,7 @@ void SimpleType::loadWithValues() {
       setMaxOccurs(ULLONG_MAX);
       addVariant(V_list);
       mode = listMode;
+      inList = true;
       break;
     case n_union:
     { // generating complextype from simpletype
@@ -132,7 +135,7 @@ void SimpleType::loadWithValues() {
       break;
     }
     case n_length:
-      if (mode == listMode) {
+       if (inList && (xsdtype != n_NOTSET || mode == restrictionAfterListMode)) {
         setMinOccurs(strtoull(atts.value.c_str(), NULL, 0));
         setMaxOccurs(strtoull(atts.value.c_str(), NULL, 0));
         break;
@@ -142,7 +145,7 @@ void SimpleType::loadWithValues() {
       length.modified = true;
       break;
     case n_minLength:
-      if (mode == listMode) {
+      if (inList && (xsdtype != n_NOTSET || mode == restrictionAfterListMode)) {
         setMinOccurs(strtoull(atts.value.c_str(), NULL, 0));
         break;
       }
@@ -150,7 +153,7 @@ void SimpleType::loadWithValues() {
       length.modified = true;
       break;
     case n_maxLength:
-      if (mode == listMode) {
+      if (inList && (xsdtype != n_NOTSET || mode == restrictionAfterListMode)) {
         setMaxOccurs(strtoull(atts.value.c_str(), NULL, 0));
         break;
       }
@@ -591,7 +594,7 @@ void SimpleType::referenceForST(SimpleType * found_ST) {
   if (construct == c_element)
     return;
 
-  if (mode == listMode)
+  if (mode == listMode || mode == restrictionAfterListMode)
     return;
 
   if (!found_ST->builtInBase.empty()) {
@@ -605,7 +608,7 @@ void SimpleType::referenceForST(SimpleType * found_ST) {
   value.applyReference(found_ST->value);
 
   mode = found_ST->mode;
-  if (found_ST->mode != listMode) {
+  if (found_ST->mode != listMode && found_ST->mode != restrictionAfterListMode) {
     type.upload(found_ST->getType().convertedValue);
   }
 }
@@ -877,7 +880,7 @@ LengthType::LengthType(SimpleType * a_simpleType)
 , upper(ULLONG_MAX) {
 }
 
-void LengthType::applyReference(const LengthType & other) {
+void LengthType::applyReference(const LengthType & other) { 
   if (!modified) modified = other.modified;
   if (other.facet_minLength > facet_minLength) facet_minLength = other.facet_minLength;
   if (other.facet_maxLength < facet_maxLength) facet_maxLength = other.facet_maxLength;
@@ -903,6 +906,7 @@ void LengthType::applyFacets() // only for string types and list types without Q
     }
     case SimpleType::extensionMode:
     case SimpleType::listMode:
+    case SimpleType::restrictionAfterListMode:
       lower = facet_minLength;
       upper = facet_maxLength;
       break;
@@ -1603,3 +1607,4 @@ void ValueType::printToFile(FILE * file) const {
 
   fputc(')', file);
 }
+
diff --git a/xsdconvert/SimpleType.hh b/xsdconvert/SimpleType.hh
index 5c89eb376..9a141048f 100644
--- a/xsdconvert/SimpleType.hh
+++ b/xsdconvert/SimpleType.hh
@@ -200,7 +200,8 @@ public:
     noMode,
     restrictionMode,
     extensionMode,
-    listMode
+    listMode,
+    restrictionAfterListMode
   };
 
 protected:
@@ -236,6 +237,12 @@ protected:
   bool addedToTypeSubstitution;
   BlockValue block;
   
+  //Special use: true if an element of a complexType is restricted to a list
+  // in a simpleType, or a simpleType is a restriction of a list, which is a 
+  // restriction of a simpleType
+  bool inList;
+  // We are inside a list if inList is true and mode == listMode
+  
   //Element substitution
   void addToSubstitutions();
 
@@ -370,6 +377,10 @@ public:
       return block;
   }
   
+  void setList(const bool value) {
+      inList = value;
+  }
+  
   void addToNameDepList(SimpleType * t) {
       //If the type has a substitution, we add the namedep to the substitution
       if(subsGroup != NULL && this != (SimpleType*)subsGroup){
@@ -386,3 +397,4 @@ public:
 };
 
 #endif /* SIMPLETYPE_H_ */
+
diff --git a/xsdconvert/XMLParser.cc b/xsdconvert/XMLParser.cc
index 3a99d8576..d9857d9e9 100644
--- a/xsdconvert/XMLParser.cc
+++ b/xsdconvert/XMLParser.cc
@@ -353,7 +353,15 @@ void XMLParser::endelementHandler(const xmlChar * localname) {
      tag == n_documentation){
     inside_annotation.pop_back();
   }
-
+  
+  if(tag == n_list) {
+      if(module->hasDefinedMainType()) {
+          SimpleType& st = (SimpleType&)(module->getLastMainType());
+          if(st.getXsdtype() == n_NOTSET){
+            st.setMode(SimpleType::restrictionAfterListMode);
+          }
+      }
+  }
 
   --actualDepth;
   if (actualDepth == 0 || actualDepth == 1) {
@@ -836,3 +844,4 @@ void XMLParser::TagAttributes::fillUp(TagAttributeName * att_name_e, Mstring * a
     }
   }
 }
+
-- 
GitLab