# pyfec -- fast forward error correction library with Python interface # # Copyright (C) 2007 Allmydata, Inc. # Author: Zooko Wilcox-O'Hearn # mailto:zooko@zooko.com # # This file is part of pyfec. # # This program is free software; you can redistribute it and/or modify it # under the terms of the GNU General Public License as published by the Free # Software Foundation; either version 2 of the License, or (at your option) # any later version. This program also comes with the added permission that, # in the case that you are obligated to release a derived work under this # licence (as per section 2.b of the GPL), you may delay the fulfillment of # this obligation for up to 12 months. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program; if not, write to the Free Software # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. import easyfec, fec import array, random def encode_to_files_easyfec(inf, prefix, k, m): """ Encode inf, writing the shares to a file named $prefix+$sharenum. """ l = [ open(prefix+str(sharenum), "wb") for sharenum in range(m) ] def cb(blocks, length): assert len(blocks) == len(l) for i in range(len(blocks)): l[i].write(blocks[i]) encode_file_stringy_easyfec(inf, cb, k, m, chunksize=4096) def encode_to_files_stringy(inf, prefix, k, m): """ Encode inf, writing the shares to a file named named $prefix+$sharenum. """ l = [ open(prefix+str(sharenum), "wb") for sharenum in range(m) ] def cb(blocks, length): assert len(blocks) == len(l) for i in range(len(blocks)): l[i].write(blocks[i]) encode_file_stringy(inf, cb, k, m, chunksize=4096) def encode_to_files(inf, prefix, k, m): """ Encode inf, writing the shares to named $prefix+$sharenum. """ l = [ open(prefix+str(sharenum), "wb") for sharenum in range(m) ] def cb(blocks, length): assert len(blocks) == len(l) for i in range(len(blocks)): l[i].write(blocks[i]) encode_file(inf, cb, k, m, chunksize=4096) def decode_from_files(outf, filesize, prefix, k, m): """ Decode from the first k files in the current directory whose names begin with prefix, writing the results to outf. """ import os infs = [] sharenums = [] listd = os.listdir(".") random.shuffle(listd) for f in listd: if f.startswith(prefix): infs.append(open(f, "rb")) sharenums.append(int(f[len(prefix):])) if len(infs) == k: break CHUNKSIZE = 4096 dec = fec.Decoder(k, m) while True: x = [ inf.read(CHUNKSIZE) for inf in infs ] decblocks = dec.decode(x, sharenums) for decblock in decblocks: if len(decblock) == 0: raise "error -- probably share was too short -- was it stored in a file which got truncated? chunksizes: %s" % ([len(chunk) for chunk in x],) if filesize >= len(decblock): outf.write(decblock) filesize -= len(decblock) # print "filesize is now %s after subtracting %s" % (filesize, len(decblock),) else: outf.write(decblock[:filesize]) return def encode_file(inf, cb, k, m, chunksize=4096): """ Read in the contents of inf, encode, and call cb with the results. First, k "input blocks" will be read from inf, each input block being of size chunksize. Then these k blocks will be encoded into m "result blocks". Then cb will be invoked, passing a list of the m result blocks as its first argument, and the length of the encoded data as its second argument. (The length of the encoded data is always equal to k*chunksize, until the last iteration, when the end of the file has been reached and less than k*chunksize bytes could be read from the file.) This procedure is iterated until the end of the file is reached, in which case the space of the input blocks that is unused is filled with zeroes before encoding. Note that the sequence passed in calls to cb() contains mutable array objects in its first k elements whose contents will be overwritten when the next segment is read from the input file. Therefore the implementation of cb() has to either be finished with those first k arrays before returning, or if it wants to keep the contents of those arrays for subsequent use after it has returned then it must make a copy of them to keep. @param inf the file object from which to read the data @param cb the callback to be invoked with the results @param k the number of shares required to reconstruct the file @param m the total number of shares created @param chunksize how much data to read from inf for each of the k input blocks """ enc = fec.Encoder(k, m) l = tuple([ array.array('c') for i in range(k) ]) indatasize = k*chunksize # will be reset to shorter upon EOF ZEROES=array.array('c', ['\x00'])*chunksize while indatasize == k*chunksize: # This loop body executes once per segment. i = 0 while (i