| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | #!/usr/bin/env python | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-07 23:56:14 +08:00
										 |  |  | ##  This Source Code Form is subject to the terms of the Mozilla Public | 
					
						
							|  |  |  | ##  License, v. 2.0. If a copy of the MPL was not distributed with this | 
					
						
							|  |  |  | ##  file, You can obtain one at https://mozilla.org/MPL/2.0/. | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | ## | 
					
						
							| 
									
										
										
										
											2020-03-10 22:26:49 +08:00
										 |  |  | ##  Copyright (c) 2007-2020 VMware, Inc. or its affiliates.  All rights reserved. | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | ## | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from __future__ import nested_scopes | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  | from __future__ import print_function | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from amqp_codegen import * | 
					
						
							|  |  |  | import string | 
					
						
							|  |  |  | import re | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Coming up with a proper encoding of AMQP tables in JSON is too much | 
					
						
							|  |  |  | # hassle at this stage. Given that the only default value we are | 
					
						
							|  |  |  | # interested in is for the empty table, we only support that. | 
					
						
							|  |  |  | def convertTable(d): | 
					
						
							|  |  |  |     if len(d) == 0: | 
					
						
							|  |  |  |         return "[]" | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         raise Exception('Non-empty table defaults not supported ' + d) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | erlangDefaultValueTypeConvMap = { | 
					
						
							|  |  |  |     bool : lambda x: str(x).lower(), | 
					
						
							|  |  |  |     int : lambda x: str(x), | 
					
						
							|  |  |  |     float : lambda x: str(x), | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     dict: convertTable | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  | try: | 
					
						
							|  |  |  |     _checkIfPython2 = unicode | 
					
						
							|  |  |  |     erlangDefaultValueTypeConvMap[str] = lambda x: "<<\"" + x + "\">>" | 
					
						
							|  |  |  |     erlangDefaultValueTypeConvMap[unicode] = lambda x: "<<\"" + x.encode("utf-8") + "\">>" | 
					
						
							|  |  |  | except NameError: | 
					
						
							|  |  |  |     erlangDefaultValueTypeConvMap[bytes] = lambda x: "<<\"" + x + "\">>" | 
					
						
							|  |  |  |     erlangDefaultValueTypeConvMap[str] = lambda x: "<<\"" + x + "\">>" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | def erlangize(s): | 
					
						
							|  |  |  |     s = s.replace('-', '_') | 
					
						
							|  |  |  |     s = s.replace(' ', '_') | 
					
						
							|  |  |  |     return s | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AmqpMethod.erlangName = lambda m: "'" + erlangize(m.klass.name) + '.' + erlangize(m.name) + "'" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AmqpClass.erlangName = lambda c: "'" + erlangize(c.name) + "'" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def erlangConstantName(s): | 
					
						
							|  |  |  |     return '_'.join(re.split('[- ]', s.upper())) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class PackedMethodBitField: | 
					
						
							|  |  |  |     def __init__(self, index): | 
					
						
							|  |  |  |         self.index = index | 
					
						
							|  |  |  |         self.domain = 'bit' | 
					
						
							|  |  |  |         self.contents = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def extend(self, f): | 
					
						
							|  |  |  |         self.contents.append(f) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def count(self): | 
					
						
							|  |  |  |         return len(self.contents) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def full(self): | 
					
						
							|  |  |  |         return self.count() == 8 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def multiLineFormat(things, prologue, separator, lineSeparator, epilogue, thingsPerLine = 4): | 
					
						
							|  |  |  |     r = [prologue] | 
					
						
							|  |  |  |     i = 0 | 
					
						
							|  |  |  |     for t in things: | 
					
						
							|  |  |  |         if i != 0: | 
					
						
							|  |  |  |             if i % thingsPerLine == 0: | 
					
						
							|  |  |  |                 r += [lineSeparator] | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 r += [separator] | 
					
						
							|  |  |  |         r += [t] | 
					
						
							|  |  |  |         i += 1 | 
					
						
							|  |  |  |     r += [epilogue] | 
					
						
							|  |  |  |     return "".join(r) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def prettyType(typeName, subTypes, typesPerLine = 4): | 
					
						
							|  |  |  |     """Pretty print a type signature made up of many alternative subtypes""" | 
					
						
							|  |  |  |     sTs = multiLineFormat(subTypes, | 
					
						
							|  |  |  |                           "( ", " | ", "\n       | ", " )", | 
					
						
							|  |  |  |                           thingsPerLine = typesPerLine) | 
					
						
							| 
									
										
										
										
											2016-06-27 21:09:17 +08:00
										 |  |  |     return "-type %s ::\n       %s." % (typeName, sTs) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | def printFileHeader(): | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("""%%   Autogenerated code. Do not edit.
 | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | %% | 
					
						
							| 
									
										
										
										
											2020-07-07 23:56:14 +08:00
										 |  |  | %%  This Source Code Form is subject to the terms of the Mozilla Public | 
					
						
							|  |  |  | %%  License, v. 2.0. If a copy of the MPL was not distributed with this | 
					
						
							|  |  |  | %%  file, You can obtain one at https://mozilla.org/MPL/2.0/. | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | %% | 
					
						
							| 
									
										
										
										
											2020-03-10 22:26:49 +08:00
										 |  |  | %%  Copyright (c) 2007-2020 VMware, Inc. or its affiliates.  All rights reserved. | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  | %%""")
 | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | def genErl(spec): | 
					
						
							|  |  |  |     def erlType(domain): | 
					
						
							|  |  |  |         return erlangize(spec.resolveDomain(domain)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def fieldTypeList(fields): | 
					
						
							|  |  |  |         return '[' + ', '.join([erlType(f.domain) for f in fields]) + ']' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def fieldNameList(fields): | 
					
						
							|  |  |  |         return '[' + ', '.join([erlangize(f.name) for f in fields]) + ']' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def fieldTempList(fields): | 
					
						
							|  |  |  |         return '[' + ', '.join(['F' + str(f.index) for f in fields]) + ']' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def fieldMapList(fields): | 
					
						
							|  |  |  |         return ', '.join([erlangize(f.name) + " = F" + str(f.index) for f in fields]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def genLookupMethodName(m): | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("lookup_method_name({%d, %d}) -> %s;" % (m.klass.index, m.index, m.erlangName())) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def genLookupClassName(c): | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("lookup_class_name(%d) -> %s;" % (c.index, c.erlangName())) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def genMethodId(m): | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("method_id(%s) -> {%d, %d};" % (m.erlangName(), m.klass.index, m.index)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def genMethodHasContent(m): | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("method_has_content(%s) -> %s;" % (m.erlangName(), str(m.hasContent).lower())) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def genMethodIsSynchronous(m): | 
					
						
							|  |  |  |         hasNoWait = "nowait" in fieldNameList(m.arguments) | 
					
						
							|  |  |  |         if m.isSynchronous and hasNoWait: | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |           print("is_method_synchronous(#%s{nowait = NoWait}) -> not(NoWait);" % (m.erlangName())) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |           print("is_method_synchronous(#%s{}) -> %s;" % (m.erlangName(), str(m.isSynchronous).lower())) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def genMethodFieldTypes(m): | 
					
						
							|  |  |  |         """Not currently used - may be useful in future?""" | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("method_fieldtypes(%s) -> %s;" % (m.erlangName(), fieldTypeList(m.arguments))) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def genMethodFieldNames(m): | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("method_fieldnames(%s) -> %s;" % (m.erlangName(), fieldNameList(m.arguments))) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def packMethodFields(fields): | 
					
						
							|  |  |  |         packed = [] | 
					
						
							|  |  |  |         bitfield = None | 
					
						
							|  |  |  |         for f in fields: | 
					
						
							|  |  |  |             if erlType(f.domain) == 'bit': | 
					
						
							|  |  |  |                 if not(bitfield) or bitfield.full(): | 
					
						
							|  |  |  |                     bitfield = PackedMethodBitField(f.index) | 
					
						
							|  |  |  |                     packed.append(bitfield) | 
					
						
							|  |  |  |                 bitfield.extend(f) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 bitfield = None | 
					
						
							|  |  |  |                 packed.append(f) | 
					
						
							|  |  |  |         return packed | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def methodFieldFragment(f): | 
					
						
							|  |  |  |         type = erlType(f.domain) | 
					
						
							|  |  |  |         p = 'F' + str(f.index) | 
					
						
							|  |  |  |         if type == 'shortstr': | 
					
						
							|  |  |  |             return p+'Len:8/unsigned, '+p+':'+p+'Len/binary' | 
					
						
							|  |  |  |         elif type == 'longstr': | 
					
						
							|  |  |  |             return p+'Len:32/unsigned, '+p+':'+p+'Len/binary' | 
					
						
							|  |  |  |         elif type == 'octet': | 
					
						
							|  |  |  |             return p+':8/unsigned' | 
					
						
							|  |  |  |         elif type == 'short': | 
					
						
							|  |  |  |             return p+':16/unsigned' | 
					
						
							|  |  |  |         elif type == 'long': | 
					
						
							|  |  |  |             return p+':32/unsigned' | 
					
						
							|  |  |  |         elif type == 'longlong': | 
					
						
							|  |  |  |             return p+':64/unsigned' | 
					
						
							|  |  |  |         elif type == 'timestamp': | 
					
						
							|  |  |  |             return p+':64/unsigned' | 
					
						
							|  |  |  |         elif type == 'bit': | 
					
						
							|  |  |  |             return p+'Bits:8' | 
					
						
							|  |  |  |         elif type == 'table': | 
					
						
							|  |  |  |             return p+'Len:32/unsigned, '+p+'Tab:'+p+'Len/binary' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def genFieldPostprocessing(packed, hasContent): | 
					
						
							|  |  |  |         for f in packed: | 
					
						
							|  |  |  |             type = erlType(f.domain) | 
					
						
							|  |  |  |             if type == 'bit': | 
					
						
							|  |  |  |                 for index in range(f.count()): | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                     print("  F%d = ((F%dBits band %d) /= 0)," % \ | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |                           (f.index + index, | 
					
						
							|  |  |  |                            f.index, | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                            1 << index)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |             elif type == 'table': | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                 print("  F%d = rabbit_binary_parser:parse_table(F%dTab)," % \ | 
					
						
							|  |  |  |                       (f.index, f.index)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |             # We skip the check on content-bearing methods for | 
					
						
							|  |  |  |             # speed. This is a sanity check, not a security thing. | 
					
						
							|  |  |  |             elif type == 'shortstr' and not hasContent: | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                 print("  rabbit_binary_parser:assert_utf8(F%d)," % (f.index)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def genMethodRecord(m): | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("method_record(%s) -> #%s{};" % (m.erlangName(), m.erlangName())) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def genDecodeMethodFields(m): | 
					
						
							|  |  |  |         packedFields = packMethodFields(m.arguments) | 
					
						
							|  |  |  |         binaryPattern = ', '.join([methodFieldFragment(f) for f in packedFields]) | 
					
						
							|  |  |  |         if binaryPattern: | 
					
						
							|  |  |  |             restSeparator = ', ' | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             restSeparator = '' | 
					
						
							|  |  |  |         recordConstructorExpr = '#%s{%s}' % (m.erlangName(), fieldMapList(m.arguments)) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("decode_method_fields(%s, <<%s>>) ->" % (m.erlangName(), binaryPattern)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |         genFieldPostprocessing(packedFields, m.hasContent) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("  %s;" % (recordConstructorExpr,)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def genDecodeProperties(c): | 
					
						
							|  |  |  |         def presentBin(fields): | 
					
						
							|  |  |  |             ps = ', '.join(['P' + str(f.index) + ':1' for f in fields]) | 
					
						
							|  |  |  |             return '<<' + ps + ', _:%d, R0/binary>>' % (16 - len(fields),) | 
					
						
							|  |  |  |         def writePropFieldLine(field): | 
					
						
							|  |  |  |             i = str(field.index) | 
					
						
							|  |  |  |             if field.domain == 'bit': | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                 print("  {F%s, R%s} = {P%s =/= 0, R%s}," % \ | 
					
						
							|  |  |  |                     (i, str(field.index + 1), i, i)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                 print("  {F%s, R%s} = if P%s =:= 0 -> {undefined, R%s}; true -> ?%s_VAL(R%s, L%s, V%s, X%s) end," % \ | 
					
						
							|  |  |  |                     (i, str(field.index + 1), i, i, erlType(field.domain).upper(), i, i, i, i)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if len(c.fields) == 0: | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |             print("decode_properties(%d, <<>>) ->" % (c.index,)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |             print(("decode_properties(%d, %s) ->" % | 
					
						
							|  |  |  |                    (c.index, presentBin(c.fields)))) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |             for field in c.fields: | 
					
						
							|  |  |  |                 writePropFieldLine(field) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |             print("  <<>> = %s," % ('R' + str(len(c.fields)))) | 
					
						
							|  |  |  |         print("  #'P_%s'{%s};" % (erlangize(c.name), fieldMapList(c.fields))) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def genFieldPreprocessing(packed): | 
					
						
							|  |  |  |         for f in packed: | 
					
						
							|  |  |  |             type = erlType(f.domain) | 
					
						
							|  |  |  |             if type == 'bit': | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                 print("  F%dBits = (%s)," % \ | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |                       (f.index, | 
					
						
							|  |  |  |                        ' bor '.join(['(bitvalue(F%d) bsl %d)' % (x.index, x.index - f.index) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                                      for x in f.contents]))) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |             elif type == 'table': | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                 print("  F%dTab = rabbit_binary_generator:generate_table(F%d)," % (f.index, f.index)) | 
					
						
							|  |  |  |                 print("  F%dLen = size(F%dTab)," % (f.index, f.index)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |             elif type == 'shortstr': | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                 print("  F%dLen = shortstr_size(F%d)," % (f.index, f.index)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |             elif type == 'longstr': | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                 print("  F%dLen = size(F%d)," % (f.index, f.index)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def genEncodeMethodFields(m): | 
					
						
							|  |  |  |         packedFields = packMethodFields(m.arguments) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("encode_method_fields(#%s{%s}) ->" % (m.erlangName(), fieldMapList(m.arguments))) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |         genFieldPreprocessing(packedFields) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("  <<%s>>;" % (', '.join([methodFieldFragment(f) for f in packedFields]))) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def genEncodeProperties(c): | 
					
						
							|  |  |  |         def presentBin(fields): | 
					
						
							|  |  |  |             ps = ', '.join(['P' + str(f.index) + ':1' for f in fields]) | 
					
						
							|  |  |  |             return '<<' + ps + ', 0:%d>>' % (16 - len(fields),) | 
					
						
							|  |  |  |         def writePropFieldLine(field): | 
					
						
							|  |  |  |             i = str(field.index) | 
					
						
							|  |  |  |             if field.domain == 'bit': | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                 print("  {P%s, R%s} = {F%s =:= 1, R%s}," % \ | 
					
						
							|  |  |  |                     (i, str(field.index + 1), i, i)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                 print("  {P%s, R%s} = if F%s =:= undefined -> {0, R%s}; true -> {1, [?%s_PROP(F%s, L%s) | R%s]} end," % \ | 
					
						
							|  |  |  |                     (i, str(field.index + 1), i, i, erlType(field.domain).upper(), i, i, i)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("encode_properties(#'P_%s'{%s}) ->" % (erlangize(c.name), fieldMapList(c.fields))) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |         if len(c.fields) == 0: | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |             print("  <<>>;") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |             print("  R0 = [<<>>],") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |             for field in c.fields: | 
					
						
							|  |  |  |                 writePropFieldLine(field) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |             print("  list_to_binary([%s | lists:reverse(R%s)]);" % \ | 
					
						
							|  |  |  |                 (presentBin(c.fields), str(len(c.fields)))) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def messageConstantClass(cls): | 
					
						
							|  |  |  |         # We do this because 0.8 uses "soft error" and 8.1 uses "soft-error". | 
					
						
							|  |  |  |         return erlangConstantName(cls) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def genLookupException(c,v,cls): | 
					
						
							|  |  |  |         mCls = messageConstantClass(cls) | 
					
						
							|  |  |  |         if mCls == 'SOFT_ERROR': genLookupException1(c,'false') | 
					
						
							|  |  |  |         elif mCls == 'HARD_ERROR': genLookupException1(c, 'true') | 
					
						
							|  |  |  |         elif mCls == '': pass | 
					
						
							|  |  |  |         else: raise Exception('Unknown constant class' + cls) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def genLookupException1(c,hardErrorBoolStr): | 
					
						
							|  |  |  |         n = erlangConstantName(c) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print('lookup_amqp_exception(%s) -> {%s, ?%s, <<"%s">>};' % \ | 
					
						
							|  |  |  |               (n.lower(), hardErrorBoolStr, n, n)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def genAmqpException(c,v,cls): | 
					
						
							|  |  |  |         n = erlangConstantName(c) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print('amqp_exception(?%s) -> %s;' % \ | 
					
						
							|  |  |  |             (n, n.lower())) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     methods = spec.allMethods() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     printFileHeader() | 
					
						
							|  |  |  |     module = "rabbit_framing_amqp_%d_%d" % (spec.major, spec.minor) | 
					
						
							|  |  |  |     if spec.revision != 0: | 
					
						
							|  |  |  |         module = "%s_%d" % (module, spec.revision) | 
					
						
							|  |  |  |     if module == "rabbit_framing_amqp_8_0": | 
					
						
							|  |  |  |         module = "rabbit_framing_amqp_0_8" | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("-module(%s)." % module) | 
					
						
							|  |  |  |     print("""-include("rabbit_framing.hrl").
 | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | -export([version/0]). | 
					
						
							|  |  |  | -export([lookup_method_name/1]). | 
					
						
							|  |  |  | -export([lookup_class_name/1]). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -export([method_id/1]). | 
					
						
							|  |  |  | -export([method_has_content/1]). | 
					
						
							|  |  |  | -export([is_method_synchronous/1]). | 
					
						
							|  |  |  | -export([method_record/1]). | 
					
						
							|  |  |  | -export([method_fieldnames/1]). | 
					
						
							|  |  |  | -export([decode_method_fields/2]). | 
					
						
							|  |  |  | -export([decode_properties/2]). | 
					
						
							|  |  |  | -export([encode_method_fields/1]). | 
					
						
							|  |  |  | -export([encode_properties/1]). | 
					
						
							|  |  |  | -export([lookup_amqp_exception/1]). | 
					
						
							|  |  |  | -export([amqp_exception/1]). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  | """)
 | 
					
						
							|  |  |  |     print("%% Various types") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("""-export_type([amqp_field_type/0, amqp_property_type/0,
 | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |               amqp_table/0, amqp_array/0, amqp_value/0, | 
					
						
							|  |  |  |               amqp_method_name/0, amqp_method/0, amqp_method_record/0, | 
					
						
							|  |  |  |               amqp_method_field_name/0, amqp_property_record/0, | 
					
						
							|  |  |  |               amqp_exception/0, amqp_exception_code/0, amqp_class_id/0]). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-27 21:09:17 +08:00
										 |  |  | -type amqp_field_type() :: | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |       'longstr' | 'signedint' | 'decimal' | 'timestamp' | | 
					
						
							| 
									
										
										
										
											2015-12-04 17:57:59 +08:00
										 |  |  |       'unsignedbyte' | 'unsignedshort' | 'unsignedint' | | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |       'table' | 'byte' | 'double' | 'float' | 'long' | | 
					
						
							| 
									
										
										
										
											2016-06-27 21:09:17 +08:00
										 |  |  |       'short' | 'bool' | 'binary' | 'void' | 'array'. | 
					
						
							|  |  |  | -type amqp_property_type() :: | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |       'shortstr' | 'longstr' | 'octet' | 'short' | 'long' | | 
					
						
							| 
									
										
										
										
											2016-06-27 21:09:17 +08:00
										 |  |  |       'longlong' | 'timestamp' | 'bit' | 'table'. | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-27 21:09:17 +08:00
										 |  |  | -type amqp_table() :: [{binary(), amqp_field_type(), amqp_value()}]. | 
					
						
							|  |  |  | -type amqp_array() :: [{amqp_field_type(), amqp_value()}]. | 
					
						
							|  |  |  | -type amqp_value() :: binary() |    % longstr | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |                       integer() |   % signedint | 
					
						
							|  |  |  |                       {non_neg_integer(), non_neg_integer()} | % decimal | 
					
						
							|  |  |  |                       amqp_table() | | 
					
						
							|  |  |  |                       amqp_array() | | 
					
						
							|  |  |  |                       byte() |      % byte | 
					
						
							|  |  |  |                       float() |     % double | 
					
						
							|  |  |  |                       integer() |   % long | 
					
						
							|  |  |  |                       integer() |   % short | 
					
						
							|  |  |  |                       boolean() |   % bool | 
					
						
							|  |  |  |                       binary() |    % binary | 
					
						
							|  |  |  |                       'undefined' | % void | 
					
						
							| 
									
										
										
										
											2016-06-27 21:09:17 +08:00
										 |  |  |                       non_neg_integer(). % timestamp | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  | """)
 | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print(prettyType("amqp_method_name()", | 
					
						
							|  |  |  |                      [m.erlangName() for m in methods])) | 
					
						
							|  |  |  |     print(prettyType("amqp_method()", | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |                      ["{%s, %s}" % (m.klass.index, m.index) for m in methods], | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |                      6)) | 
					
						
							|  |  |  |     print(prettyType("amqp_method_record()", | 
					
						
							|  |  |  |                      ["#%s{}" % (m.erlangName()) for m in methods])) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |     fieldNames = set() | 
					
						
							|  |  |  |     for m in methods: | 
					
						
							| 
									
										
										
										
											2015-12-18 22:55:22 +08:00
										 |  |  |         fieldNames.update([erlangize(f.name) for f in m.arguments]) | 
					
						
							|  |  |  |     fieldNames = [f for f in fieldNames] | 
					
						
							|  |  |  |     fieldNames.sort() | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print(prettyType("amqp_method_field_name()", | 
					
						
							|  |  |  |                      fieldNames)) | 
					
						
							|  |  |  |     print(prettyType("amqp_property_record()", | 
					
						
							|  |  |  |                      ["#'P_%s'{}" % erlangize(c.name) for c in spec.allClasses()])) | 
					
						
							|  |  |  |     print(prettyType("amqp_exception()", | 
					
						
							|  |  |  |                      ["'%s'" % erlangConstantName(c).lower() for (c, v, cls) in spec.constants])) | 
					
						
							|  |  |  |     print(prettyType("amqp_exception_code()", | 
					
						
							|  |  |  |                      ["%i" % v for (c, v, cls) in spec.constants])) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |     classIds = set() | 
					
						
							|  |  |  |     for m in spec.allMethods(): | 
					
						
							|  |  |  |         classIds.add(m.klass.index) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print(prettyType("amqp_class_id()", | 
					
						
							|  |  |  |                      ["%i" % ci for ci in classIds])) | 
					
						
							|  |  |  |     print(prettyType("amqp_class_name()", | 
					
						
							|  |  |  |                      ["%s" % c.erlangName() for c in spec.allClasses()])) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("""
 | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | %% Method signatures | 
					
						
							| 
									
										
										
										
											2016-06-27 21:09:17 +08:00
										 |  |  | -spec version() -> {non_neg_integer(), non_neg_integer(), non_neg_integer()}. | 
					
						
							|  |  |  | -spec lookup_method_name(amqp_method()) -> amqp_method_name(). | 
					
						
							|  |  |  | -spec lookup_class_name(amqp_class_id()) -> amqp_class_name(). | 
					
						
							|  |  |  | -spec method_id(amqp_method_name()) -> amqp_method(). | 
					
						
							|  |  |  | -spec method_has_content(amqp_method_name()) -> boolean(). | 
					
						
							|  |  |  | -spec is_method_synchronous(amqp_method_record()) -> boolean(). | 
					
						
							|  |  |  | -spec method_record(amqp_method_name()) -> amqp_method_record(). | 
					
						
							|  |  |  | -spec method_fieldnames(amqp_method_name()) -> [amqp_method_field_name()]. | 
					
						
							|  |  |  | -spec decode_method_fields(amqp_method_name(), binary()) -> | 
					
						
							|  |  |  |           amqp_method_record() | rabbit_types:connection_exit(). | 
					
						
							|  |  |  | -spec decode_properties(non_neg_integer(), binary()) -> amqp_property_record(). | 
					
						
							|  |  |  | -spec encode_method_fields(amqp_method_record()) -> binary(). | 
					
						
							|  |  |  | -spec encode_properties(amqp_property_record()) -> binary(). | 
					
						
							|  |  |  | -spec lookup_amqp_exception(amqp_exception()) -> | 
					
						
							|  |  |  |           {boolean(), amqp_exception_code(), binary()}. | 
					
						
							|  |  |  | -spec amqp_exception(amqp_exception_code()) -> amqp_exception(). | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | bitvalue(true) -> 1; | 
					
						
							|  |  |  | bitvalue(false) -> 0; | 
					
						
							|  |  |  | bitvalue(undefined) -> 0. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | shortstr_size(S) -> | 
					
						
							|  |  |  |     case size(S) of | 
					
						
							|  |  |  |         Len when Len =< 255 -> Len; | 
					
						
							|  |  |  |         _                   -> exit(method_field_shortstr_overflow) | 
					
						
							|  |  |  |     end. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -define(SHORTSTR_VAL(R, L, V, X), | 
					
						
							|  |  |  |         begin | 
					
						
							|  |  |  |             <<L:8/unsigned, V:L/binary, X/binary>> = R, | 
					
						
							|  |  |  |             {V, X} | 
					
						
							|  |  |  |         end). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -define(LONGSTR_VAL(R, L, V, X), | 
					
						
							|  |  |  |         begin | 
					
						
							|  |  |  |             <<L:32/unsigned, V:L/binary, X/binary>> = R, | 
					
						
							|  |  |  |             {V, X} | 
					
						
							|  |  |  |         end). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -define(SHORT_VAL(R, L, V, X), | 
					
						
							|  |  |  |         begin | 
					
						
							|  |  |  |             <<V:8/unsigned, X/binary>> = R, | 
					
						
							|  |  |  |             {V, X} | 
					
						
							|  |  |  |         end). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -define(LONG_VAL(R, L, V, X), | 
					
						
							|  |  |  |         begin | 
					
						
							|  |  |  |             <<V:32/unsigned, X/binary>> = R, | 
					
						
							|  |  |  |             {V, X} | 
					
						
							|  |  |  |         end). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -define(LONGLONG_VAL(R, L, V, X), | 
					
						
							|  |  |  |         begin | 
					
						
							|  |  |  |             <<V:64/unsigned, X/binary>> = R, | 
					
						
							|  |  |  |             {V, X} | 
					
						
							|  |  |  |         end). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -define(OCTET_VAL(R, L, V, X), | 
					
						
							|  |  |  |         begin | 
					
						
							|  |  |  |             <<V:8/unsigned, X/binary>> = R, | 
					
						
							|  |  |  |             {V, X} | 
					
						
							|  |  |  |         end). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -define(TABLE_VAL(R, L, V, X), | 
					
						
							|  |  |  |         begin | 
					
						
							|  |  |  |             <<L:32/unsigned, V:L/binary, X/binary>> = R, | 
					
						
							|  |  |  |             {rabbit_binary_parser:parse_table(V), X} | 
					
						
							|  |  |  |         end). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -define(TIMESTAMP_VAL(R, L, V, X), | 
					
						
							|  |  |  |         begin | 
					
						
							|  |  |  |             <<V:64/unsigned, X/binary>> = R, | 
					
						
							|  |  |  |             {V, X} | 
					
						
							|  |  |  |         end). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -define(SHORTSTR_PROP(X, L), | 
					
						
							|  |  |  |         begin | 
					
						
							|  |  |  |             L = size(X), | 
					
						
							|  |  |  |             if L < 256 -> <<L:8, X:L/binary>>; | 
					
						
							|  |  |  |                true    -> exit(content_properties_shortstr_overflow) | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |         end). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -define(LONGSTR_PROP(X, L), | 
					
						
							|  |  |  |         begin | 
					
						
							|  |  |  |             L = size(X), | 
					
						
							|  |  |  |             <<L:32, X:L/binary>> | 
					
						
							|  |  |  |         end). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -define(OCTET_PROP(X, L),     <<X:8/unsigned>>). | 
					
						
							|  |  |  | -define(SHORT_PROP(X, L),     <<X:16/unsigned>>). | 
					
						
							|  |  |  | -define(LONG_PROP(X, L),      <<X:32/unsigned>>). | 
					
						
							|  |  |  | -define(LONGLONG_PROP(X, L),  <<X:64/unsigned>>). | 
					
						
							|  |  |  | -define(TIMESTAMP_PROP(X, L), <<X:64/unsigned>>). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -define(TABLE_PROP(X, T), | 
					
						
							|  |  |  |         begin | 
					
						
							|  |  |  |             T = rabbit_binary_generator:generate_table(X), | 
					
						
							|  |  |  |             <<(size(T)):32, T/binary>> | 
					
						
							|  |  |  |         end). | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  | """)
 | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |     version = "{%d, %d, %d}" % (spec.major, spec.minor, spec.revision) | 
					
						
							|  |  |  |     if version == '{8, 0, 0}': version = '{0, 8, 0}' | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("version() -> %s." % (version)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for m in methods: genLookupMethodName(m) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("lookup_method_name({_ClassId, _MethodId} = Id) -> exit({unknown_method_id, Id}).") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for c in spec.allClasses(): genLookupClassName(c) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("lookup_class_name(ClassId) -> exit({unknown_class_id, ClassId}).") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for m in methods: genMethodId(m) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("method_id(Name) -> exit({unknown_method_name, Name}).") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for m in methods: genMethodHasContent(m) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("method_has_content(Name) -> exit({unknown_method_name, Name}).") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for m in methods: genMethodIsSynchronous(m) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("is_method_synchronous(Name) -> exit({unknown_method_name, Name}).") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for m in methods: genMethodRecord(m) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("method_record(Name) -> exit({unknown_method_name, Name}).") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for m in methods: genMethodFieldNames(m) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("method_fieldnames(Name) -> exit({unknown_method_name, Name}).") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for m in methods: genDecodeMethodFields(m) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("decode_method_fields(Name, BinaryFields) ->") | 
					
						
							|  |  |  |     print("  rabbit_misc:frame_error(Name, BinaryFields).") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for c in spec.allClasses(): genDecodeProperties(c) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("decode_properties(ClassId, _BinaryFields) -> exit({unknown_class_id, ClassId}).") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for m in methods: genEncodeMethodFields(m) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("encode_method_fields(Record) -> exit({unknown_method_name, element(1, Record)}).") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for c in spec.allClasses(): genEncodeProperties(c) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("encode_properties(Record) -> exit({unknown_properties_record, Record}).") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (c,v,cls) in spec.constants: genLookupException(c,v,cls) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("lookup_amqp_exception(Code) ->") | 
					
						
							|  |  |  |     print("  rabbit_log:warning(\"Unknown AMQP error code '~p'~n\", [Code]),") | 
					
						
							|  |  |  |     print("  {true, ?INTERNAL_ERROR, <<\"INTERNAL_ERROR\">>}.") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for(c,v,cls) in spec.constants: genAmqpException(c,v,cls) | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("amqp_exception(_Code) -> undefined.") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | def genHrl(spec): | 
					
						
							|  |  |  |     def fieldNameList(fields): | 
					
						
							|  |  |  |         return ', '.join([erlangize(f.name) for f in fields]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def fieldNameListDefaults(fields): | 
					
						
							|  |  |  |         def fillField(field): | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |             result = erlangize(field.name) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |             if field.defaultvalue != None: | 
					
						
							|  |  |  |                 conv_fn = erlangDefaultValueTypeConvMap[type(field.defaultvalue)] | 
					
						
							|  |  |  |                 result += ' = ' + conv_fn(field.defaultvalue) | 
					
						
							|  |  |  |             return result | 
					
						
							|  |  |  |         return ', '.join([fillField(f) for f in fields]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     methods = spec.allMethods() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     printFileHeader() | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("-define(PROTOCOL_PORT, %d)." % (spec.port)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (c,v,cls) in spec.constants: | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("-define(%s, %s)." % (erlangConstantName(c), v)) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("%% Method field records.") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |     for m in methods: | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("-record(%s, {%s})." % (m.erlangName(), fieldNameListDefaults(m.arguments))) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |     print("%% Class property records.") | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  |     for c in spec.allClasses(): | 
					
						
							| 
									
										
										
										
											2015-12-17 21:03:26 +08:00
										 |  |  |         print("-record('P_%s', {%s})." % (erlangize(c.name), fieldNameList(c.fields))) | 
					
						
							| 
									
										
										
										
											2015-06-22 18:11:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def generateErl(specPath): | 
					
						
							|  |  |  |     genErl(AmqpSpec(specPath)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def generateHrl(specPath): | 
					
						
							|  |  |  |     genHrl(AmqpSpec(specPath)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							|  |  |  |     do_main_dict({"header": generateHrl, | 
					
						
							|  |  |  |                   "body": generateErl}) | 
					
						
							|  |  |  | 
 |