misc.py 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. # misc.py -- For dealing with python2.4 oddness
  2. # Copyright (C) 2008 Canonical Ltd.
  3. #
  4. # This program is free software; you can redistribute it and/or
  5. # modify it under the terms of the GNU General Public License
  6. # as published by the Free Software Foundation; version 2
  7. # of the License or (at your option) a later version.
  8. #
  9. # This program is distributed in the hope that it will be useful,
  10. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. # GNU General Public License for more details.
  13. #
  14. # You should have received a copy of the GNU General Public License
  15. # along with this program; if not, write to the Free Software
  16. # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
  17. # MA 02110-1301, USA.
  18. """Misc utilities to work with python <2.6.
  19. These utilities can all be deleted when dulwich decides it wants to stop
  20. support for python <2.6.
  21. """
  22. try:
  23. import hashlib
  24. except ImportError:
  25. import sha
  26. try:
  27. from urlparse import parse_qs
  28. except ImportError:
  29. from cgi import parse_qs
  30. try:
  31. from os import SEEK_END
  32. except ImportError:
  33. SEEK_END = 2
  34. import struct
  35. class defaultdict(dict):
  36. """A python 2.4 equivalent of collections.defaultdict."""
  37. def __init__(self, default_factory=None, *a, **kw):
  38. if (default_factory is not None and
  39. not hasattr(default_factory, '__call__')):
  40. raise TypeError('first argument must be callable')
  41. dict.__init__(self, *a, **kw)
  42. self.default_factory = default_factory
  43. def __getitem__(self, key):
  44. try:
  45. return dict.__getitem__(self, key)
  46. except KeyError:
  47. return self.__missing__(key)
  48. def __missing__(self, key):
  49. if self.default_factory is None:
  50. raise KeyError(key)
  51. self[key] = value = self.default_factory()
  52. return value
  53. def __reduce__(self):
  54. if self.default_factory is None:
  55. args = tuple()
  56. else:
  57. args = self.default_factory,
  58. return type(self), args, None, None, self.items()
  59. def copy(self):
  60. return self.__copy__()
  61. def __copy__(self):
  62. return type(self)(self.default_factory, self)
  63. def __deepcopy__(self, memo):
  64. import copy
  65. return type(self)(self.default_factory,
  66. copy.deepcopy(self.items()))
  67. def __repr__(self):
  68. return 'defaultdict(%s, %s)' % (self.default_factory,
  69. dict.__repr__(self))
  70. def make_sha(source=''):
  71. """A python2.4 workaround for the sha/hashlib module fiasco."""
  72. try:
  73. return hashlib.sha1(source)
  74. except NameError:
  75. sha1 = sha.sha(source)
  76. return sha1
  77. def unpack_from(fmt, buf, offset=0):
  78. """A python2.4 workaround for struct missing unpack_from."""
  79. try:
  80. return struct.unpack_from(fmt, buf, offset)
  81. except AttributeError:
  82. b = buf[offset:offset+struct.calcsize(fmt)]
  83. return struct.unpack(fmt, b)