repository.py 23 KB

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