store.py 18 KB

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