forked from gallexis/PyTorrent
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpiece.py
122 lines (91 loc) · 3.72 KB
/
piece.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
__author__ = 'alexisgallepe'
import hashlib
import math
import time
import logging
from pubsub import pub
from block import Block, BLOCK_SIZE, State
class Piece(object):
def __init__(self, piece_index: int, piece_size: int, piece_hash: str):
self.piece_index: int = piece_index
self.piece_size: int = piece_size
self.piece_hash: str = piece_hash
self.is_full: bool = False
self.files = []
self.raw_data: bytes = b''
self.number_of_blocks: int = int(math.ceil(float(piece_size) / BLOCK_SIZE))
self.blocks: list[Block] = []
self._init_blocks()
def update_block_status(self): # if block is pending for too long : set it free
for i, block in enumerate(self.blocks):
if block.state == State.PENDING and (time.time() - block.last_seen) > 5:
self.blocks[i] = Block()
def set_block(self, offset, data):
index = int(offset / BLOCK_SIZE)
if not self.is_full and not self.blocks[index].state == State.FULL:
self.blocks[index].data = data
self.blocks[index].state = State.FULL
def get_block(self, block_offset, block_length):
return self.raw_data[block_offset:block_length]
def get_empty_block(self):
if self.is_full:
return None
for block_index, block in enumerate(self.blocks):
if block.state == State.FREE:
self.blocks[block_index].state = State.PENDING
self.blocks[block_index].last_seen = time.time()
return self.piece_index, block_index * BLOCK_SIZE, block.block_size
return None
def are_all_blocks_full(self):
for block in self.blocks:
if block.state == State.FREE or block.state == State.PENDING:
return False
return True
def set_to_full(self):
data = self._merge_blocks()
if not self._valid_blocks(data):
self._init_blocks()
return False
self.is_full = True
self.raw_data = data
self._write_piece_on_disk()
pub.sendMessage('PiecesManager.PieceCompleted', piece_index=self.piece_index)
return True
def _init_blocks(self):
self.blocks = []
if self.number_of_blocks > 1:
for i in range(self.number_of_blocks):
self.blocks.append(Block())
# Last block of last piece, the special block
if (self.piece_size % BLOCK_SIZE) > 0:
self.blocks[self.number_of_blocks - 1].block_size = self.piece_size % BLOCK_SIZE
else:
self.blocks.append(Block(block_size=int(self.piece_size)))
def _write_piece_on_disk(self):
for file in self.files:
path_file = file["path"]
file_offset = file["fileOffset"]
piece_offset = file["pieceOffset"]
length = file["length"]
try:
f = open(path_file, 'r+b') # Already existing file
except IOError:
f = open(path_file, 'wb') # New file
except Exception:
logging.exception("Can't write to file")
return
f.seek(file_offset)
f.write(self.raw_data[piece_offset:piece_offset + length])
f.close()
def _merge_blocks(self):
buf = b''
for block in self.blocks:
buf += block.data
return buf
def _valid_blocks(self, piece_raw_data):
hashed_piece_raw_data = hashlib.sha1(piece_raw_data).digest()
if hashed_piece_raw_data == self.piece_hash:
return True
logging.warning("Error Piece Hash")
logging.debug("{} : {}".format(hashed_piece_raw_data, self.piece_hash))
return False