web-dev-qa-db-ja.com

トレントファイルからSHA1ハッシュを抽出します

私はこれに対する答えを探し回っていましたが、あなたのためにそれを行うソフトウェアを見つけることができるだけのようです。 Pythonでこれを行う方法を知っている人はいますか?

22
user271528

ダウンロードされたファイルのハッシュを。torrentファイルのハッシュと照合するpythonコードの一部を作成しました。チェックしたい場合破損のダウンロードは、これが役立つ場合があります。

これを使用するには、 bencode package が必要です。 Bencodeは、.torrentファイルで使用されるシリアル化形式です。リスト、辞書、文字列、数値をJSONのようにマーシャリングできます。

コードは、_info['pieces']_文字列に含まれるハッシュを取ります。

_torrent_file = open(sys.argv[1], "rb")
metainfo = bencode.bdecode(torrent_file.read())
info = metainfo['info']
pieces = StringIO.StringIO(info['pieces'])
_

その文字列には、20バイトのハッシュが連続して含まれています(各ピースに1つ)。次に、これらのハッシュは、ディスク上のファイルの断片のハッシュと比較されます。

このコードの唯一の複雑な部分は、単一のトレントpieceが複数のファイルにまたがることができるため、複数ファイルのトレントを処理することです(内部的には、BitTorrentは複数ファイルのダウンロードを単一の連続ファイルとして扱います)。私はそれを抽象化するためにジェネレーター関数pieces_generator()を使用しています。

これをより詳細に理解するには、 BitTorrent仕様 を読むことをお勧めします。

以下の完全なコード:

_import sys, os, hashlib, StringIO, bencode

def pieces_generator(info):
    """Yield pieces from download file(s)."""
    piece_length = info['piece length']
    if 'files' in info: # yield pieces from a multi-file torrent
        piece = ""
        for file_info in info['files']:
            path = os.sep.join([info['name']] + file_info['path'])
            print path
            sfile = open(path.decode('UTF-8'), "rb")
            while True:
                piece += sfile.read(piece_length-len(piece))
                if len(piece) != piece_length:
                    sfile.close()
                    break
                yield piece
                piece = ""
        if piece != "":
            yield piece
    else: # yield pieces from a single file torrent
        path = info['name']
        print path
        sfile = open(path.decode('UTF-8'), "rb")
        while True:
            piece = sfile.read(piece_length)
            if not piece:
                sfile.close()
                return
            yield piece

def corruption_failure():
    """Display error message and exit"""
    print("download corrupted")
    exit(1)

def main():
    # Open torrent file
    torrent_file = open(sys.argv[1], "rb")
    metainfo = bencode.bdecode(torrent_file.read())
    info = metainfo['info']
    pieces = StringIO.StringIO(info['pieces'])
    # Iterate through pieces
    for piece in pieces_generator(info):
        # Compare piece hash with expected hash
        piece_hash = hashlib.sha1(piece).digest()
        if (piece_hash != pieces.read(20)):
            corruption_failure()
    # ensure we've read all pieces 
    if pieces.read():
        corruption_failure()

if __name__ == "__main__":
    main()
_
29
Alex Jasmin

トレントファイルからHASH値を抽出した方法は次のとおりです。

#!/usr/bin/python

import sys, os, hashlib, StringIO
import bencode



def main():
    # Open torrent file
    torrent_file = open(sys.argv[1], "rb")
    metainfo = bencode.bdecode(torrent_file.read())
    info = metainfo['info']
    print hashlib.sha1(bencode.bencode(info)).hexdigest()    

if __name__ == "__main__":
    main()

これは、コマンドを実行するのと同じです。

transmissioncli -i test.torrent 2>/dev/null | grep "^hash:" | awk '{print $2}'

それが役に立てば幸い :)

17
grundic