[Author Prev][Author Next][Thread Prev][Thread Next][Author Index][Thread Index]
[or-cvs] [thandy/master] Add ability to create .torrent metadata when making a package
Author: Sebastian Hahn <sebastian@xxxxxxxxxxxxxx>
Date: Tue, 11 Aug 2009 13:43:06 +0200
Subject: Add ability to create .torrent metadata when making a package
Commit: aa0d32f4b675e155e6e004604bf8b6ee4e607873
Also update the spec with the information where the .torrent
metadata file will be stored.
---
doc/HOWTO | 3 +-
lib/thandy/SignerCLI.py | 14 ++++++++
lib/thandy/bt_compat.py | 76 +++++++++++++++++++++++++++++++++++++++++++++
lib/thandy/master_keys.py | 3 ++
specs/thandy-spec.txt | 6 +++
5 files changed, 101 insertions(+), 1 deletions(-)
create mode 100644 lib/thandy/bt_compat.py
diff --git a/doc/HOWTO b/doc/HOWTO
index a5937f3..4ec6068 100644
--- a/doc/HOWTO
+++ b/doc/HOWTO
@@ -217,7 +217,8 @@ and notes:
* For the actual data files (i.e. the packages themselves), for now you
just copy them into the repository manually. Later there will be some
- version of thandy-server insert that handles them too.
+ version of thandy-server insert that handles them too. The same goes
+ for the BitTorrent metadata files.
.GENERATING TIMESTAMPS.
diff --git a/lib/thandy/SignerCLI.py b/lib/thandy/SignerCLI.py
index d0feef6..06536aa 100644
--- a/lib/thandy/SignerCLI.py
+++ b/lib/thandy/SignerCLI.py
@@ -7,6 +7,7 @@ import sys
import thandy.keys
import thandy.formats
import thandy.util
+import thandy.bt_compat
json = thandy.util.importJSON()
@@ -78,6 +79,19 @@ def makepackage(args):
f = open(location, 'w')
json.dump(signable, f, indent=1, sort_keys=True)
f.close()
+ btcomp = thandy.bt_compat.BtCompat()
+ # If we have bittorrent support, we always create the appropriate metadata.
+ thandy.bt_compat.BtCompat.setUseBt(True)
+ if not thandy.bt_compat.BtCompat.shouldUseBt():
+ print "Not generating BitTorrent metadata files"
+ return
+ print "Generating BitTorrent metadata file"
+ metaFile = btcomp.makeMetaFile(dataFile)
+ mfLocation = os.path.split(btcomp.getBtMetadataLocation(location, dataFile))[-1]
+ print "Writing BitTorrent Metadata file to %s"%mfLocation
+ f = open(mfLocation, 'w')
+ f.write(metaFile)
+ f.close()
def makebundle(args):
options, args = getopt.getopt(args, "", "keyid=")
diff --git a/lib/thandy/bt_compat.py b/lib/thandy/bt_compat.py
new file mode 100644
index 0000000..43cecad
--- /dev/null
+++ b/lib/thandy/bt_compat.py
@@ -0,0 +1,76 @@
+# Copyright 2008 The Tor Project, Inc. See LICENSE for licensing information.
+
+import os.path
+import time
+
+import thandy.master_keys
+
+no_bt = None
+try:
+ import BitTorrent.bencode
+ import BitTorrent.btformats
+ import BitTorrent.download
+except ImportError:
+ no_bt = True
+from sha import sha # XXX Use PyCrypto here?
+
+
+class BtCompat:
+ """Interface for different bittorrent implementations"""
+
+ usingBt = False
+
+ def __init__(self):
+ self.tUrl = thandy.master_keys.DEFAULT_TRACKER
+ if not no_bt:
+ assert(self.tUrl is not None and self.tUrl != "")
+ self.pieceLength = 2 ** 18 # Piece length of 262144 bytes
+
+ # XXX Do we need to be thread-safe here and below?
+ @staticmethod
+ def shouldUseBt():
+ return BtCompat.usingBt
+
+ @staticmethod
+ def setUseBt(useBt):
+ if no_bt:
+ return
+ BtCompat.usingBt = useBt
+
+ @staticmethod
+ def getBtMetadataLocation(packagepath, filepath, pathprefix=""):
+ """Given a path for the package, the path for a file of that
+ package, and an optional prefix, return the path for the
+ .torrent metadata file. Always return Unix-like paths, to
+ ensure compatibility with fetching the path from a
+ webserver.
+ """
+ return (os.path.join(pathprefix, os.path.dirname(packagepath),
+ os.path.basename(filepath)) + ".torrent"
+ ).replace("\\", "/")
+
+ def makeMetaFile(self, file):
+ """Given a path to a file, create the contents of a .torrent
+ metadata file and return them.
+ """
+ size = os.path.getsize(file)
+ filename = os.path.basename(file)
+ pieces = []
+ p = 0
+ h = open(file, 'rb')
+ while p < size:
+ x = h.read(min(self.pieceLength, size - p))
+ pieces.append(sha(x).digest())
+ p += self.pieceLength
+ if p > size:
+ p = size
+ h.close()
+ info = {'pieces': ''.join(pieces),
+ 'piece length': self.pieceLength, 'length': size,
+ 'name': filename}
+ # Check we didn't screw up with the info
+ BitTorrent.btformats.check_info(info)
+ data = {'info': info, 'announce': self.tUrl,
+ 'creation date': long(time.time())}
+ return BitTorrent.bencode.bencode(data)
+
diff --git a/lib/thandy/master_keys.py b/lib/thandy/master_keys.py
index 05c50a8..be13f52 100644
--- a/lib/thandy/master_keys.py
+++ b/lib/thandy/master_keys.py
@@ -17,3 +17,6 @@ DEFAULT_MIRRORLIST = {
}
],
}
+
+DEFAULT_TRACKER = NONE
+
diff --git a/specs/thandy-spec.txt b/specs/thandy-spec.txt
index daadd05..eae908b 100644
--- a/specs/thandy-spec.txt
+++ b/specs/thandy-spec.txt
@@ -248,6 +248,12 @@
file that makes up a package, its hash, and what procedure
is used to install it.
+ /pkginfo/packagename/os-arch/version/(some filename).torrent
+
+ The .torrent metadata file used to download that file. The
+ file name is exactly the same as specified in the package
+ file.
+
/packages/packagename/os-arch/version/(some filename)
The actual package file. Its naming convention will depend
--
1.5.6.5