repository.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432
  1. from configparser import RawConfigParser
  2. from binascii import hexlify
  3. import errno
  4. import os
  5. import re
  6. import shutil
  7. import struct
  8. from zlib import crc32
  9. from .hashindex import NSIndex
  10. from .helpers import Error, IntegrityError, read_msgpack, write_msgpack, unhexlify, UpgradableLock
  11. from .lrucache import LRUCache
  12. MAX_OBJECT_SIZE = 20 * 1024 * 1024
  13. MAGIC = b'ATTICSEG'
  14. TAG_PUT = 0
  15. TAG_DELETE = 1
  16. TAG_COMMIT = 2
  17. class Repository(object):
  18. """Filesystem based transactional key value store
  19. On disk layout:
  20. dir/README
  21. dir/config
  22. dir/data/<X / SEGMENTS_PER_DIR>/<X>
  23. dir/index.X
  24. dir/hints.X
  25. """
  26. DEFAULT_MAX_SEGMENT_SIZE = 5 * 1024 * 1024
  27. DEFAULT_SEGMENTS_PER_DIR = 10000
  28. class DoesNotExist(Error):
  29. """Repository {} does not exist"""
  30. class AlreadyExists(Error):
  31. """Repository {} already exists"""
  32. class InvalidRepository(Error):
  33. """{} is not a valid repository"""
  34. def __init__(self, path, create=False):
  35. self.path = path
  36. self.io = None
  37. self.lock = None
  38. if create:
  39. self.create(path)
  40. self.open(path)
  41. def __del__(self):
  42. self.close()
  43. def create(self, path):
  44. """Create a new empty repository at `path`
  45. """
  46. if os.path.exists(path) and (not os.path.isdir(path) or os.listdir(path)):
  47. raise self.AlreadyExists(path)
  48. if not os.path.exists(path):
  49. os.mkdir(path)
  50. with open(os.path.join(path, 'README'), 'w') as fd:
  51. fd.write('This is an Attic repository\n')
  52. os.mkdir(os.path.join(path, 'data'))
  53. config = RawConfigParser()
  54. config.add_section('repository')
  55. config.set('repository', 'version', '1')
  56. config.set('repository', 'segments_per_dir', self.DEFAULT_SEGMENTS_PER_DIR)
  57. config.set('repository', 'max_segment_size', self.DEFAULT_MAX_SEGMENT_SIZE)
  58. config.set('repository', 'id', hexlify(os.urandom(32)).decode('ascii'))
  59. with open(os.path.join(path, 'config'), 'w') as fd:
  60. config.write(fd)
  61. def open(self, path):
  62. self.head = None
  63. self.path = path
  64. if not os.path.isdir(path):
  65. raise self.DoesNotExist(path)
  66. self.config = RawConfigParser()
  67. self.config.read(os.path.join(self.path, 'config'))
  68. if not 'repository' in self.config.sections() or self.config.getint('repository', 'version') != 1:
  69. raise self.InvalidRepository(path)
  70. self.lock = UpgradableLock(os.path.join(path, 'config'))
  71. self.max_segment_size = self.config.getint('repository', 'max_segment_size')
  72. self.segments_per_dir = self.config.getint('repository', 'segments_per_dir')
  73. self.id = unhexlify(self.config.get('repository', 'id').strip())
  74. self.rollback()
  75. def close(self):
  76. if self.lock:
  77. self.rollback()
  78. self.lock.release()
  79. self.lock = None
  80. def commit(self, rollback=True):
  81. """Commit transaction
  82. """
  83. self.io.write_commit()
  84. self.compact_segments()
  85. self.write_index()
  86. self.rollback()
  87. def _available_indices(self, reverse=False):
  88. names = [int(name[6:]) for name in os.listdir(self.path) if re.match('index\.\d+', name)]
  89. names.sort(reverse=reverse)
  90. return names
  91. def open_index(self, head, read_only=False):
  92. if head is None:
  93. self.index = NSIndex.create(os.path.join(self.path, 'index.tmp').encode('utf-8'))
  94. self.segments = {}
  95. self.compact = set()
  96. else:
  97. if read_only:
  98. self.index = NSIndex((os.path.join(self.path, 'index.%d') % head).encode('utf-8'), readonly=True)
  99. else:
  100. shutil.copy(os.path.join(self.path, 'index.%d' % head),
  101. os.path.join(self.path, 'index.tmp'))
  102. self.index = NSIndex(os.path.join(self.path, 'index.tmp').encode('utf-8'))
  103. hints = read_msgpack(os.path.join(self.path, 'hints.%d' % head))
  104. if hints[b'version'] != 1:
  105. raise ValueError('Unknown hints file version: %d' % hints['version'])
  106. self.segments = hints[b'segments']
  107. self.compact = set(hints[b'compact'])
  108. def write_index(self):
  109. hints = {b'version': 1,
  110. b'segments': self.segments,
  111. b'compact': list(self.compact)}
  112. write_msgpack(os.path.join(self.path, 'hints.%d' % self.io.head), hints)
  113. self.index.flush()
  114. os.rename(os.path.join(self.path, 'index.tmp'),
  115. os.path.join(self.path, 'index.%d' % self.io.head))
  116. # Remove old indices
  117. current = '.%d' % self.io.head
  118. for name in os.listdir(self.path):
  119. if not name.startswith('index.') and not name.startswith('hints.'):
  120. continue
  121. if name.endswith(current):
  122. continue
  123. os.unlink(os.path.join(self.path, name))
  124. def compact_segments(self):
  125. """Compact sparse segments by copying data into new segments
  126. """
  127. if not self.compact:
  128. return
  129. def lookup(tag, key):
  130. return tag == TAG_PUT and self.index.get(key, (-1, -1))[0] == segment
  131. segments = self.segments
  132. for segment in sorted(self.compact):
  133. if segments[segment] > 0:
  134. for tag, key, data in self.io.iter_objects(segment, lookup, include_data=True):
  135. new_segment, offset = self.io.write_put(key, data)
  136. self.index[key] = new_segment, offset
  137. segments.setdefault(new_segment, 0)
  138. segments[new_segment] += 1
  139. segments[segment] -= 1
  140. assert segments[segment] == 0
  141. self.io.write_commit()
  142. for segment in self.compact:
  143. assert self.segments.pop(segment) == 0
  144. self.io.delete_segment(segment)
  145. self.compact = set()
  146. def recover(self, path):
  147. """Recover missing index by replaying logs"""
  148. start = None
  149. available = self._available_indices()
  150. if available:
  151. start = available[-1]
  152. self.open_index(start)
  153. for segment, filename in self.io._segment_names():
  154. if start is not None and segment <= start:
  155. continue
  156. self.segments[segment] = 0
  157. for tag, key, offset in self.io.iter_objects(segment):
  158. if tag == TAG_PUT:
  159. try:
  160. s, _ = self.index[key]
  161. self.compact.add(s)
  162. self.segments[s] -= 1
  163. except KeyError:
  164. pass
  165. self.index[key] = segment, offset
  166. self.segments[segment] += 1
  167. elif tag == TAG_DELETE:
  168. try:
  169. s, _ = self.index.pop(key)
  170. self.segments[s] -= 1
  171. self.compact.add(s)
  172. self.compact.add(segment)
  173. except KeyError:
  174. pass
  175. if self.segments[segment] == 0:
  176. self.compact.add(segment)
  177. if self.io.head is not None:
  178. self.write_index()
  179. def rollback(self):
  180. """
  181. """
  182. self._active_txn = False
  183. if self.io:
  184. self.io.close()
  185. self.io = LoggedIO(self.path, self.max_segment_size, self.segments_per_dir)
  186. if self.io.head is not None and not os.path.exists(os.path.join(self.path, 'index.%d' % self.io.head)):
  187. self.lock.upgrade()
  188. self.recover(self.path)
  189. self.open_index(self.io.head, read_only=True)
  190. def _len(self):
  191. return len(self.index)
  192. def get(self, id):
  193. try:
  194. segment, offset = self.index[id]
  195. return self.io.read(segment, offset, id)
  196. except KeyError:
  197. raise self.DoesNotExist(self.path)
  198. def get_many(self, ids, peek=None):
  199. for id in ids:
  200. yield self.get(id)
  201. def put(self, id, data, wait=True):
  202. if not self._active_txn:
  203. self._active_txn = True
  204. self.lock.upgrade()
  205. self.open_index(self.io.head)
  206. try:
  207. segment, _ = self.index[id]
  208. self.segments[segment] -= 1
  209. self.compact.add(segment)
  210. segment = self.io.write_delete(id)
  211. self.segments.setdefault(segment, 0)
  212. self.compact.add(segment)
  213. except KeyError:
  214. pass
  215. segment, offset = self.io.write_put(id, data)
  216. self.segments.setdefault(segment, 0)
  217. self.segments[segment] += 1
  218. self.index[id] = segment, offset
  219. def delete(self, id, wait=True):
  220. if not self._active_txn:
  221. self._active_txn = True
  222. self.lock.upgrade()
  223. self.open_index(self.io.head)
  224. try:
  225. segment, offset = self.index.pop(id)
  226. self.segments[segment] -= 1
  227. self.compact.add(segment)
  228. segment = self.io.write_delete(id)
  229. self.compact.add(segment)
  230. self.segments.setdefault(segment, 0)
  231. except KeyError:
  232. raise self.DoesNotExist(self.path)
  233. def add_callback(self, cb, data):
  234. cb(None, None, data)
  235. class LoggedIO(object):
  236. header_fmt = struct.Struct('<IIB')
  237. assert header_fmt.size == 9
  238. put_header_fmt = struct.Struct('<IIB32s')
  239. assert put_header_fmt.size == 41
  240. header_no_crc_fmt = struct.Struct('<IB')
  241. assert header_no_crc_fmt.size == 5
  242. crc_fmt = struct.Struct('<I')
  243. assert crc_fmt.size == 4
  244. _commit = header_no_crc_fmt.pack(9, TAG_COMMIT)
  245. COMMIT = crc_fmt.pack(crc32(_commit)) + _commit
  246. def __init__(self, path, limit, segments_per_dir, capacity=100):
  247. self.path = path
  248. self.fds = LRUCache(capacity)
  249. self.segment = None
  250. self.limit = limit
  251. self.segments_per_dir = segments_per_dir
  252. self.offset = 0
  253. self._write_fd = None
  254. self.head = None
  255. self.cleanup()
  256. def close(self):
  257. for segment in list(self.fds.keys()):
  258. self.fds.pop(segment).close()
  259. self.close_segment()
  260. self.fds = None # Just to make sure we're disabled
  261. def _segment_names(self, reverse=False):
  262. for dirpath, dirs, filenames in os.walk(os.path.join(self.path, 'data')):
  263. dirs.sort(key=int, reverse=reverse)
  264. filenames.sort(key=int, reverse=reverse)
  265. for filename in filenames:
  266. yield int(filename), os.path.join(dirpath, filename)
  267. def cleanup(self):
  268. """Delete segment files left by aborted transactions
  269. """
  270. self.head = None
  271. self.segment = 0
  272. for segment, filename in self._segment_names(reverse=True):
  273. if self.is_complete_segment(filename):
  274. self.head = segment
  275. self.segment = self.head + 1
  276. return
  277. else:
  278. os.unlink(filename)
  279. def is_complete_segment(self, filename):
  280. with open(filename, 'rb') as fd:
  281. try:
  282. fd.seek(-self.header_fmt.size, os.SEEK_END)
  283. except Exception as e:
  284. # return False if segment file is empty or too small
  285. if e.errno == errno.EINVAL:
  286. return False
  287. raise e
  288. return fd.read(self.header_fmt.size) == self.COMMIT
  289. def segment_filename(self, segment):
  290. return os.path.join(self.path, 'data', str(segment // self.segments_per_dir), str(segment))
  291. def get_write_fd(self, no_new=False):
  292. if not no_new and self.offset and self.offset > self.limit:
  293. self.close_segment()
  294. if not self._write_fd:
  295. if self.segment % self.segments_per_dir == 0:
  296. dirname = os.path.join(self.path, 'data', str(self.segment // self.segments_per_dir))
  297. if not os.path.exists(dirname):
  298. os.mkdir(dirname)
  299. self._write_fd = open(self.segment_filename(self.segment), 'ab')
  300. self._write_fd.write(MAGIC)
  301. self.offset = 8
  302. return self._write_fd
  303. def get_fd(self, segment):
  304. try:
  305. return self.fds[segment]
  306. except KeyError:
  307. fd = open(self.segment_filename(segment), 'rb')
  308. self.fds[segment] = fd
  309. return fd
  310. def delete_segment(self, segment):
  311. try:
  312. os.unlink(self.segment_filename(segment))
  313. except OSError:
  314. pass
  315. def iter_objects(self, segment, lookup=None, include_data=False):
  316. fd = self.get_fd(segment)
  317. fd.seek(0)
  318. if fd.read(8) != MAGIC:
  319. raise IntegrityError('Invalid segment header')
  320. offset = 8
  321. header = fd.read(self.header_fmt.size)
  322. while header:
  323. crc, size, tag = self.header_fmt.unpack(header)
  324. if size > MAX_OBJECT_SIZE:
  325. raise IntegrityError('Invalid segment object size')
  326. rest = fd.read(size - self.header_fmt.size)
  327. if crc32(rest, crc32(memoryview(header)[4:])) & 0xffffffff != crc:
  328. raise IntegrityError('Segment checksum mismatch')
  329. if tag not in (TAG_PUT, TAG_DELETE, TAG_COMMIT):
  330. raise IntegrityError('Invalid segment entry header')
  331. key = None
  332. if tag in (TAG_PUT, TAG_DELETE):
  333. key = rest[:32]
  334. if not lookup or lookup(tag, key):
  335. if include_data:
  336. yield tag, key, rest[32:]
  337. else:
  338. yield tag, key, offset
  339. offset += size
  340. header = fd.read(self.header_fmt.size)
  341. def read(self, segment, offset, id):
  342. if segment == self.segment:
  343. self._write_fd.flush()
  344. fd = self.get_fd(segment)
  345. fd.seek(offset)
  346. header = fd.read(self.put_header_fmt.size)
  347. crc, size, tag, key = self.put_header_fmt.unpack(header)
  348. if size > MAX_OBJECT_SIZE:
  349. raise IntegrityError('Invalid segment object size')
  350. data = fd.read(size - self.put_header_fmt.size)
  351. if crc32(data, crc32(memoryview(header)[4:])) & 0xffffffff != crc:
  352. raise IntegrityError('Segment checksum mismatch')
  353. if tag != TAG_PUT or id != key:
  354. raise IntegrityError('Invalid segment entry header')
  355. return data
  356. def write_put(self, id, data):
  357. size = len(data) + self.put_header_fmt.size
  358. fd = self.get_write_fd()
  359. offset = self.offset
  360. header = self.header_no_crc_fmt.pack(size, TAG_PUT)
  361. crc = self.crc_fmt.pack(crc32(data, crc32(id, crc32(header))) & 0xffffffff)
  362. fd.write(b''.join((crc, header, id, data)))
  363. self.offset += size
  364. return self.segment, offset
  365. def write_delete(self, id):
  366. fd = self.get_write_fd()
  367. header = self.header_no_crc_fmt.pack(self.put_header_fmt.size, TAG_DELETE)
  368. crc = self.crc_fmt.pack(crc32(id, crc32(header)) & 0xffffffff)
  369. fd.write(b''.join((crc, header, id)))
  370. self.offset += self.put_header_fmt.size
  371. return self.segment
  372. def write_commit(self):
  373. fd = self.get_write_fd(no_new=True)
  374. header = self.header_no_crc_fmt.pack(self.header_fmt.size, TAG_COMMIT)
  375. crc = self.crc_fmt.pack(crc32(header) & 0xffffffff)
  376. fd.write(b''.join((crc, header)))
  377. self.head = self.segment
  378. self.close_segment()
  379. def close_segment(self):
  380. if self._write_fd:
  381. self.segment += 1
  382. self.offset = 0
  383. os.fsync(self._write_fd)
  384. self._write_fd.close()
  385. self._write_fd = None