qapi.py: Fix schema parser to check syntax systematically
authorMarkus Armbruster <armbru@redhat.com>
Sat, 27 Jul 2013 15:41:58 +0000 (17:41 +0200)
committerAnthony Liguori <aliguori@us.ibm.com>
Mon, 29 Jul 2013 15:37:11 +0000 (10:37 -0500)
Fixes at least the following parser bugs:

* accepts any token in place of a colon

* treats comma as optional

* crashes when closing braces or brackets are missing

Signed-off-by: Markus Armbruster <armbru@redhat.com>
Reviewed-by: Eric Blake <eblake@redhat.com>
Message-id: 1374939721-7876-7-git-send-email-armbru@redhat.com
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
18 files changed:
scripts/qapi.py
tests/qapi-schema/missing-colon.err
tests/qapi-schema/missing-colon.exit
tests/qapi-schema/missing-colon.out
tests/qapi-schema/missing-comma-list.err
tests/qapi-schema/missing-comma-list.exit
tests/qapi-schema/missing-comma-list.out
tests/qapi-schema/missing-comma-object.err
tests/qapi-schema/missing-comma-object.exit
tests/qapi-schema/missing-comma-object.out
tests/qapi-schema/trailing-comma-list.err
tests/qapi-schema/trailing-comma-list.exit
tests/qapi-schema/trailing-comma-list.out
tests/qapi-schema/trailing-comma-object.err
tests/qapi-schema/trailing-comma-object.exit
tests/qapi-schema/trailing-comma-object.out
tests/qapi-schema/unclosed-list.err
tests/qapi-schema/unclosed-object.err

index 0b48a1e50a7b14c066faa3fa804cfdce1827dd26..12fb29a04773e3eb981e114da897c47e5eeeb49f 100644 (file)
@@ -106,24 +106,42 @@ class QAPISchema:
 
     def get_members(self):
         expr = OrderedDict()
-        while self.tok != '}':
+        if self.tok == '}':
+            self.accept()
+            return expr
+        if self.tok != "'":
+            raise QAPISchemaError(self, 'Expected string or "}"')
+        while True:
             key = self.val
             self.accept()
-            self.accept()        # :
+            if self.tok != ':':
+                raise QAPISchemaError(self, 'Expected ":"')
+            self.accept()
             expr[key] = self.get_expr()
-            if self.tok == ',':
+            if self.tok == '}':
                 self.accept()
-        self.accept()
-        return expr
+                return expr
+            if self.tok != ',':
+                raise QAPISchemaError(self, 'Expected "," or "}"')
+            self.accept()
+            if self.tok != "'":
+                raise QAPISchemaError(self, 'Expected string')
 
     def get_values(self):
         expr = []
-        while self.tok != ']':
+        if self.tok == ']':
+            self.accept()
+            return expr
+        if not self.tok in [ '{', '[', "'" ]:
+            raise QAPISchemaError(self, 'Expected "{", "[", "]" or string')
+        while True:
             expr.append(self.get_expr())
-            if self.tok == ',':
+            if self.tok == ']':
                 self.accept()
-        self.accept()
-        return expr
+                return expr
+            if self.tok != ',':
+                raise QAPISchemaError(self, 'Expected "," or "]"')
+            self.accept()
 
     def get_expr(self):
         if self.tok == '{':
@@ -132,9 +150,11 @@ class QAPISchema:
         elif self.tok == '[':
             self.accept()
             expr = self.get_values()
-        else:
+        elif self.tok == "'":
             expr = self.val
             self.accept()
+        else:
+            raise QAPISchemaError(self, 'Expected "{", "[" or string')
         return expr
 
 def parse_schema(fp):
