Browse Source

Merge pull request #1022 from misnyo/nyaa

nyaa.si fixed
Adam Tauber 7 years ago
parent
commit
f4df27fa59
4 changed files with 142 additions and 85 deletions
  1. 40
    49
      searx/engines/nyaa.py
  2. 2
    3
      searx/engines/tokyotoshokan.py
  3. 9
    0
      searx/utils.py
  4. 91
    33
      tests/unit/engines/test_nyaa.py

+ 40
- 49
searx/engines/nyaa.py View File

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
  @provide-api  no
5
  @provide-api  no
6
  @using-api    no
6
  @using-api    no
7
  @results      HTML
7
  @results      HTML
12
 from lxml import html
12
 from lxml import html
13
 from searx.engines.xpath import extract_text
13
 from searx.engines.xpath import extract_text
14
 from searx.url_utils import urlencode
14
 from searx.url_utils import urlencode
15
+from searx.utils import get_torrent_size, int_or_zero
15
 
16
 
16
 # engine dependent config
17
 # engine dependent config
17
 categories = ['files', 'images', 'videos', 'music']
18
 categories = ['files', 'images', 'videos', 'music']
18
 paging = True
19
 paging = True
19
 
20
 
20
 # search-url
21
 # search-url
21
-base_url = 'http://www.nyaa.se/'
22
+base_url = 'http://www.nyaa.si/'
22
 search_url = base_url + '?page=search&{query}&offset={offset}'
23
 search_url = base_url + '?page=search&{query}&offset={offset}'
23
 
24
 
24
 # xpath queries
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
 # do search-request
36
 # do search-request
72
     dom = html.fromstring(resp.text)
47
     dom = html.fromstring(resp.text)
73
 
48
 
74
     for result in dom.xpath(xpath_results):
49
     for result in dom.xpath(xpath_results):
50
+        # defaults
51
+        filesize = 0
52
+        magnet_link = ""
53
+        torrent_link = ""
54
+
75
         # category in which our torrent belongs
55
         # category in which our torrent belongs
76
-        category = result.xpath(xpath_category)[0].attrib.get('title')
56
+        try:
57
+            category = result.xpath(xpath_category)[0].attrib.get('title')
58
+        except:
59
+            pass
77
 
60
 
78
         # torrent title
61
         # torrent title
79
         page_a = result.xpath(xpath_title)[0]
62
         page_a = result.xpath(xpath_title)[0]
80
         title = extract_text(page_a)
63
         title = extract_text(page_a)
81
 
64
 
82
         # link to the page
65
         # link to the page
83
-        href = page_a.attrib.get('href')
66
+        href = base_url + page_a.attrib.get('href')
84
 
67
 
85
-        # link to the torrent file
86
-        torrent_link = result.xpath(xpath_torrent_file)[0].attrib.get('href')
87
-
88
-        # torrent size
89
-        try:
90
-            file_size, suffix = result.xpath(xpath_filesize)[0].split(' ')
91
-            file_size = int(float(file_size) * get_filesize_mul(suffix))
92
-        except:
93
-            file_size = None
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
94
 
76
 
95
         # seed count
77
         # seed count
96
         seed = int_or_zero(result.xpath(xpath_seeds))
78
         seed = int_or_zero(result.xpath(xpath_seeds))
101
         # torrent downloads count
83
         # torrent downloads count
102
         downloads = int_or_zero(result.xpath(xpath_downloads))
84
         downloads = int_or_zero(result.xpath(xpath_downloads))
103
 
85
 
86
+        # let's try to calculate the torrent size
87
+        try:
88
+            filesize_info = result.xpath(xpath_filesize)[0]
89
+            filesize, filesize_multiplier = filesize_info.split()
90
+            filesize = get_torrent_size(filesize, filesize_multiplier)
91
+        except:
92
+            pass
93
+
104
         # content string contains all information not included into template
94
         # content string contains all information not included into template
105
         content = 'Category: "{category}". Downloaded {downloads} times.'
95
         content = 'Category: "{category}". Downloaded {downloads} times.'
106
         content = content.format(category=category, downloads=downloads)
96
         content = content.format(category=category, downloads=downloads)
110
                         'content': content,
100
                         'content': content,
111
                         'seed': seed,
101
                         'seed': seed,
112
                         'leech': leech,
102
                         'leech': leech,
113
-                        'filesize': file_size,
103
+                        'filesize': filesize,
114
                         'torrentfile': torrent_link,
104
                         'torrentfile': torrent_link,
105
+                        'magnetlink': magnet_link,
115
                         'template': 'torrent.html'})
106
                         'template': 'torrent.html'})
116
 
107
 
117
     return results
108
     return results

+ 2
- 3
searx/engines/tokyotoshokan.py View File

14
 from lxml import html
14
 from lxml import html
15
 from searx.engines.xpath import extract_text
15
 from searx.engines.xpath import extract_text
16
 from datetime import datetime
16
 from datetime import datetime
17
-from searx.engines.nyaa import int_or_zero, get_filesize_mul
18
 from searx.url_utils import urlencode
17
 from searx.url_utils import urlencode
18
+from searx.utils import get_torrent_size, int_or_zero
19
 
19
 
20
 # engine dependent config
20
 # engine dependent config
21
 categories = ['files', 'videos', 'music']
21
 categories = ['files', 'videos', 'music']
76
                 try:
76
                 try:
77
                     # ('1.228', 'GB')
77
                     # ('1.228', 'GB')
78
                     groups = size_re.match(item).groups()
78
                     groups = size_re.match(item).groups()
79
-                    multiplier = get_filesize_mul(groups[1])
80
-                    params['filesize'] = int(multiplier * float(groups[0]))
79
+                    params['filesize'] = get_torrent_size(groups[0], groups[1])
81
                 except:
80
                 except:
82
                     pass
81
                     pass
83
             elif item.startswith('Date:'):
82
             elif item.startswith('Date:'):

+ 9
- 0
searx/utils.py View File

290
         return 0
290
         return 0
291
 
291
 
292
 
292
 
293
+# convert a variable to integer or return 0 if it's not a number
294
+def int_or_zero(num):
295
+    if isinstance(num, list):
296
+        if len(num) < 1:
297
+            return 0
298
+        num = num[0]
299
+    return convert_str_to_int(num)
300
+
301
+
293
 def is_valid_lang(lang):
302
 def is_valid_lang(lang):
294
     is_abbr = (len(lang) == 2)
303
     is_abbr = (len(lang) == 2)
295
     if is_abbr:
304
     if is_abbr:

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

13
         params = nyaa.request(query, dic)
13
         params = nyaa.request(query, dic)
14
         self.assertTrue('url' in params)
14
         self.assertTrue('url' in params)
15
         self.assertTrue(query in params['url'])
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
     def test_response(self):
18
     def test_response(self):
19
         resp = mock.Mock(text='<html></html>')
19
         resp = mock.Mock(text='<html></html>')
20
         self.assertEqual(nyaa.response(resp), [])
20
         self.assertEqual(nyaa.response(resp), [])
21
 
21
 
22
         html = """
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
         </table>
102
         </table>
49
         """
103
         """
50
 
104
 
52
         results = nyaa.response(resp)
106
         results = nyaa.response(resp)
53
 
107
 
54
         self.assertEqual(type(results), list)
108
         self.assertEqual(type(results), list)
55
-        self.assertEqual(len(results), 1)
109
+        self.assertEqual(len(results), 2)
56
 
110
 
57
         r = results[0]
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
         self.assertEqual(r['seed'], 1)
118
         self.assertEqual(r['seed'], 1)
65
         self.assertEqual(r['leech'], 3)
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)