minicompat.py 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. """Python version compatibility support for minidom."""
  2. # This module should only be imported using "import *".
  3. #
  4. # The following names are defined:
  5. #
  6. # NodeList -- lightest possible NodeList implementation
  7. #
  8. # EmptyNodeList -- lightest possible NodeList that is guaranteed to
  9. # remain empty (immutable)
  10. #
  11. # StringTypes -- tuple of defined string types
  12. #
  13. # defproperty -- function used in conjunction with GetattrMagic;
  14. # using these together is needed to make them work
  15. # as efficiently as possible in both Python 2.2+
  16. # and older versions. For example:
  17. #
  18. # class MyClass(GetattrMagic):
  19. # def _get_myattr(self):
  20. # return something
  21. #
  22. # defproperty(MyClass, "myattr",
  23. # "return some value")
  24. #
  25. # For Python 2.2 and newer, this will construct a
  26. # property object on the class, which avoids
  27. # needing to override __getattr__(). It will only
  28. # work for read-only attributes.
  29. #
  30. # For older versions of Python, inheriting from
  31. # GetattrMagic will use the traditional
  32. # __getattr__() hackery to achieve the same effect,
  33. # but less efficiently.
  34. #
  35. # defproperty() should be used for each version of
  36. # the relevant _get_<property>() function.
  37. __all__ = ["NodeList", "EmptyNodeList", "StringTypes", "defproperty"]
  38. import xml.dom
  39. try:
  40. unicode
  41. except NameError:
  42. StringTypes = type(''),
  43. else:
  44. StringTypes = type(''), type(unicode(''))
  45. class NodeList(list):
  46. __slots__ = ()
  47. def item(self, index):
  48. if 0 <= index < len(self):
  49. return self[index]
  50. def _get_length(self):
  51. return len(self)
  52. def _set_length(self, value):
  53. raise xml.dom.NoModificationAllowedErr(
  54. "attempt to modify read-only attribute 'length'")
  55. length = property(_get_length, _set_length,
  56. doc="The number of nodes in the NodeList.")
  57. # For backward compatibility
  58. def __setstate__(self, state):
  59. if state is None:
  60. state = []
  61. self[:] = state
  62. class EmptyNodeList(tuple):
  63. __slots__ = ()
  64. def __add__(self, other):
  65. NL = NodeList()
  66. NL.extend(other)
  67. return NL
  68. def __radd__(self, other):
  69. NL = NodeList()
  70. NL.extend(other)
  71. return NL
  72. def item(self, index):
  73. return None
  74. def _get_length(self):
  75. return 0
  76. def _set_length(self, value):
  77. raise xml.dom.NoModificationAllowedErr(
  78. "attempt to modify read-only attribute 'length'")
  79. length = property(_get_length, _set_length,
  80. doc="The number of nodes in the NodeList.")
  81. def defproperty(klass, name, doc):
  82. get = getattr(klass, ("_get_" + name)).im_func
  83. def set(self, value, name=name):
  84. raise xml.dom.NoModificationAllowedErr(
  85. "attempt to modify read-only attribute " + repr(name))
  86. assert not hasattr(klass, "_set_" + name), \
  87. "expected not to find _set_" + name
  88. prop = property(get, set, doc=doc)
  89. setattr(klass, name, prop)