2015-07-26 01:40:07 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
2015-09-23 18:13:00 +02:00
|
|
|
import libtorrent as lt
|
2015-07-26 01:40:07 +02:00
|
|
|
|
|
|
|
import unittest
|
2016-01-04 05:34:57 +01:00
|
|
|
import time
|
|
|
|
import os
|
|
|
|
import shutil
|
2015-10-10 03:21:04 +02:00
|
|
|
import binascii
|
2015-07-26 01:40:07 +02:00
|
|
|
|
2016-07-27 06:15:26 +02:00
|
|
|
|
|
|
|
class test_create_torrent(unittest.TestCase):
|
|
|
|
|
|
|
|
def test_from_torrent_info(self):
|
|
|
|
ti = lt.torrent_info('unordered.torrent')
|
|
|
|
ct = lt.create_torrent(ti)
|
|
|
|
entry = ct.generate()
|
|
|
|
content = lt.bencode(entry).strip()
|
2016-08-10 13:26:02 +02:00
|
|
|
with open('unordered.torrent', 'rb') as f:
|
|
|
|
file_content = bytearray(f.read().strip())
|
|
|
|
print(content)
|
|
|
|
print(file_content)
|
|
|
|
print(entry)
|
2016-07-27 06:15:26 +02:00
|
|
|
self.assertEqual(content, file_content)
|
|
|
|
|
2016-03-31 05:35:47 +02:00
|
|
|
class test_torrent_handle(unittest.TestCase):
|
|
|
|
|
|
|
|
def test_torrent_handle(self):
|
2016-08-21 18:15:19 +02:00
|
|
|
ses = lt.session({'alert_mask': lt.alert.category_t.all_categories, 'enable_dht': False})
|
2016-03-31 05:35:47 +02:00
|
|
|
ti = lt.torrent_info('url_seed_multi.torrent');
|
|
|
|
h = ses.add_torrent({'ti': ti, 'save_path': os.getcwd()})
|
|
|
|
|
|
|
|
h.prioritize_files([0,1])
|
|
|
|
self.assertEqual(h.file_priorities(), [0,1])
|
|
|
|
|
|
|
|
h.prioritize_pieces([0])
|
|
|
|
self.assertEqual(h.piece_priorities(), [0])
|
2015-07-26 01:40:07 +02:00
|
|
|
|
2016-04-02 21:52:37 +02:00
|
|
|
# also test the overload that takes a list of piece->priority mappings
|
|
|
|
h.prioritize_pieces([(0, 1)])
|
|
|
|
self.assertEqual(h.piece_priorities(), [1])
|
|
|
|
|
2016-10-16 03:29:01 +02:00
|
|
|
def test_scrape(self):
|
|
|
|
ses = lt.session({'alert_mask': lt.alert.category_t.all_categories, 'enable_dht': False})
|
|
|
|
ti = lt.torrent_info('url_seed_multi.torrent');
|
|
|
|
h = ses.add_torrent({'ti': ti, 'save_path': os.getcwd()})
|
|
|
|
# this is just to make sure this function can be called like this
|
|
|
|
# from python
|
|
|
|
h.scrape_tracker()
|
|
|
|
|
2015-07-26 01:40:07 +02:00
|
|
|
class test_torrent_info(unittest.TestCase):
|
|
|
|
|
|
|
|
def test_bencoded_constructor(self):
|
|
|
|
info = lt.torrent_info({ 'info': {'name': 'test_torrent', 'length': 1234,
|
|
|
|
'piece length': 16 * 1024,
|
|
|
|
'pieces': 'aaaaaaaaaaaaaaaaaaaa'}})
|
|
|
|
|
|
|
|
self.assertEqual(info.num_files(), 1)
|
|
|
|
|
|
|
|
f = info.files()
|
|
|
|
self.assertEqual(f.file_path(0), 'test_torrent')
|
|
|
|
self.assertEqual(f.file_size(0), 1234)
|
|
|
|
self.assertEqual(info.total_size(), 1234)
|
|
|
|
|
2016-03-31 06:19:23 +02:00
|
|
|
def test_metadata(self):
|
|
|
|
ti = lt.torrent_info('base.torrent');
|
|
|
|
|
|
|
|
self.assertTrue(len(ti.metadata()) != 0)
|
|
|
|
self.assertTrue(len(ti.hash_for_piece(0)) != 0)
|
|
|
|
|
2016-04-07 01:05:08 +02:00
|
|
|
def test_iterable_files(self):
|
2016-04-07 08:23:21 +02:00
|
|
|
|
|
|
|
# this detects whether libtorrent was built with deprecated APIs
|
|
|
|
# the file_strage object is only iterable for backwards compatibility
|
|
|
|
if not hasattr(lt, 'version'): return
|
|
|
|
|
2016-08-21 18:15:19 +02:00
|
|
|
ses = lt.session({'alert_mask': lt.alert.category_t.all_categories, 'enable_dht': False})
|
2016-04-07 01:05:08 +02:00
|
|
|
ti = lt.torrent_info('url_seed_multi.torrent');
|
|
|
|
files = ti.files()
|
|
|
|
|
|
|
|
idx = 0
|
|
|
|
expected = ['bar.txt', 'var.txt']
|
|
|
|
for f in files:
|
2016-11-03 06:01:48 +01:00
|
|
|
print(f.path)
|
2016-04-07 01:05:08 +02:00
|
|
|
|
|
|
|
self.assertEqual(os.path.split(f.path)[1], expected[idx])
|
2016-07-05 17:25:31 +02:00
|
|
|
self.assertEqual(os.path.split(os.path.split(f.path)[0]), ('temp', 'foo'))
|
2016-04-07 01:05:08 +02:00
|
|
|
idx += 1
|
|
|
|
|
2016-01-04 05:34:57 +01:00
|
|
|
class test_alerts(unittest.TestCase):
|
|
|
|
|
|
|
|
def test_alert(self):
|
|
|
|
|
2016-08-21 18:15:19 +02:00
|
|
|
ses = lt.session({'alert_mask': lt.alert.category_t.all_categories, 'enable_dht': False})
|
2016-01-04 05:34:57 +01:00
|
|
|
ti = lt.torrent_info('base.torrent');
|
2016-02-27 16:32:13 +01:00
|
|
|
h = ses.add_torrent({'ti': ti, 'save_path': os.getcwd()})
|
|
|
|
st = h.status()
|
2016-01-04 05:34:57 +01:00
|
|
|
time.sleep(1)
|
|
|
|
ses.remove_torrent(h)
|
2016-01-06 06:17:51 +01:00
|
|
|
ses.wait_for_alert(1000) # milliseconds
|
2016-01-04 05:34:57 +01:00
|
|
|
alerts = ses.pop_alerts()
|
|
|
|
for a in alerts:
|
2016-01-06 06:17:51 +01:00
|
|
|
print(a.message())
|
|
|
|
|
|
|
|
print(st.next_announce)
|
2016-02-27 16:32:13 +01:00
|
|
|
self.assertEqual(st.name, 'temp')
|
2016-01-06 06:17:51 +01:00
|
|
|
print(st.errc.message())
|
|
|
|
print(st.pieces)
|
|
|
|
print(st.last_seen_complete)
|
|
|
|
print(st.completed_time)
|
|
|
|
print(st.progress)
|
|
|
|
print(st.num_pieces)
|
|
|
|
print(st.distributed_copies)
|
|
|
|
print(st.paused)
|
|
|
|
print(st.info_hash)
|
2016-02-27 16:32:13 +01:00
|
|
|
self.assertEqual(st.save_path, os.getcwd())
|
2016-01-04 05:34:57 +01:00
|
|
|
|
2016-04-04 05:50:06 +02:00
|
|
|
def test_pop_alerts(self):
|
2016-08-21 18:15:19 +02:00
|
|
|
ses = lt.session({'alert_mask': lt.alert.category_t.all_categories, 'enable_dht': False})
|
2016-04-04 05:50:06 +02:00
|
|
|
|
|
|
|
ses.async_add_torrent({"ti": lt.torrent_info("base.torrent"), "save_path": "."})
|
|
|
|
# this will cause an error (because of duplicate torrents) and the
|
|
|
|
# torrent_info object created here will be deleted once the alert goes out
|
|
|
|
# of scope. When that happens, it will decrement the python object, to allow
|
|
|
|
# it to release the object.
|
|
|
|
# we're trying to catch the error described in this post, with regards to
|
|
|
|
# torrent_info.
|
|
|
|
# https://mail.python.org/pipermail/cplusplus-sig/2007-June/012130.html
|
|
|
|
ses.async_add_torrent({"ti": lt.torrent_info("base.torrent"), "save_path": "."})
|
|
|
|
time.sleep(1)
|
|
|
|
for i in range(0, 10):
|
|
|
|
alerts = ses.pop_alerts()
|
|
|
|
for a in alerts:
|
|
|
|
print(a.message())
|
|
|
|
time.sleep(0.1)
|
|
|
|
|
2015-07-26 01:40:07 +02:00
|
|
|
class test_bencoder(unittest.TestCase):
|
|
|
|
|
|
|
|
def test_bencode(self):
|
|
|
|
|
|
|
|
encoded = lt.bencode({'a': 1, 'b': [1,2,3], 'c': 'foo'})
|
2016-01-11 04:53:54 +01:00
|
|
|
self.assertEqual(encoded, b'd1:ai1e1:bli1ei2ei3ee1:c3:fooe')
|
2015-07-26 01:40:07 +02:00
|
|
|
|
|
|
|
def test_bdecode(self):
|
|
|
|
|
2016-01-11 04:53:54 +01:00
|
|
|
encoded = b'd1:ai1e1:bli1ei2ei3ee1:c3:fooe'
|
2015-07-26 01:40:07 +02:00
|
|
|
decoded = lt.bdecode(encoded)
|
2016-01-11 04:53:54 +01:00
|
|
|
self.assertEqual(decoded, {b'a': 1, b'b': [1,2,3], b'c': b'foo'})
|
2015-07-26 01:40:07 +02:00
|
|
|
|
2015-10-10 03:21:04 +02:00
|
|
|
class test_sha1hash(unittest.TestCase):
|
|
|
|
|
|
|
|
def test_sha1hash(self):
|
|
|
|
h = 'a0'*20
|
|
|
|
s = lt.sha1_hash(binascii.unhexlify(h))
|
|
|
|
self.assertEqual(h, str(s))
|
|
|
|
|
2015-11-11 06:57:51 +01:00
|
|
|
|
2016-01-14 02:35:54 +01:00
|
|
|
class test_session(unittest.TestCase):
|
2016-03-31 05:35:47 +02:00
|
|
|
|
2016-01-14 02:35:54 +01:00
|
|
|
def test_post_session_stats(self):
|
2016-08-21 18:15:19 +02:00
|
|
|
s = lt.session({'alert_mask': lt.alert.category_t.stats_notification, 'enable_dht': False})
|
2016-01-14 02:35:54 +01:00
|
|
|
s.post_session_stats()
|
|
|
|
a = s.wait_for_alert(1000)
|
|
|
|
self.assertTrue(isinstance(a, lt.session_stats_alert))
|
|
|
|
self.assertTrue(isinstance(a.values, dict))
|
|
|
|
self.assertTrue(len(a.values) > 0)
|
|
|
|
|
2016-11-03 06:01:48 +01:00
|
|
|
def test_unknown_settings(self):
|
|
|
|
try:
|
|
|
|
s = lt.session({'unexpected-key-name': 42})
|
|
|
|
self.assertFalse('should have thrown an exception')
|
|
|
|
except Exception as e:
|
|
|
|
print(e)
|
|
|
|
|
2016-04-04 06:52:21 +02:00
|
|
|
def test_deprecated_settings(self):
|
|
|
|
|
|
|
|
# this detects whether libtorrent was built with deprecated APIs
|
|
|
|
if hasattr(lt, 'version'):
|
2016-08-21 18:15:19 +02:00
|
|
|
s = lt.session({'enable_dht': False})
|
2016-04-04 06:52:21 +02:00
|
|
|
sett = lt.session_settings()
|
|
|
|
sett.num_want = 10;
|
|
|
|
s.set_settings(sett)
|
|
|
|
s.set_settings({'num_want': 33})
|
|
|
|
self.assertEqual(s.get_settings()['num_want'], 33)
|
|
|
|
|
|
|
|
def test_apply_settings(self):
|
|
|
|
|
2016-08-21 18:15:19 +02:00
|
|
|
s = lt.session({'enable_dht': False})
|
2016-04-27 18:28:22 +02:00
|
|
|
s.apply_settings({'num_want': 66, 'user_agent': 'test123'})
|
2016-04-04 06:52:21 +02:00
|
|
|
self.assertEqual(s.get_settings()['num_want'], 66)
|
2016-04-27 18:28:22 +02:00
|
|
|
self.assertEqual(s.get_settings()['user_agent'], 'test123')
|
2016-04-04 06:52:21 +02:00
|
|
|
|
2016-03-31 05:35:47 +02:00
|
|
|
|
2015-07-26 01:40:07 +02:00
|
|
|
if __name__ == '__main__':
|
2016-01-11 04:53:54 +01:00
|
|
|
print(lt.__version__)
|
2016-04-01 23:55:15 +02:00
|
|
|
shutil.copy(os.path.join('..', '..', 'test', 'test_torrents', 'url_seed_multi.torrent'), '.')
|
|
|
|
shutil.copy(os.path.join('..', '..', 'test', 'test_torrents', 'base.torrent'), '.')
|
2016-07-27 06:15:26 +02:00
|
|
|
shutil.copy(os.path.join('..', '..', 'test', 'test_torrents', 'unordered.torrent'), '.')
|
2015-11-11 06:57:51 +01:00
|
|
|
unittest.main()
|
2015-07-26 01:40:07 +02:00
|
|
|
|