Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 1 | """An implementation of the Zephyr Abstract Syntax Definition Language. |
| 2 | |
| 3 | See http://asdl.sourceforge.net/ and |
Senthil Kumaran | 2b80fa6 | 2010-05-18 13:48:45 +0000 | [diff] [blame] | 4 | http://www.cs.princeton.edu/research/techreps/TR-554-97 |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 5 | |
| 6 | Only supports top level module decl, not view. I'm guessing that view |
| 7 | is intended to support the browser and I'm not interested in the |
| 8 | browser. |
Martin v. Löwis | eae93b7 | 2006-02-28 00:12:47 +0000 | [diff] [blame] | 9 | |
| 10 | Changes for Python: Add support for module versions |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 11 | """ |
| 12 | |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 13 | import os |
Guido van Rossum | 805365e | 2007-05-07 22:24:25 +0000 | [diff] [blame] | 14 | import sys |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 15 | import traceback |
| 16 | |
| 17 | import spark |
| 18 | |
Guido van Rossum | 805365e | 2007-05-07 22:24:25 +0000 | [diff] [blame] | 19 | def output(string): |
| 20 | sys.stdout.write(string + "\n") |
| 21 | |
| 22 | |
Benjamin Peterson | 87c8d87 | 2009-06-11 22:54:11 +0000 | [diff] [blame] | 23 | class Token(object): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 24 | # spark seems to dispatch in the parser based on a token's |
| 25 | # type attribute |
| 26 | def __init__(self, type, lineno): |
| 27 | self.type = type |
| 28 | self.lineno = lineno |
| 29 | |
| 30 | def __str__(self): |
| 31 | return self.type |
| 32 | |
| 33 | def __repr__(self): |
| 34 | return str(self) |
| 35 | |
| 36 | class Id(Token): |
| 37 | def __init__(self, value, lineno): |
| 38 | self.type = 'Id' |
| 39 | self.value = value |
| 40 | self.lineno = lineno |
| 41 | |
| 42 | def __str__(self): |
| 43 | return self.value |
Tim Peters | 710ab3b | 2006-02-28 18:30:36 +0000 | [diff] [blame] | 44 | |
Martin v. Löwis | eae93b7 | 2006-02-28 00:12:47 +0000 | [diff] [blame] | 45 | class String(Token): |
| 46 | def __init__(self, value, lineno): |
| 47 | self.type = 'String' |
| 48 | self.value = value |
| 49 | self.lineno = lineno |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 50 | |
Guido van Rossum | 805365e | 2007-05-07 22:24:25 +0000 | [diff] [blame] | 51 | class ASDLSyntaxError(Exception): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 52 | |
| 53 | def __init__(self, lineno, token=None, msg=None): |
| 54 | self.lineno = lineno |
| 55 | self.token = token |
| 56 | self.msg = msg |
| 57 | |
| 58 | def __str__(self): |
| 59 | if self.msg is None: |
| 60 | return "Error at '%s', line %d" % (self.token, self.lineno) |
| 61 | else: |
| 62 | return "%s, line %d" % (self.msg, self.lineno) |
| 63 | |
| 64 | class ASDLScanner(spark.GenericScanner, object): |
| 65 | |
| 66 | def tokenize(self, input): |
| 67 | self.rv = [] |
| 68 | self.lineno = 1 |
| 69 | super(ASDLScanner, self).tokenize(input) |
| 70 | return self.rv |
| 71 | |
| 72 | def t_id(self, s): |
| 73 | r"[\w\.]+" |
| 74 | # XXX doesn't distinguish upper vs. lower, which is |
| 75 | # significant for ASDL. |
| 76 | self.rv.append(Id(s, self.lineno)) |
Tim Peters | 710ab3b | 2006-02-28 18:30:36 +0000 | [diff] [blame] | 77 | |
Martin v. Löwis | eae93b7 | 2006-02-28 00:12:47 +0000 | [diff] [blame] | 78 | def t_string(self, s): |
| 79 | r'"[^"]*"' |
| 80 | self.rv.append(String(s, self.lineno)) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 81 | |
| 82 | def t_xxx(self, s): # not sure what this production means |
| 83 | r"<=" |
| 84 | self.rv.append(Token(s, self.lineno)) |
| 85 | |
| 86 | def t_punctuation(self, s): |
| 87 | r"[\{\}\*\=\|\(\)\,\?\:]" |
| 88 | self.rv.append(Token(s, self.lineno)) |
| 89 | |
| 90 | def t_comment(self, s): |
| 91 | r"\-\-[^\n]*" |
| 92 | pass |
| 93 | |
| 94 | def t_newline(self, s): |
| 95 | r"\n" |
| 96 | self.lineno += 1 |
| 97 | |
| 98 | def t_whitespace(self, s): |
| 99 | r"[ \t]+" |
| 100 | pass |
| 101 | |
| 102 | def t_default(self, s): |
| 103 | r" . +" |
Collin Winter | 4902e69 | 2007-08-30 18:18:27 +0000 | [diff] [blame] | 104 | raise ValueError("unmatched input: %r" % s) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 105 | |
| 106 | class ASDLParser(spark.GenericParser, object): |
| 107 | def __init__(self): |
| 108 | super(ASDLParser, self).__init__("module") |
| 109 | |
| 110 | def typestring(self, tok): |
| 111 | return tok.type |
| 112 | |
| 113 | def error(self, tok): |
| 114 | raise ASDLSyntaxError(tok.lineno, tok) |
| 115 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 116 | def p_module_0(self, info): |
Benjamin Peterson | 6cb2b92 | 2011-03-12 18:28:16 -0600 | [diff] [blame] | 117 | " module ::= Id Id { } " |
| 118 | module, name, _0, _1 = info |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 119 | if module.value != "module": |
| 120 | raise ASDLSyntaxError(module.lineno, |
| 121 | msg="expected 'module', found %s" % module) |
Benjamin Peterson | 6cb2b92 | 2011-03-12 18:28:16 -0600 | [diff] [blame] | 122 | return Module(name, None) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 123 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 124 | def p_module(self, info): |
Benjamin Peterson | 6cb2b92 | 2011-03-12 18:28:16 -0600 | [diff] [blame] | 125 | " module ::= Id Id { definitions } " |
| 126 | module, name, _0, definitions, _1 = info |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 127 | if module.value != "module": |
| 128 | raise ASDLSyntaxError(module.lineno, |
| 129 | msg="expected 'module', found %s" % module) |
Benjamin Peterson | 6cb2b92 | 2011-03-12 18:28:16 -0600 | [diff] [blame] | 130 | return Module(name, definitions) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 131 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 132 | def p_definition_0(self, definition): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 133 | " definitions ::= definition " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 134 | return definition[0] |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 135 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 136 | def p_definition_1(self, definitions): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 137 | " definitions ::= definition definitions " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 138 | return definitions[0] + definitions[1] |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 139 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 140 | def p_definition(self, info): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 141 | " definition ::= Id = type " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 142 | id, _, type = info |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 143 | return [Type(id, type)] |
| 144 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 145 | def p_type_0(self, product): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 146 | " type ::= product " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 147 | return product[0] |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 148 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 149 | def p_type_1(self, sum): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 150 | " type ::= sum " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 151 | return Sum(sum[0]) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 152 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 153 | def p_type_2(self, info): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 154 | " type ::= sum Id ( fields ) " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 155 | sum, id, _0, attributes, _1 = info |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 156 | if id.value != "attributes": |
| 157 | raise ASDLSyntaxError(id.lineno, |
| 158 | msg="expected attributes, found %s" % id) |
Martin v. Löwis | 49c5da1 | 2006-03-01 22:49:05 +0000 | [diff] [blame] | 159 | if attributes: |
| 160 | attributes.reverse() |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 161 | return Sum(sum, attributes) |
| 162 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 163 | def p_product(self, info): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 164 | " product ::= ( fields ) " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 165 | _0, fields, _1 = info |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 166 | # XXX can't I just construct things in the right order? |
Tim Peters | 536cf99 | 2005-12-25 23:18:31 +0000 | [diff] [blame] | 167 | fields.reverse() |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 168 | return Product(fields) |
| 169 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 170 | def p_sum_0(self, constructor): |
Georg Brandl | 1f01deb | 2009-01-03 22:47:39 +0000 | [diff] [blame] | 171 | " sum ::= constructor " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 172 | return [constructor[0]] |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 173 | |
Neal Norwitz | bda355f | 2008-03-17 18:03:56 +0000 | [diff] [blame] | 174 | def p_sum_1(self, info): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 175 | " sum ::= constructor | sum " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 176 | constructor, _, sum = info |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 177 | return [constructor] + sum |
| 178 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 179 | def p_sum_2(self, info): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 180 | " sum ::= constructor | sum " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 181 | constructor, _, sum = info |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 182 | return [constructor] + sum |
| 183 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 184 | def p_constructor_0(self, id): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 185 | " constructor ::= Id " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 186 | return Constructor(id[0]) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 187 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 188 | def p_constructor_1(self, info): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 189 | " constructor ::= Id ( fields ) " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 190 | id, _0, fields, _1 = info |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 191 | # XXX can't I just construct things in the right order? |
Tim Peters | 536cf99 | 2005-12-25 23:18:31 +0000 | [diff] [blame] | 192 | fields.reverse() |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 193 | return Constructor(id, fields) |
| 194 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 195 | def p_fields_0(self, field): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 196 | " fields ::= field " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 197 | return [field[0]] |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 198 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 199 | def p_fields_1(self, info): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 200 | " fields ::= field , fields " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 201 | field, _, fields = info |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 202 | return fields + [field] |
| 203 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 204 | def p_field_0(self, type_): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 205 | " field ::= Id " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 206 | return Field(type_[0]) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 207 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 208 | def p_field_1(self, info): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 209 | " field ::= Id Id " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 210 | type, name = info |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 211 | return Field(type, name) |
| 212 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 213 | def p_field_2(self, info): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 214 | " field ::= Id * Id " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 215 | type, _, name = info |
Benjamin Peterson | 87c8d87 | 2009-06-11 22:54:11 +0000 | [diff] [blame] | 216 | return Field(type, name, seq=True) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 217 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 218 | def p_field_3(self, info): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 219 | " field ::= Id ? Id " |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 220 | type, _, name = info |
Benjamin Peterson | 87c8d87 | 2009-06-11 22:54:11 +0000 | [diff] [blame] | 221 | return Field(type, name, opt=True) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 222 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 223 | def p_field_4(self, type_): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 224 | " field ::= Id * " |
Benjamin Peterson | 87c8d87 | 2009-06-11 22:54:11 +0000 | [diff] [blame] | 225 | return Field(type_[0], seq=True) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 226 | |
Guido van Rossum | 1bc535d | 2007-05-15 18:46:22 +0000 | [diff] [blame] | 227 | def p_field_5(self, type_): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 228 | " field ::= Id ? " |
Benjamin Peterson | 87c8d87 | 2009-06-11 22:54:11 +0000 | [diff] [blame] | 229 | return Field(type[0], opt=True) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 230 | |
| 231 | builtin_types = ("identifier", "string", "int", "bool", "object") |
| 232 | |
| 233 | # below is a collection of classes to capture the AST of an AST :-) |
| 234 | # not sure if any of the methods are useful yet, but I'm adding them |
| 235 | # piecemeal as they seem helpful |
| 236 | |
Benjamin Peterson | 87c8d87 | 2009-06-11 22:54:11 +0000 | [diff] [blame] | 237 | class AST(object): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 238 | pass # a marker class |
| 239 | |
| 240 | class Module(AST): |
Benjamin Peterson | 6cb2b92 | 2011-03-12 18:28:16 -0600 | [diff] [blame] | 241 | def __init__(self, name, dfns): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 242 | self.name = name |
| 243 | self.dfns = dfns |
| 244 | self.types = {} # maps type name to value (from dfns) |
| 245 | for type in dfns: |
| 246 | self.types[type.name.value] = type.value |
| 247 | |
| 248 | def __repr__(self): |
| 249 | return "Module(%s, %s)" % (self.name, self.dfns) |
| 250 | |
| 251 | class Type(AST): |
| 252 | def __init__(self, name, value): |
| 253 | self.name = name |
| 254 | self.value = value |
| 255 | |
| 256 | def __repr__(self): |
| 257 | return "Type(%s, %s)" % (self.name, self.value) |
| 258 | |
| 259 | class Constructor(AST): |
| 260 | def __init__(self, name, fields=None): |
| 261 | self.name = name |
| 262 | self.fields = fields or [] |
| 263 | |
| 264 | def __repr__(self): |
| 265 | return "Constructor(%s, %s)" % (self.name, self.fields) |
| 266 | |
| 267 | class Field(AST): |
Benjamin Peterson | 87c8d87 | 2009-06-11 22:54:11 +0000 | [diff] [blame] | 268 | def __init__(self, type, name=None, seq=False, opt=False): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 269 | self.type = type |
| 270 | self.name = name |
| 271 | self.seq = seq |
| 272 | self.opt = opt |
| 273 | |
| 274 | def __repr__(self): |
| 275 | if self.seq: |
Benjamin Peterson | 87c8d87 | 2009-06-11 22:54:11 +0000 | [diff] [blame] | 276 | extra = ", seq=True" |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 277 | elif self.opt: |
Benjamin Peterson | 87c8d87 | 2009-06-11 22:54:11 +0000 | [diff] [blame] | 278 | extra = ", opt=True" |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 279 | else: |
| 280 | extra = "" |
| 281 | if self.name is None: |
| 282 | return "Field(%s%s)" % (self.type, extra) |
| 283 | else: |
| 284 | return "Field(%s, %s%s)" % (self.type, self.name, extra) |
| 285 | |
| 286 | class Sum(AST): |
| 287 | def __init__(self, types, attributes=None): |
| 288 | self.types = types |
| 289 | self.attributes = attributes or [] |
| 290 | |
| 291 | def __repr__(self): |
| 292 | if self.attributes is None: |
| 293 | return "Sum(%s)" % self.types |
| 294 | else: |
| 295 | return "Sum(%s, %s)" % (self.types, self.attributes) |
| 296 | |
| 297 | class Product(AST): |
| 298 | def __init__(self, fields): |
| 299 | self.fields = fields |
| 300 | |
| 301 | def __repr__(self): |
| 302 | return "Product(%s)" % self.fields |
| 303 | |
| 304 | class VisitorBase(object): |
| 305 | |
Benjamin Peterson | 87c8d87 | 2009-06-11 22:54:11 +0000 | [diff] [blame] | 306 | def __init__(self, skip=False): |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 307 | self.cache = {} |
| 308 | self.skip = skip |
| 309 | |
| 310 | def visit(self, object, *args): |
| 311 | meth = self._dispatch(object) |
| 312 | if meth is None: |
| 313 | return |
| 314 | try: |
| 315 | meth(object, *args) |
Guido van Rossum | 805365e | 2007-05-07 22:24:25 +0000 | [diff] [blame] | 316 | except Exception: |
Alexandre Vassalotti | df6f3fd | 2009-07-17 05:35:59 +0000 | [diff] [blame] | 317 | output("Error visiting" + repr(object)) |
| 318 | output(str(sys.exc_info()[1])) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 319 | traceback.print_exc() |
| 320 | # XXX hack |
| 321 | if hasattr(self, 'file'): |
| 322 | self.file.flush() |
| 323 | os._exit(1) |
| 324 | |
| 325 | def _dispatch(self, object): |
| 326 | assert isinstance(object, AST), repr(object) |
| 327 | klass = object.__class__ |
| 328 | meth = self.cache.get(klass) |
| 329 | if meth is None: |
| 330 | methname = "visit" + klass.__name__ |
| 331 | if self.skip: |
| 332 | meth = getattr(self, methname, None) |
| 333 | else: |
| 334 | meth = getattr(self, methname) |
| 335 | self.cache[klass] = meth |
| 336 | return meth |
| 337 | |
| 338 | class Check(VisitorBase): |
| 339 | |
| 340 | def __init__(self): |
Benjamin Peterson | 87c8d87 | 2009-06-11 22:54:11 +0000 | [diff] [blame] | 341 | super(Check, self).__init__(skip=True) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 342 | self.cons = {} |
| 343 | self.errors = 0 |
| 344 | self.types = {} |
| 345 | |
| 346 | def visitModule(self, mod): |
| 347 | for dfn in mod.dfns: |
| 348 | self.visit(dfn) |
| 349 | |
| 350 | def visitType(self, type): |
| 351 | self.visit(type.value, str(type.name)) |
| 352 | |
| 353 | def visitSum(self, sum, name): |
| 354 | for t in sum.types: |
| 355 | self.visit(t, name) |
| 356 | |
| 357 | def visitConstructor(self, cons, name): |
| 358 | key = str(cons.name) |
| 359 | conflict = self.cons.get(key) |
| 360 | if conflict is None: |
| 361 | self.cons[key] = name |
| 362 | else: |
Guido van Rossum | 805365e | 2007-05-07 22:24:25 +0000 | [diff] [blame] | 363 | output("Redefinition of constructor %s" % key) |
| 364 | output("Defined in %s and %s" % (conflict, name)) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 365 | self.errors += 1 |
| 366 | for f in cons.fields: |
| 367 | self.visit(f, key) |
| 368 | |
| 369 | def visitField(self, field, name): |
| 370 | key = str(field.type) |
| 371 | l = self.types.setdefault(key, []) |
| 372 | l.append(name) |
| 373 | |
| 374 | def visitProduct(self, prod, name): |
| 375 | for f in prod.fields: |
| 376 | self.visit(f, name) |
| 377 | |
| 378 | def check(mod): |
| 379 | v = Check() |
| 380 | v.visit(mod) |
| 381 | |
| 382 | for t in v.types: |
Fred Drake | 34e4f52 | 2006-12-29 04:42:48 +0000 | [diff] [blame] | 383 | if t not in mod.types and not t in builtin_types: |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 384 | v.errors += 1 |
| 385 | uses = ", ".join(v.types[t]) |
Guido van Rossum | 805365e | 2007-05-07 22:24:25 +0000 | [diff] [blame] | 386 | output("Undefined type %s, used in %s" % (t, uses)) |
Tim Peters | 536cf99 | 2005-12-25 23:18:31 +0000 | [diff] [blame] | 387 | |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 388 | return not v.errors |
| 389 | |
| 390 | def parse(file): |
| 391 | scanner = ASDLScanner() |
| 392 | parser = ASDLParser() |
| 393 | |
| 394 | buf = open(file).read() |
| 395 | tokens = scanner.tokenize(buf) |
| 396 | try: |
| 397 | return parser.parse(tokens) |
Guido van Rossum | 805365e | 2007-05-07 22:24:25 +0000 | [diff] [blame] | 398 | except ASDLSyntaxError: |
Georg Brandl | 8a73278 | 2009-04-12 11:34:13 +0000 | [diff] [blame] | 399 | err = sys.exc_info()[1] |
| 400 | output(str(err)) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 401 | lines = buf.split("\n") |
Guido van Rossum | 805365e | 2007-05-07 22:24:25 +0000 | [diff] [blame] | 402 | output(lines[err.lineno - 1]) # lines starts at 0, files at 1 |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 403 | |
| 404 | if __name__ == "__main__": |
| 405 | import glob |
| 406 | import sys |
| 407 | |
| 408 | if len(sys.argv) > 1: |
| 409 | files = sys.argv[1:] |
| 410 | else: |
| 411 | testdir = "tests" |
| 412 | files = glob.glob(testdir + "/*.asdl") |
Tim Peters | 536cf99 | 2005-12-25 23:18:31 +0000 | [diff] [blame] | 413 | |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 414 | for file in files: |
Guido van Rossum | 805365e | 2007-05-07 22:24:25 +0000 | [diff] [blame] | 415 | output(file) |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 416 | mod = parse(file) |
Georg Brandl | 8a73278 | 2009-04-12 11:34:13 +0000 | [diff] [blame] | 417 | if not mod: |
| 418 | break |
Guido van Rossum | 805365e | 2007-05-07 22:24:25 +0000 | [diff] [blame] | 419 | output("module", mod.name) |
| 420 | output(len(mod.dfns), "definitions") |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 421 | if not check(mod): |
Guido van Rossum | 805365e | 2007-05-07 22:24:25 +0000 | [diff] [blame] | 422 | output("Check failed") |
Jeremy Hylton | 3e0055f | 2005-10-20 19:59:25 +0000 | [diff] [blame] | 423 | else: |
| 424 | for dfn in mod.dfns: |
Guido van Rossum | 805365e | 2007-05-07 22:24:25 +0000 | [diff] [blame] | 425 | output(dfn.type) |