store.py 15 KB

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