tahoe-lafs/src/allmydata/storageserver.py

164 lines
6.1 KiB
Python
Raw Normal View History

import os, re
from foolscap import Referenceable
from twisted.application import service
from zope.interface import implements
from allmydata.interfaces import RIStorageServer, RIBucketWriter, \
RIBucketReader
2007-03-30 03:19:52 +00:00
from allmydata import interfaces
from allmydata.util import bencode, fileutil, idlib
2007-04-18 17:43:33 +00:00
from allmydata.util.assertutil import precondition
2007-03-30 03:19:52 +00:00
# store/
# store/incoming # temp dirs named $STORAGEINDEX/$SHARENUM which will be moved to store/$STORAGEINDEX/$SHARENUM on success
# store/$STORAGEINDEX
# store/$STORAGEINDEX/$SHARENUM
# store/$STORAGEINDEX/$SHARENUM/blocksize
# store/$STORAGEINDEX/$SHARENUM/data
# store/$STORAGEINDEX/$SHARENUM/blockhashes
# store/$STORAGEINDEX/$SHARENUM/sharehashtree
2007-03-30 03:19:52 +00:00
# $SHARENUM matches this regex:
NUM_RE=re.compile("[0-9]*")
2007-03-30 03:19:52 +00:00
class BucketWriter(Referenceable):
implements(RIBucketWriter)
def __init__(self, incominghome, finalhome, blocksize):
self.incominghome = incominghome
2007-03-30 03:19:52 +00:00
self.finalhome = finalhome
self.blocksize = blocksize
self.closed = False
self._next_segnum = 0
fileutil.make_dirs(incominghome)
2007-03-30 03:19:52 +00:00
self._write_file('blocksize', str(blocksize))
def _write_file(self, fname, data):
open(os.path.join(self.incominghome, fname), 'wb').write(data)
2007-03-30 03:19:52 +00:00
def remote_put_block(self, segmentnum, data):
precondition(not self.closed)
# all blocks but the last will be of size self.blocksize, however the
# last one may be short, and we don't know the total number of
# segments so we can't tell which is which.
assert len(data) <= self.blocksize
assert segmentnum == self._next_segnum # must write in sequence
self._next_segnum = segmentnum + 1
f = fileutil.open_or_create(os.path.join(self.incominghome, 'data'))
2007-03-30 03:19:52 +00:00
f.seek(self.blocksize*segmentnum)
f.write(data)
def remote_put_plaintext_hashes(self, hashes):
precondition(not self.closed)
# TODO: verify the length of blockhashes.
# TODO: tighten foolscap schema to require exactly 32 bytes.
self._write_file('plaintext_hashes', ''.join(hashes))
def remote_put_crypttext_hashes(self, hashes):
precondition(not self.closed)
# TODO: verify the length of blockhashes.
# TODO: tighten foolscap schema to require exactly 32 bytes.
self._write_file('crypttext_hashes', ''.join(hashes))
2007-03-30 03:19:52 +00:00
def remote_put_block_hashes(self, blockhashes):
precondition(not self.closed)
# TODO: verify the length of blockhashes.
# TODO: tighten foolscap schema to require exactly 32 bytes.
self._write_file('blockhashes', ''.join(blockhashes))
def remote_put_share_hashes(self, sharehashes):
precondition(not self.closed)
self._write_file('sharehashes', bencode.bencode(sharehashes))
2007-03-30 03:19:52 +00:00
def remote_put_thingA(self, data):
precondition(not self.closed)
self._write_file('thingA', data)
def remote_close(self):
2007-03-30 03:19:52 +00:00
precondition(not self.closed)
# TODO assert or check the completeness and consistency of the data that has been written
fileutil.make_dirs(os.path.dirname(self.finalhome))
fileutil.rename(self.incominghome, self.finalhome)
try:
os.rmdir(os.path.dirname(self.incominghome))
except OSError:
# Perhaps the directory wasn't empty. In any case, ignore the error.
pass
2007-03-30 03:19:52 +00:00
self.closed = True
def str2l(s):
""" split string (pulled from storage) into a list of blockids """
return [ s[i:i+interfaces.HASH_SIZE] for i in range(0, len(s), interfaces.HASH_SIZE) ]
class BucketReader(Referenceable):
implements(RIBucketReader)
2007-03-30 03:19:52 +00:00
def __init__(self, home):
self.home = home
self.blocksize = int(self._read_file('blocksize'))
def _read_file(self, fname):
return open(os.path.join(self.home, fname), 'rb').read()
def remote_get_block(self, blocknum):
f = open(os.path.join(self.home, 'data'), 'rb')
f.seek(self.blocksize * blocknum)
return f.read(self.blocksize) # this might be short for the last block
2007-03-30 03:19:52 +00:00
def remote_get_block_hashes(self):
return str2l(self._read_file('blockhashes'))
def remote_get_share_hashes(self):
hashes = bencode.bdecode(self._read_file('sharehashes'))
# tuples come through bdecode(bencode()) as lists, which violates the
# schema
return [tuple(i) for i in hashes]
def remote_get_thingA(self):
return self._read_file('thingA')
class StorageServer(service.MultiService, Referenceable):
implements(RIStorageServer)
name = 'storageserver'
2007-03-30 03:19:52 +00:00
def __init__(self, storedir):
fileutil.make_dirs(storedir)
self.storedir = storedir
self.incomingdir = os.path.join(storedir, 'incoming')
self._clean_incomplete()
fileutil.make_dirs(self.incomingdir)
2007-03-30 03:19:52 +00:00
service.MultiService.__init__(self)
def _clean_incomplete(self):
fileutil.rm_dir(self.incomingdir)
2007-03-30 03:19:52 +00:00
def remote_allocate_buckets(self, storage_index, sharenums, sharesize,
2007-03-30 03:19:52 +00:00
blocksize, canary):
alreadygot = set()
bucketwriters = {} # k: shnum, v: BucketWriter
for shnum in sharenums:
incominghome = os.path.join(self.incomingdir, idlib.b2a(storage_index), "%d"%shnum)
finalhome = os.path.join(self.storedir, idlib.b2a(storage_index), "%d"%shnum)
if os.path.exists(incominghome) or os.path.exists(finalhome):
alreadygot.add(shnum)
else:
bucketwriters[shnum] = BucketWriter(incominghome, finalhome, blocksize)
2007-03-30 03:19:52 +00:00
return alreadygot, bucketwriters
def remote_get_buckets(self, storage_index):
2007-03-30 03:19:52 +00:00
bucketreaders = {} # k: sharenum, v: BucketReader
storagedir = os.path.join(self.storedir, idlib.b2a(storage_index))
try:
for f in os.listdir(storagedir):
if NUM_RE.match(f):
bucketreaders[int(f)] = BucketReader(os.path.join(storagedir, f))
except OSError:
# Commonly caused by there being no buckets at all.
pass
2007-03-30 03:19:52 +00:00
return bucketreaders