backend_open62541_datatypes.py 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. from datatypes import *
  2. import datetime
  3. import re
  4. import logging
  5. logger = logging.getLogger(__name__)
  6. def generateBooleanCode(value):
  7. if value:
  8. return "true"
  9. return "false"
  10. # Strip invalid characters to create valid C identifiers (variable names etc):
  11. def makeCIdentifier(value):
  12. return re.sub(r'[^\w]', '', value)
  13. # Escape C strings:
  14. def makeCLiteral(value):
  15. return re.sub(r'(?<!\\)"', r'\\"', value.replace('\\', r'\\').replace('"', r'\"').replace('\n', r'\\n').replace('\r', r''))
  16. def splitStringLiterals(value, splitLength=500):
  17. """
  18. Split a string literal longer than splitLength into smaller literals.
  19. E.g. "Some very long text" will be split into "Some ver" "y long te" "xt"
  20. On VS2008 there is a maximum allowed length of a single string literal.
  21. """
  22. value = value.strip()
  23. if len(value) < splitLength or splitLength == 0:
  24. return "\"" + re.sub(r'(?<!\\)"', r'\\"', value) + "\""
  25. ret = ""
  26. tmp = value
  27. while len(tmp) > splitLength:
  28. ret += "\"" + tmp[:splitLength].replace('"', r'\"') + "\" "
  29. tmp = tmp[splitLength:]
  30. ret += "\"" + re.sub(r'(?<!\\)"', r'\\"', tmp) + "\" "
  31. return ret
  32. def generateStringCode(value, alloc=False):
  33. value = makeCLiteral(value)
  34. return u"UA_STRING{}({})".format("_ALLOC" if alloc else "", splitStringLiterals(value))
  35. def generateXmlElementCode(value, alloc=False):
  36. value = makeCLiteral(value)
  37. return u"UA_XMLELEMENT{}({})".format("_ALLOC" if alloc else "", splitStringLiterals(value))
  38. def generateByteStringCode(value, valueName, global_var_code, isPointer):
  39. if isinstance(value, str):
  40. # PY3 returns a byte array for b64decode, while PY2 returns a string.
  41. # Therefore convert it to bytes
  42. asciiarray = bytearray()
  43. asciiarray.extend(value)
  44. asciiarray = list(asciiarray)
  45. else:
  46. asciiarray = list(value)
  47. asciiarraystr = str(asciiarray).rstrip(']').lstrip('[')
  48. cleanValueName = re.sub(r"->", "__", re.sub(r"\.", "_", valueName))
  49. global_var_code.append("static const UA_Byte {cleanValueName}_byteArray[{len}] = {{{data}}};".format(
  50. len=len(asciiarray), data=asciiarraystr, cleanValueName=cleanValueName
  51. ))
  52. # Cast away const with '(UA_Byte *)(void*)(uintptr_t)' since we know that UA_Server_addNode_begin will copy the content
  53. return "{instance}{accessor}length = {len};\n{instance}{accessor}data = (UA_Byte *)(void*)(uintptr_t){cleanValueName}_byteArray;"\
  54. .format(len=len(asciiarray), instance=valueName, cleanValueName=cleanValueName,
  55. accessor='->' if isPointer else '.')
  56. def generateLocalizedTextCode(value, alloc=False):
  57. vt = makeCLiteral(value.text)
  58. return u"UA_LOCALIZEDTEXT{}(\"{}\", {})".format("_ALLOC" if alloc else "", '' if value.locale is None else value.locale,
  59. splitStringLiterals(vt))
  60. def generateQualifiedNameCode(value, alloc=False,):
  61. vn = makeCLiteral(value.name)
  62. return u"UA_QUALIFIEDNAME{}(ns[{}], {})".format("_ALLOC" if alloc else "",
  63. str(value.ns), splitStringLiterals(vn))
  64. def generateNodeIdCode(value):
  65. if not value:
  66. return "UA_NODEID_NUMERIC(0, 0)"
  67. if value.i != None:
  68. return "UA_NODEID_NUMERIC(ns[%s], %s)" % (value.ns, value.i)
  69. elif value.s != None:
  70. v = makeCLiteral(value.s)
  71. return u"UA_NODEID_STRING(ns[%s], \"%s\")" % (value.ns, v)
  72. raise Exception(str(value) + " no NodeID generation for bytestring and guid..")
  73. def generateExpandedNodeIdCode(value):
  74. if value.i != None:
  75. return "UA_EXPANDEDNODEID_NUMERIC(ns[%s], %s)" % (str(value.ns), str(value.i))
  76. elif value.s != None:
  77. vs = makeCLiteral(value.s)
  78. return u"UA_EXPANDEDNODEID_STRING(ns[%s], \"%s\")" % (str(value.ns), vs)
  79. raise Exception(str(value) + " no NodeID generation for bytestring and guid..")
  80. def generateDateTimeCode(value):
  81. epoch = datetime.datetime.utcfromtimestamp(0)
  82. mSecsSinceEpoch = int((value - epoch).total_seconds() * 1000.0)
  83. return "( (UA_DateTime)(" + str(mSecsSinceEpoch) + " * UA_DATETIME_MSEC) + UA_DATETIME_UNIX_EPOCH)"
  84. def generateNodeValueCode(prepend , node, instanceName, valueName, global_var_code, asIndirect=False):
  85. if type(node) in [Boolean, Byte, SByte, Int16, UInt16, Int32, UInt32, Int64, UInt64, Float, Double]:
  86. return prepend + "(UA_" + node.__class__.__name__ + ") " + str(node.value) + ";"
  87. elif type(node) == String:
  88. return prepend + generateStringCode(node.value, alloc=asIndirect) + ";"
  89. elif type(node) == XmlElement:
  90. return prepend + generateXmlElementCode(node.value, alloc=asIndirect) + ";"
  91. elif type(node) == ByteString:
  92. # replace whitespaces between tags and remove newlines
  93. return prepend + "UA_BYTESTRING_NULL;" if not node.value else generateByteStringCode(
  94. node.value, valueName, global_var_code, isPointer=asIndirect)
  95. # the replacements done here is just for the array form can be workable in C code. It doesn't couses any problem
  96. # because the core data used here is already in byte form. So, there is no way we disturb it.
  97. elif type(node) == LocalizedText:
  98. return prepend + generateLocalizedTextCode(node, alloc=asIndirect) + ";"
  99. elif type(node) == NodeId:
  100. return prepend + generateNodeIdCode(node) + ";"
  101. elif type(node) == ExpandedNodeId:
  102. return prepend + generateExpandedNodeIdCode(node) + ";"
  103. elif type(node) == DateTime:
  104. return prepend + generateDateTimeCode(node.value) + ";"
  105. elif type(node) == QualifiedName:
  106. return prepend + generateQualifiedNameCode(node.value, alloc=asIndirect) + ";"
  107. elif type(node) == StatusCode:
  108. raise Exception("generateNodeValueCode for type " + node.__class__.name + " not implemented")
  109. elif type(node) == DiagnosticInfo:
  110. raise Exception("generateNodeValueCode for type " + node.__class__.name + " not implemented")
  111. elif type(node) == Guid:
  112. raise Exception("generateNodeValueCode for type " + node.__class__.name + " not implemented")
  113. elif type(node) == ExtensionObject:
  114. if asIndirect == False:
  115. return prepend + "*" + str(instanceName) + ";"
  116. return prepend + str(instanceName) + ";"