mirror of
https://github.com/tahoe-lafs/tahoe-lafs.git
synced 2025-01-05 20:54:09 +00:00
272 lines
11 KiB
Python
272 lines
11 KiB
Python
# 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<len(l)):
|
|
# This loop body executes once per chunk.
|
|
a = l[i]
|
|
i += 1
|
|
del a[:]
|
|
try:
|
|
a.fromfile(inf, chunksize)
|
|
except EOFError:
|
|
indatasize = i*chunksize + len(a)
|
|
|
|
# padding
|
|
a.fromstring("\x00" * (chunksize-len(a)))
|
|
while (i<len(l)):
|
|
a = l[i]
|
|
a[:] = ZEROES
|
|
i += 1
|
|
|
|
# print "about to encode()... len(l[0]): %s, l[0]: %s" % (len(l[0]), type(l[0]),),
|
|
res = enc.encode(l)
|
|
# print "...finished to encode()"
|
|
cb(res, indatasize)
|
|
|
|
def encode_file_stringy(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 part
|
|
of the input shares that is unused is filled with zeroes before encoding.
|
|
|
|
@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)
|
|
indatasize = k*chunksize # will be reset to shorter upon EOF
|
|
while indatasize == k*chunksize:
|
|
# This loop body executes once per segment.
|
|
i = 0
|
|
l = []
|
|
ZEROES = '\x00'*chunksize
|
|
while i<k:
|
|
# This loop body executes once per chunk.
|
|
i += 1
|
|
l.append(inf.read(chunksize))
|
|
if len(l[-1]) < chunksize:
|
|
indatasize = i*chunksize + len(l[-1])
|
|
|
|
# padding
|
|
l[-1] = l[-1] + "\x00" * (chunksize-len(l[-1]))
|
|
while i<k:
|
|
l.append(ZEROES)
|
|
i += 1
|
|
|
|
# print "about to encode()... len(l[0]): %s, l[0]: %s" % (len(l[0]), type(l[0]),),
|
|
res = enc.encode(l)
|
|
# print "...finished to encode()"
|
|
cb(res, indatasize)
|
|
|
|
def encode_file_not_really(inf, cb, k, m, chunksize=4096):
|
|
"""
|
|
Read in the contents of inf, and call cb with the results.
|
|
|
|
@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<len(l)):
|
|
# This loop body executes once per chunk.
|
|
a = l[i]
|
|
i += 1
|
|
del a[:]
|
|
try:
|
|
a.fromfile(inf, chunksize)
|
|
except EOFError:
|
|
indatasize = i*chunksize + len(a)
|
|
|
|
# padding
|
|
a.fromstring("\x00" * (chunksize-len(a)))
|
|
while (i<len(l)):
|
|
a[:] = ZEROES
|
|
i += 1
|
|
|
|
# print "about to encode()... len(l[0]): %s, l[0]: %s" % (len(l[0]), type(l[0]),),
|
|
# res = enc.encode(l)
|
|
# print "...finished to encode()"
|
|
cb(l, indatasize)
|
|
|
|
def encode_file_stringy_easyfec(inf, cb, k, m, chunksize=4096):
|
|
"""
|
|
Read in the contents of inf, encode, and call cb with the results.
|
|
|
|
First, chunksize*k bytes will be read from inf, then 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 that is unused is filled with zeroes before
|
|
encoding.
|
|
|
|
@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 = easyfec.Encoder(k, m)
|
|
|
|
indatasize = k*chunksize # will be reset to shorter upon EOF
|
|
indata = inf.read(indatasize)
|
|
while indata:
|
|
res = enc.encode(indata)
|
|
cb(res, indatasize)
|
|
indata = inf.read(indatasize)
|
|
|