repository.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545
  1. from configparser import RawConfigParser
  2. from binascii import hexlify
  3. from itertools import islice
  4. import errno
  5. import os
  6. import shutil
  7. import struct
  8. import sys
  9. import time
  10. from zlib import crc32
  11. from .hashindex import NSIndex
  12. from .helpers import Error, IntegrityError, read_msgpack, write_msgpack, unhexlify, UpgradableLock
  13. from .lrucache import LRUCache
  14. MAX_OBJECT_SIZE = 20 * 1024 * 1024
  15. MAGIC = b'ATTICSEG'
  16. TAG_PUT = 0
  17. TAG_DELETE = 1
  18. TAG_COMMIT = 2
  19. class Repository(object):
  20. """Filesystem based transactional key value store
  21. On disk layout:
  22. dir/README
  23. dir/config
  24. dir/data/<X / SEGMENTS_PER_DIR>/<X>
  25. dir/index.X
  26. dir/hints.X
  27. """
  28. DEFAULT_MAX_SEGMENT_SIZE = 5 * 1024 * 1024
  29. DEFAULT_SEGMENTS_PER_DIR = 10000
  30. class DoesNotExist(Error):
  31. """Repository {} does not exist"""
  32. class AlreadyExists(Error):
  33. """Repository {} already exists"""
  34. class InvalidRepository(Error):
  35. """{} is not a valid repository"""
  36. class CheckNeeded(Error):
  37. '''Inconsistency detected. Please run "attic check {}"'''
  38. def __init__(self, path, create=False):
  39. self.path = path
  40. self.io = None
  41. self.lock = None
  42. self.index = None
  43. self._active_txn = False
  44. if create:
  45. self.create(path)
  46. self.open(path)
  47. def __del__(self):
  48. self.close()
  49. def create(self, path):
  50. """Create a new empty repository at `path`
  51. """
  52. if os.path.exists(path) and (not os.path.isdir(path) or os.listdir(path)):
  53. raise self.AlreadyExists(path)
  54. if not os.path.exists(path):
  55. os.mkdir(path)
  56. with open(os.path.join(path, 'README'), 'w') as fd:
  57. fd.write('This is an Attic repository\n')
  58. os.mkdir(os.path.join(path, 'data'))
  59. config = RawConfigParser()
  60. config.add_section('repository')
  61. config.set('repository', 'version', '1')
  62. config.set('repository', 'segments_per_dir', self.DEFAULT_SEGMENTS_PER_DIR)
  63. config.set('repository', 'max_segment_size', self.DEFAULT_MAX_SEGMENT_SIZE)
  64. config.set('repository', 'id', hexlify(os.urandom(32)).decode('ascii'))
  65. with open(os.path.join(path, 'config'), 'w') as fd:
  66. config.write(fd)
  67. def get_index_transaction_id(self):
  68. indicies = sorted((int(name[6:]) for name in os.listdir(self.path) if name.startswith('index.') and name[6:].isdigit()))
  69. if indicies:
  70. return indicies[-1]
  71. else:
  72. return None
  73. def get_transaction_id(self):
  74. index_transaction_id = self.get_index_transaction_id()
  75. segments_transaction_id = self.io.get_segments_transaction_id(index_transaction_id or 0)
  76. if index_transaction_id != segments_transaction_id:
  77. raise self.CheckNeeded(self.path)
  78. return index_transaction_id
  79. def open(self, path):
  80. self.path = path
  81. if not os.path.isdir(path):
  82. raise self.DoesNotExist(path)
  83. self.config = RawConfigParser()
  84. self.config.read(os.path.join(self.path, 'config'))
  85. if not 'repository' in self.config.sections() or self.config.getint('repository', 'version') != 1:
  86. raise self.InvalidRepository(path)
  87. self.lock = UpgradableLock(os.path.join(path, 'config'))
  88. self.max_segment_size = self.config.getint('repository', 'max_segment_size')
  89. self.segments_per_dir = self.config.getint('repository', 'segments_per_dir')
  90. self.id = unhexlify(self.config.get('repository', 'id').strip())
  91. self.io = LoggedIO(self.path, self.max_segment_size, self.segments_per_dir)
  92. def close(self):
  93. if self.lock:
  94. if self.io:
  95. self.io.close()
  96. self.io = None
  97. self.lock.release()
  98. self.lock = None
  99. def commit(self):
  100. """Commit transaction
  101. """
  102. self.io.write_commit()
  103. self.compact_segments()
  104. self.write_index()
  105. self.rollback()
  106. def get_read_only_index(self, transaction_id):
  107. if transaction_id is None:
  108. return {}
  109. return NSIndex((os.path.join(self.path, 'index.%d') % transaction_id).encode('utf-8'), readonly=True)
  110. def get_index(self, transaction_id):
  111. self.lock.upgrade()
  112. if transaction_id is None:
  113. self.index = NSIndex.create(os.path.join(self.path, 'index.tmp').encode('utf-8'))
  114. self.segments = {}
  115. self.compact = set()
  116. else:
  117. self.io.cleanup(transaction_id)
  118. shutil.copy(os.path.join(self.path, 'index.%d' % transaction_id),
  119. os.path.join(self.path, 'index.tmp'))
  120. self.index = NSIndex(os.path.join(self.path, 'index.tmp').encode('utf-8'))
  121. hints = read_msgpack(os.path.join(self.path, 'hints.%d' % transaction_id))
  122. if hints[b'version'] != 1:
  123. raise ValueError('Unknown hints file version: %d' % hints['version'])
  124. self.segments = hints[b'segments']
  125. self.compact = set(hints[b'compact'])
  126. def write_index(self):
  127. hints = {b'version': 1,
  128. b'segments': self.segments,
  129. b'compact': list(self.compact)}
  130. transaction_id = self.io.get_segments_transaction_id()
  131. write_msgpack(os.path.join(self.path, 'hints.%d' % transaction_id), hints)
  132. self.index.flush()
  133. os.rename(os.path.join(self.path, 'index.tmp'),
  134. os.path.join(self.path, 'index.%d' % transaction_id))
  135. # Remove old indices
  136. current = '.%d' % transaction_id
  137. for name in os.listdir(self.path):
  138. if not name.startswith('index.') and not name.startswith('hints.'):
  139. continue
  140. if name.endswith(current):
  141. continue
  142. os.unlink(os.path.join(self.path, name))
  143. def compact_segments(self):
  144. """Compact sparse segments by copying data into new segments
  145. """
  146. if not self.compact:
  147. return
  148. def lookup(tag, key):
  149. return tag == TAG_PUT and self.index.get(key, (-1, -1))[0] == segment
  150. segments = self.segments
  151. for segment in sorted(self.compact):
  152. if segments[segment] > 0:
  153. for tag, key, data in self.io.iter_objects(segment, lookup, include_data=True):
  154. new_segment, offset = self.io.write_put(key, data)
  155. self.index[key] = new_segment, offset
  156. segments.setdefault(new_segment, 0)
  157. segments[new_segment] += 1
  158. segments[segment] -= 1
  159. assert segments[segment] == 0
  160. self.io.write_commit()
  161. for segment in self.compact:
  162. assert self.segments.pop(segment) == 0
  163. self.io.delete_segment(segment)
  164. self.compact = set()
  165. def check(self, progress=False, repair=False):
  166. """Check repository consistency
  167. This method verifies all segment checksums and makes sure
  168. the index is consistent with the data stored in the segments.
  169. """
  170. error_found = False
  171. def report_progress(msg, error=False):
  172. nonlocal error_found
  173. if error:
  174. error_found = True
  175. if error or progress:
  176. print(msg, file=sys.stderr)
  177. sys.stderr.flush()
  178. assert not self._active_txn
  179. report_progress('Starting repository check...')
  180. index_transaction_id = self.get_index_transaction_id()
  181. segments_transaction_id = self.io.get_segments_transaction_id(index_transaction_id)
  182. if index_transaction_id is None and segments_transaction_id is None:
  183. return True
  184. if segments_transaction_id is not None:
  185. transaction_id = segments_transaction_id
  186. else:
  187. transaction_id = index_transaction_id
  188. self.get_index(None)
  189. if index_transaction_id == segments_transaction_id:
  190. current_index = self.get_read_only_index(transaction_id)
  191. else:
  192. current_index = None
  193. report_progress('No suitable index found', error=True)
  194. progress_time = None
  195. for segment, filename in self.io.segment_iterator():
  196. if segment > transaction_id:
  197. if repair:
  198. report_progress('Deleting uncommitted segment {}'.format(segment), error=True)
  199. self.io.delete_segment(segment)
  200. else:
  201. report_progress('Uncommitted segment {} found'.format(segment), error=True)
  202. continue
  203. if progress:
  204. if int(time.time()) != progress_time:
  205. progress_time = int(time.time())
  206. report_progress('Checking segment {}/{}'.format(segment, transaction_id))
  207. try:
  208. objects = list(self.io.iter_objects(segment))
  209. except (IntegrityError, struct.error):
  210. report_progress('Error reading segment {}'.format(segment), error=True)
  211. objects = []
  212. if repair:
  213. self.io.recover_segment(segment, filename)
  214. objects = list(self.io.iter_objects(segment))
  215. self.segments[segment] = 0
  216. for tag, key, offset in objects:
  217. if tag == TAG_PUT:
  218. try:
  219. s, _ = self.index[key]
  220. self.compact.add(s)
  221. self.segments[s] -= 1
  222. report_progress('Key found in more than one segment. Segment={}, key={}'.format(segment, hexlify(key)), error=True)
  223. except KeyError:
  224. pass
  225. self.index[key] = segment, offset
  226. self.segments[segment] += 1
  227. elif tag == TAG_DELETE:
  228. try:
  229. s, _ = self.index.pop(key)
  230. self.segments[s] -= 1
  231. self.compact.add(s)
  232. self.compact.add(segment)
  233. except KeyError:
  234. pass
  235. elif tag == TAG_COMMIT:
  236. continue
  237. else:
  238. report_progress('Unexpected tag {} in segment {}'.format(tag, segment), error=True)
  239. # We might need to add a commit tag if no committed segment is found
  240. if repair and segments_transaction_id is None:
  241. report_progress('Adding commit tag to segment {}'.format(transaction_id))
  242. self.io.segment = transaction_id + 1
  243. self.io.write_commit()
  244. self.io.close_segment()
  245. if current_index and len(current_index) != len(self.index):
  246. report_progress('Index object count mismatch. {} != {}'.format(len(current_index), len(self.index)), error=True)
  247. if not error_found:
  248. report_progress('Repository check complete, no errors found.')
  249. if repair:
  250. self.write_index()
  251. else:
  252. # Delete temporary index file
  253. self.index = None
  254. os.unlink(os.path.join(self.path, 'index.tmp'))
  255. self.rollback()
  256. return not error_found or repair
  257. def rollback(self):
  258. """
  259. """
  260. self.index = None
  261. self._active_txn = False
  262. def __len__(self):
  263. if not self.index:
  264. self.index = self.get_read_only_index(self.get_transaction_id())
  265. return len(self.index)
  266. def list(self, limit=None, marker=None):
  267. if not self.index:
  268. self.index = self.get_read_only_index(self.get_transaction_id())
  269. return [id_ for id_, _ in islice(self.index.iteritems(marker=marker), limit)]
  270. def get(self, id_):
  271. if not self.index:
  272. self.index = self.get_read_only_index(self.get_transaction_id())
  273. try:
  274. segment, offset = self.index[id_]
  275. return self.io.read(segment, offset, id_)
  276. except KeyError:
  277. raise self.DoesNotExist(self.path)
  278. def get_many(self, ids, is_preloaded=False):
  279. for id_ in ids:
  280. yield self.get(id_)
  281. def put(self, id, data, wait=True):
  282. if not self._active_txn:
  283. self.get_index(self.get_transaction_id())
  284. self._active_txn = True
  285. try:
  286. segment, _ = self.index[id]
  287. self.segments[segment] -= 1
  288. self.compact.add(segment)
  289. segment = self.io.write_delete(id)
  290. self.segments.setdefault(segment, 0)
  291. self.compact.add(segment)
  292. except KeyError:
  293. pass
  294. segment, offset = self.io.write_put(id, data)
  295. self.segments.setdefault(segment, 0)
  296. self.segments[segment] += 1
  297. self.index[id] = segment, offset
  298. def delete(self, id, wait=True):
  299. if not self._active_txn:
  300. self.get_index(self.get_transaction_id())
  301. self._active_txn = True
  302. try:
  303. segment, offset = self.index.pop(id)
  304. self.segments[segment] -= 1
  305. self.compact.add(segment)
  306. segment = self.io.write_delete(id)
  307. self.compact.add(segment)
  308. self.segments.setdefault(segment, 0)
  309. except KeyError:
  310. raise self.DoesNotExist(self.path)
  311. def preload(self, ids):
  312. """Preload objects (only applies to remote repositories
  313. """
  314. class LoggedIO(object):
  315. header_fmt = struct.Struct('<IIB')
  316. assert header_fmt.size == 9
  317. put_header_fmt = struct.Struct('<IIB32s')
  318. assert put_header_fmt.size == 41
  319. header_no_crc_fmt = struct.Struct('<IB')
  320. assert header_no_crc_fmt.size == 5
  321. crc_fmt = struct.Struct('<I')
  322. assert crc_fmt.size == 4
  323. _commit = header_no_crc_fmt.pack(9, TAG_COMMIT)
  324. COMMIT = crc_fmt.pack(crc32(_commit)) + _commit
  325. def __init__(self, path, limit, segments_per_dir, capacity=100):
  326. self.path = path
  327. self.fds = LRUCache(capacity)
  328. self.segment = 0
  329. self.limit = limit
  330. self.segments_per_dir = segments_per_dir
  331. self.offset = 0
  332. self._write_fd = None
  333. def close(self):
  334. for segment in list(self.fds.keys()):
  335. self.fds.pop(segment).close()
  336. self.close_segment()
  337. self.fds = None # Just to make sure we're disabled
  338. def segment_iterator(self, reverse=False):
  339. for dirpath, dirs, filenames in os.walk(os.path.join(self.path, 'data')):
  340. dirs.sort(key=int, reverse=reverse)
  341. filenames = sorted((filename for filename in filenames if filename.isdigit()), key=int, reverse=reverse)
  342. for filename in filenames:
  343. yield int(filename), os.path.join(dirpath, filename)
  344. def get_segments_transaction_id(self, index_transaction_id=0):
  345. """Verify that the transaction id is consistent with the index transaction id
  346. """
  347. for segment, filename in self.segment_iterator(reverse=True):
  348. # if index_transaction_id is not None and segment < index_transaction_id:
  349. # # The index is newer than any committed transaction found
  350. # return -1
  351. if self.is_committed_segment(filename):
  352. return segment
  353. return None
  354. def cleanup(self, transaction_id):
  355. """Delete segment files left by aborted transactions
  356. """
  357. self.segment = transaction_id + 1
  358. for segment, filename in self.segment_iterator(reverse=True):
  359. if segment > transaction_id:
  360. os.unlink(filename)
  361. else:
  362. break
  363. def is_committed_segment(self, filename):
  364. """Check if segment ends with a COMMIT_TAG tag
  365. """
  366. with open(filename, 'rb') as fd:
  367. try:
  368. fd.seek(-self.header_fmt.size, os.SEEK_END)
  369. except Exception as e:
  370. # return False if segment file is empty or too small
  371. if e.errno == errno.EINVAL:
  372. return False
  373. raise e
  374. return fd.read(self.header_fmt.size) == self.COMMIT
  375. def segment_filename(self, segment):
  376. return os.path.join(self.path, 'data', str(segment // self.segments_per_dir), str(segment))
  377. def get_write_fd(self, no_new=False):
  378. if not no_new and self.offset and self.offset > self.limit:
  379. self.close_segment()
  380. if not self._write_fd:
  381. if self.segment % self.segments_per_dir == 0:
  382. dirname = os.path.join(self.path, 'data', str(self.segment // self.segments_per_dir))
  383. if not os.path.exists(dirname):
  384. os.mkdir(dirname)
  385. self._write_fd = open(self.segment_filename(self.segment), 'ab')
  386. self._write_fd.write(MAGIC)
  387. self.offset = 8
  388. return self._write_fd
  389. def get_fd(self, segment):
  390. try:
  391. return self.fds[segment]
  392. except KeyError:
  393. fd = open(self.segment_filename(segment), 'rb')
  394. self.fds[segment] = fd
  395. return fd
  396. def delete_segment(self, segment):
  397. try:
  398. os.unlink(self.segment_filename(segment))
  399. except OSError:
  400. pass
  401. def iter_objects(self, segment, lookup=None, include_data=False):
  402. fd = self.get_fd(segment)
  403. fd.seek(0)
  404. if fd.read(8) != MAGIC:
  405. raise IntegrityError('Invalid segment header')
  406. offset = 8
  407. header = fd.read(self.header_fmt.size)
  408. while header:
  409. crc, size, tag = self.header_fmt.unpack(header)
  410. if size > MAX_OBJECT_SIZE:
  411. raise IntegrityError('Invalid segment object size')
  412. rest = fd.read(size - self.header_fmt.size)
  413. if crc32(rest, crc32(memoryview(header)[4:])) & 0xffffffff != crc:
  414. raise IntegrityError('Segment checksum mismatch')
  415. if tag not in (TAG_PUT, TAG_DELETE, TAG_COMMIT):
  416. raise IntegrityError('Invalid segment entry header')
  417. key = None
  418. if tag in (TAG_PUT, TAG_DELETE):
  419. key = rest[:32]
  420. if not lookup or lookup(tag, key):
  421. if include_data:
  422. yield tag, key, rest[32:]
  423. else:
  424. yield tag, key, offset
  425. offset += size
  426. header = fd.read(self.header_fmt.size)
  427. def recover_segment(self, segment, filename):
  428. self.fds.pop(segment).close()
  429. # FIXME: save a copy of the original file
  430. with open(filename, 'rb') as fd:
  431. data = memoryview(fd.read())
  432. os.rename(filename, filename + '.beforerecover')
  433. print('attempting to recover ' + filename, file=sys.stderr)
  434. with open(filename, 'wb') as fd:
  435. fd.write(MAGIC)
  436. while len(data) >= self.header_fmt.size:
  437. crc, size, tag = self.header_fmt.unpack(data[:self.header_fmt.size])
  438. if size > len(data):
  439. data = data[1:]
  440. continue
  441. if crc32(data[4:size]) & 0xffffffff != crc:
  442. data = data[1:]
  443. continue
  444. fd.write(data[:size])
  445. data = data[size:]
  446. def read(self, segment, offset, id):
  447. if segment == self.segment and self._write_fd:
  448. self._write_fd.flush()
  449. fd = self.get_fd(segment)
  450. fd.seek(offset)
  451. header = fd.read(self.put_header_fmt.size)
  452. crc, size, tag, key = self.put_header_fmt.unpack(header)
  453. if size > MAX_OBJECT_SIZE:
  454. raise IntegrityError('Invalid segment object size')
  455. data = fd.read(size - self.put_header_fmt.size)
  456. if crc32(data, crc32(memoryview(header)[4:])) & 0xffffffff != crc:
  457. raise IntegrityError('Segment checksum mismatch')
  458. if tag != TAG_PUT or id != key:
  459. raise IntegrityError('Invalid segment entry header')
  460. return data
  461. def write_put(self, id, data):
  462. size = len(data) + self.put_header_fmt.size
  463. fd = self.get_write_fd()
  464. offset = self.offset
  465. header = self.header_no_crc_fmt.pack(size, TAG_PUT)
  466. crc = self.crc_fmt.pack(crc32(data, crc32(id, crc32(header))) & 0xffffffff)
  467. fd.write(b''.join((crc, header, id, data)))
  468. self.offset += size
  469. return self.segment, offset
  470. def write_delete(self, id):
  471. fd = self.get_write_fd()
  472. header = self.header_no_crc_fmt.pack(self.put_header_fmt.size, TAG_DELETE)
  473. crc = self.crc_fmt.pack(crc32(id, crc32(header)) & 0xffffffff)
  474. fd.write(b''.join((crc, header, id)))
  475. self.offset += self.put_header_fmt.size
  476. return self.segment
  477. def write_commit(self):
  478. fd = self.get_write_fd(no_new=True)
  479. header = self.header_no_crc_fmt.pack(self.header_fmt.size, TAG_COMMIT)
  480. crc = self.crc_fmt.pack(crc32(header) & 0xffffffff)
  481. fd.write(b''.join((crc, header)))
  482. self.close_segment()
  483. def close_segment(self):
  484. if self._write_fd:
  485. self.segment += 1
  486. self.offset = 0
  487. os.fsync(self._write_fd)
  488. self._write_fd.close()
  489. self._write_fd = None