Browse Source

[fix] nyaa.si fixed

misnyo 7 years ago
parent
commit
01330f71cd
2 changed files with 139 additions and 86 deletions
  1. 48
    53
      searx/engines/nyaa.py
  2. 91
    33
      tests/unit/engines/test_nyaa.py

+ 48
- 53
searx/engines/nyaa.py View File

@@ -1,7 +1,7 @@
1 1
 """
2
- Nyaa.se (Anime Bittorrent tracker)
2
+ Nyaa.si (Anime Bittorrent tracker)
3 3
 
4
- @website      http://www.nyaa.se/
4
+ @website      http://www.nyaa.si/
5 5
  @provide-api  no
6 6
  @using-api    no
7 7
  @results      HTML
@@ -12,50 +12,25 @@
12 12
 from lxml import html
13 13
 from searx.engines.xpath import extract_text
14 14
 from searx.url_utils import urlencode
15
+from searx.utils import get_torrent_size
15 16
 
16 17
 # engine dependent config
17 18
 categories = ['files', 'images', 'videos', 'music']
18 19
 paging = True
19 20
 
20 21
 # search-url
21
-base_url = 'http://www.nyaa.se/'
22
+base_url = 'http://www.nyaa.si/'
22 23
 search_url = base_url + '?page=search&{query}&offset={offset}'
23 24
 
24 25
 # xpath queries
25
-xpath_results = '//table[@class="tlist"]//tr[contains(@class, "tlistrow")]'
26
-xpath_category = './/td[@class="tlisticon"]/a'
27
-xpath_title = './/td[@class="tlistname"]/a'
28
-xpath_torrent_file = './/td[@class="tlistdownload"]/a'
29
-xpath_filesize = './/td[@class="tlistsize"]/text()'
30
-xpath_seeds = './/td[@class="tlistsn"]/text()'
31
-xpath_leeches = './/td[@class="tlistln"]/text()'
32
-xpath_downloads = './/td[@class="tlistdn"]/text()'
33
-
34
-
35
-# convert a variable to integer or return 0 if it's not a number
36
-def int_or_zero(num):
37
-    if isinstance(num, list):
38
-        if len(num) < 1:
39
-            return 0
40
-        num = num[0]
41
-    if num.isdigit():
42
-        return int(num)
43
-    return 0
44
-
45
-
46
-# get multiplier to convert torrent size to bytes
47
-def get_filesize_mul(suffix):
48
-    return {
49
-        'KB': 1024,
50
-        'MB': 1024 ** 2,
51
-        'GB': 1024 ** 3,
52
-        'TB': 1024 ** 4,
53
-
54
-        'KIB': 1024,
55
-        'MIB': 1024 ** 2,
56
-        'GIB': 1024 ** 3,
57
-        'TIB': 1024 ** 4
58
-    }[str(suffix).upper()]
26
+xpath_results = '//table[contains(@class, "torrent-list")]//tr[not(th)]'
27
+xpath_category = './/td[1]/a[1]'
28
+xpath_title = './/td[2]/a[last()]'
29
+xpath_torrent_links = './/td[3]/a'
30
+xpath_filesize = './/td[4]/text()'
31
+xpath_seeds = './/td[6]/text()'
32
+xpath_leeches = './/td[7]/text()'
33
+xpath_downloads = './/td[8]/text()'
59 34
 
60 35
 
61 36
 # do search-request
@@ -72,6 +47,14 @@ def response(resp):
72 47
     dom = html.fromstring(resp.text)
73 48
 
74 49
     for result in dom.xpath(xpath_results):
50
+        # defaults
51
+        filesize = 0
52
+        seed = 0
53
+        leech = 0
54
+        downloads = 0
55
+        magnet_link = ""
56
+        torrent_link = ""
57
+
75 58
         # category in which our torrent belongs
76 59
         category = result.xpath(xpath_category)[0].attrib.get('title')
77 60
 
@@ -80,26 +63,37 @@ def response(resp):
80 63
         title = extract_text(page_a)
81 64
 
82 65
         # link to the page
83
-        href = page_a.attrib.get('href')
84
-
85
-        # link to the torrent file
86
-        torrent_link = result.xpath(xpath_torrent_file)[0].attrib.get('href')
87
-
88
-        # torrent size
66
+        href = base_url + page_a.attrib.get('href')
67
+
68
+        for link in result.xpath(xpath_torrent_links):
69
+            url = link.attrib.get('href')
70
+            if 'magnet' in url:
71
+                # link to the magnet
72
+                magnet_link = url
73
+            else:
74
+                # link to the torrent file
75
+                torrent_link = url
76
+
77
+        # get seeders and leechers
89 78
         try:
90
-            file_size, suffix = result.xpath(xpath_filesize)[0].split(' ')
91
-            file_size = int(float(file_size) * get_filesize_mul(suffix))
79
+            seed = int(result.xpath(xpath_seeds)[0])
80
+            leech = int(result.xpath(xpath_leeches)[0])
92 81
         except:
93
-            file_size = None
94
-
95
-        # seed count
96
-        seed = int_or_zero(result.xpath(xpath_seeds))
82
+            pass
97 83
 
98
-        # leech count
99
-        leech = int_or_zero(result.xpath(xpath_leeches))
84
+        # let's try to calculate the torrent size
85
+        try:
86
+            filesize_info = result.xpath(xpath_filesize)[0]
87
+            filesize, filesize_multiplier = filesize_info.split()
88
+            filesize = get_torrent_size(filesize, filesize_multiplier)
89
+        except:
90
+            pass
100 91
 
101 92
         # torrent downloads count
102
-        downloads = int_or_zero(result.xpath(xpath_downloads))
93
+        try:
94
+            downloads = result.xpath(xpath_downloads)[0]
95
+        except:
96
+            pass
103 97
 
104 98
         # content string contains all information not included into template
105 99
         content = 'Category: "{category}". Downloaded {downloads} times.'
@@ -110,8 +104,9 @@ def response(resp):
110 104
                         'content': content,
111 105
                         'seed': seed,
112 106
                         'leech': leech,
113
-                        'filesize': file_size,
107
+                        'filesize': filesize,
114 108
                         'torrentfile': torrent_link,
109
+                        'magnetlink': magnet_link,
115 110
                         'template': 'torrent.html'})
116 111
 
117 112
     return results

+ 91
- 33
tests/unit/engines/test_nyaa.py View File

@@ -13,38 +13,92 @@ class TestNyaaEngine(SearxTestCase):
13 13
         params = nyaa.request(query, dic)
14 14
         self.assertTrue('url' in params)
15 15
         self.assertTrue(query in params['url'])
16
-        self.assertTrue('nyaa.se' in params['url'])
16
+        self.assertTrue('nyaa.si' in params['url'])
17 17
 
18 18
     def test_response(self):
19 19
         resp = mock.Mock(text='<html></html>')
20 20
         self.assertEqual(nyaa.response(resp), [])
21 21
 
22 22
         html = """
