repository.py 25 KB

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