index e69de29bb2d1d6434b8b29ae775ad8c2e48c5391..9f2a35515c4e2e85d88ce1349c91a285568f19e5 100644 (file)
@@ -0,0 +1 @@
+<stdin>:1:10: Expected ":"
index 573541ac9702dd3969c9bc859d2b91ec1f7e6e56..d00491fd7e5bb6fa28c517a0bb32b8b506539d4d 100644 (file)
@@ -1 +1 @@
-0
+1
index e67068c5a00573aa6b983cd68c7c2b1a0be12036..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 100644 (file)
@@ -1,3 +0,0 @@
-[OrderedDict([('enum', None), ('data', ['good', 'bad', 'ugly'])])]
-[None]
-[]
index e69de29bb2d1d6434b8b29ae775ad8c2e48c5391..4fe07001950d81acc59dd54437120720c1382654 100644 (file)
@@ -0,0 +1 @@
+<stdin>:2:20: Expected "," or "]"
index 573541ac9702dd3969c9bc859d2b91ec1f7e6e56..d00491fd7e5bb6fa28c517a0bb32b8b506539d4d 100644 (file)
@@ -1 +1 @@
-0
+1
index e3bd904453988a5926204cc6eaf3faa57213435a..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 100644 (file)
@@ -1,3 +0,0 @@
-[OrderedDict([('enum', 'Status'), ('data', ['good', 'bad', 'ugly'])])]
-['Status']
-[]
index e69de29bb2d1d6434b8b29ae775ad8c2e48c5391..b0121b5f3a145e4cc397a898a8f56a30c9c74fd4 100644 (file)
@@ -0,0 +1 @@
+<stdin>:2:3: Expected "," or "}"
index 573541ac9702dd3969c9bc859d2b91ec1f7e6e56..d00491fd7e5bb6fa28c517a0bb32b8b506539d4d 100644 (file)
@@ -1 +1 @@
-0
+1
index e3bd904453988a5926204cc6eaf3faa57213435a..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 100644 (file)
@@ -1,3 +0,0 @@
-[OrderedDict([('enum', 'Status'), ('data', ['good', 'bad', 'ugly'])])]
-['Status']
-[]
index e69de29bb2d1d6434b8b29ae775ad8c2e48c5391..ff839a34e9576d9ba9f5a0d0830740ef410c16fd 100644 (file)
@@ -0,0 +1 @@
+<stdin>:2:36: Expected "{", "[" or string
index 573541ac9702dd3969c9bc859d2b91ec1f7e6e56..d00491fd7e5bb6fa28c517a0bb32b8b506539d4d 100644 (file)
@@ -1 +1 @@
-0
+1
index e3bd904453988a5926204cc6eaf3faa57213435a..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 100644 (file)
@@ -1,3 +0,0 @@
-[OrderedDict([('enum', 'Status'), ('data', ['good', 'bad', 'ugly'])])]
-['Status']
-[]
index e69de29bb2d1d6434b8b29ae775ad8c2e48c5391..f5409627daf09b9c2446c1ca2e40bbf44faf7be5 100644 (file)
@@ -0,0 +1 @@
+<stdin>:2:38: Expected string
index 573541ac9702dd3969c9bc859d2b91ec1f7e6e56..d00491fd7e5bb6fa28c517a0bb32b8b506539d4d 100644 (file)
@@ -1 +1 @@
-0
+1
index e3bd904453988a5926204cc6eaf3faa57213435a..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 100644 (file)
@@ -1,3 +0,0 @@
-[OrderedDict([('enum', 'Status'), ('data', ['good', 'bad', 'ugly'])])]
-['Status']
-[]
index f9a9c2a0f5e802ecbb40346cd585bff0fa76a451..0e837a7fad64eacd420730dd5462386945a4da65 100644 (file)
@@ -1 +1 @@
-Crashed: <type 'exceptions.IndexError'>
+<stdin>:1:20: Expected "," or "]"
index f9a9c2a0f5e802ecbb40346cd585bff0fa76a451..e6dc9501dc8bfc2e97ab4cd755aec1766a06e646 100644 (file)
@@ -1 +1 @@
-Crashed: <type 'exceptions.IndexError'>
+<stdin>:1:21: Expected "," or "}"