1
+ import time
2
+ import asyncio
3
+
4
+ import richka
5
+
6
+ import aiohttp
7
+
8
+ async def __download_range (session : aiohttp .ClientSession , url : str , start : int , end : int , destination : str ) -> None :
9
+ richka .logger .info (f'Downloading part { start } -{ end } of { url } to { destination } .' )
10
+
11
+ headers = {** richka .HEADERS , ** {'range' : f'bytes={ start } -{ end } ' }}
12
+
13
+ async with session .get (url , headers = headers ) as response :
14
+ content = await response .read ()
15
+ with open (destination , 'r+b' ) as f :
16
+ f .seek (start )
17
+ f .write (content )
18
+
19
+ richka .logger .info (f'Downloaded part { start } -{ end } of { destination } .' )
20
+
21
+ async def __download_single (session : aiohttp .ClientSession , url : str , destination : str ) -> None :
22
+ richka .logger .info (f'Downloading { url } to { destination } .' )
23
+
24
+ async with session .get (url , headers = richka .HEADERS ) as response :
25
+ content = await response .read ()
26
+ with open (destination , 'r+b' ) as f :
27
+ f .write (content )
28
+
29
+ richka .logger .info (f'Downloaded { url } to { destination } .' )
30
+
31
+ async def download (url : str , destination : str ) -> float :
32
+ async with aiohttp .ClientSession () as session :
33
+ # Get file size
34
+ async with session .head (url ) as response :
35
+ file_size = int (response .headers .get ('Content-Length' , 0 ))
36
+
37
+ if not file_size or file_size / pow (1024 , 2 ) <= 10 :
38
+ if not file_size :
39
+ richka .logger .info (f'Failed to get file size, directly downloading { url } .' )
40
+ else :
41
+ richka .logger .info (f"Downloading { url } ({ file_size } ) to { destination } with signle mode." )
42
+
43
+ # Create an empty file
44
+ with open (destination , 'wb' ) as f :
45
+ f .truncate (file_size )
46
+
47
+ # Start task
48
+ start_time = time .time ()
49
+ await __download_single (session , url , destination )
50
+ end_time = time .time ()
51
+ return end_time - start_time
52
+
53
+ richka .logger .info (f'Downloading { url } ({ file_size } ) to { destination } with slicing mode.' )
54
+
55
+ # Calc slice size
56
+ part_size = file_size // richka .COROUTINE_LIMIT
57
+
58
+ # Create an empty file
59
+ with open (destination , 'wb' ) as f :
60
+ f .truncate (file_size )
61
+
62
+ # Create coroutine tasks
63
+ tasks = []
64
+ for i in range (richka .COROUTINE_LIMIT ):
65
+ start = i * part_size
66
+ end = (start + part_size - 1 ) if i < richka .COROUTINE_LIMIT - 1 else (file_size - 1 )
67
+ task = __download_range (session , url , start , end , destination )
68
+ tasks .append (task )
69
+
70
+ # Start all task
71
+ start_time = time .time ()
72
+ await asyncio .gather (* tasks )
73
+ end_time = time .time ()
74
+ return end_time - start_time
0 commit comments