23
-        <table class="tlist">
24
-          <tbody>
25
-            <tr class="trusted tlistrow">
26
-              <td class="tlisticon">
27
-                <a href="//www.nyaa.se" title="English-translated Anime">
28
-                   <img src="//files.nyaa.se" alt="English-translated Anime">
29
-                </a>
30
-              </td>
31
-              <td class="tlistname">
32
-                <a href="//www.nyaa.se/?page3">
33
-                  Sample torrent title
34
-                </a>
35
-              </td>
36
-              <td class="tlistdownload">
37
-                <a href="//www.nyaa.se/?page_dl" title="Download">
38
-                  <img src="//files.nyaa.se/www-dl.png" alt="DL">
39
-                </a>
40
-              </td>
41
-              <td class="tlistsize">10 MiB</td>
42
-              <td class="tlistsn">1</td>
43
-              <td class="tlistln">3</td>
44
-              <td class="tlistdn">666</td>
45
-              <td class="tlistmn">0</td>
46
-            </tr>
47
-          </tbody>
23
+        <table class="table table-bordered table-hover table-striped torrent-list">
24
+        <thead>
25
+        <tr>
26
+        <th class="hdr-category text-center" style="width:80px;">
27
+        <div>Category</div>
28
+        </th>
29
+        <th class="hdr-name" style="width:auto;">
30
+        <div>Name</div>
31
+        </th>
32
+        <th class="hdr-comments sorting text-center" title="Comments" style="width:50px;">
33
+        <a href="/?f=0&amp;c=0_0&amp;q=Death+Parade&amp;s=comments&amp;o=desc"></a>
34
+        <i class="fa fa-comments-o"></i>
35
+        </th>
36
+        <th class="hdr-link text-center" style="width:70px;">
37
+        <div>Link</div>
38
+        </th>
39
+        <th class="hdr-size sorting text-center" style="width:100px;">
40
+        <a href="/?f=0&amp;c=0_0&amp;q=Death+Parade&amp;s=size&amp;o=desc"></a>
41
+        <div>Size</div>
42
+        </th>
43
+        <th class="hdr-date sorting_desc text-center" title="In local time" style="width:140px;">
44
+        <a href="/?f=0&amp;c=0_0&amp;q=Death+Parade&amp;s=id&amp;o=asc"></a>
45
+        <div>Date</div>
46
+        </th>
47
+        <th class="hdr-seeders sorting text-center" title="Seeders" style="width:50px;">
48
+        <a href="/?f=0&amp;c=0_0&amp;q=Death+Parade&amp;s=seeders&amp;o=desc"></a>
49
+        <i class="fa fa-arrow-up" aria-hidden="true"></i>
50
+        </th>
51
+        <th class="hdr-leechers sorting text-center" title="Leechers" style="width:50px;">
52
+        <a href="/?f=0&amp;c=0_0&amp;q=Death+Parade&amp;s=leechers&amp;o=desc"></a>
53
+        <i class="fa fa-arrow-down" aria-hidden="true"></i>
54
+        </th>
55
+        <th class="hdr-downloads sorting text-center" title="Completed downloads" style="width:50px;">
56
+        <a href="/?f=0&amp;c=0_0&amp;q=Death+Parade&amp;s=downloads&amp;o=desc"></a>
57
+        <i class="fa fa-check" aria-hidden="true"></i>
58
+        </th>
59
+        </tr>
60
+        </thead>
61
+        <tbody>
62
+        <tr class="default">
63
+        <td style="padding:0 4px;">
64
+        <a href="/?c=1_2" title="Anime - English-translated">
65
+        <img src="/static/img/icons/nyaa/1_2.png" alt="Anime - English-translated">
66
+        </a>
67
+        </td>
68
+        <td colspan="2">
69
+        <a href="/view/1" title="Sample title 1">Sample title 1</a>
70
+        </td>
71
+        <td class="text-center" style="white-space: nowrap;">
72
+        <a href="/download/1.torrent"><i class="fa fa-fw fa-download"></i></a>
73
+        <a href="magnet:?xt=urn:btih:2"><i class="fa fa-fw fa-magnet"></i></a>
74
+        </td>
75
+        <td class="text-center">723.7 MiB</td>
76
+        <td class="text-center" data-timestamp="1503307456" title="1 week 3
77
+        days 9 hours 44 minutes 39 seconds ago">2017-08-21 11:24</td>
78
+        <td class="text-center" style="color: green;">1</td>
79
+        <td class="text-center" style="color: red;">3</td>
80
+        <td class="text-center">12</td>
81
+        </tr>
82
+        <tr class="default">
83
+        <td style="padding:0 4px;">
84
+        <a href="/?c=1_2" title="Anime - English-translated">
85
+        <img src="/static/img/icons/nyaa/1_2.png" alt="Anime - English-translated">
86
+        </a>
87
+        </td>
88
+        <td colspan="2">
89
+        <a href="/view/2" title="Sample title 2">Sample title 2</a>
90
+        </td>
91
+        <td class="text-center" style="white-space: nowrap;">
92
+        <a href="magnet:?xt=urn:btih:2"><i class="fa fa-fw fa-magnet"></i></a>
93
+        </td>
94
+        <td class="text-center">8.2 GiB</td>
95
+        <td class="text-center" data-timestamp="1491608400" title="4 months 3
96
+        weeks 4 days 19 hours 28 minutes 55 seconds ago">2017-04-08 01:40</td>
97
+        <td class="text-center" style="color: green;">10</td>
98
+        <td class="text-center" style="color: red;">1</td>
99
+        <td class="text-center">206</td>
100
+        </tr>
101
+        </tbody>
48 102
         </table>
49 103
         """
50 104
 
@@ -52,15 +106,19 @@ class TestNyaaEngine(SearxTestCase):
52 106
         results = nyaa.response(resp)
53 107
 
54 108
         self.assertEqual(type(results), list)
55
-        self.assertEqual(len(results), 1)
109
+        self.assertEqual(len(results), 2)
56 110
 
57 111
         r = results[0]
58
-        self.assertTrue(r['url'].find('www.nyaa.se/?page3') >= 0)
59
-        self.assertTrue(r['torrentfile'].find('www.nyaa.se/?page_dl') >= 0)
60
-        self.assertTrue(r['content'].find('English-translated Anime') >= 0)
61
-        self.assertTrue(r['content'].find('Downloaded 666 times.') >= 0)
112
+        self.assertTrue(r['url'].find('1') >= 0)
113
+        self.assertTrue(r['torrentfile'].find('1.torrent') >= 0)
114
+        self.assertTrue(r['content'].find('Anime - English-translated') >= 0)
115
+        self.assertTrue(r['content'].find('Downloaded 12 times.') >= 0)
62 116
 
63
-        self.assertEqual(r['title'], 'Sample torrent title')
117
+        self.assertEqual(r['title'], 'Sample title 1')
64 118
         self.assertEqual(r['seed'], 1)
65 119
         self.assertEqual(r['leech'], 3)
66
-        self.assertEqual(r['filesize'], 10 * 1024 * 1024)
120
+        self.assertEqual(r['filesize'], 723700000)
121
+
122
+        r = results[1]
123
+        self.assertTrue(r['url'].find('2') >= 0)
124
+        self.assertTrue(r['magnetlink'].find('magnet:') >= 0)