]> git.lizzy.rs Git - nhentai.git/blob - nhentai/cmdline.py
Merge branch 'pull/221' into master
[nhentai.git] / nhentai / cmdline.py
1 # coding: utf-8
2
3 import os
4 import sys
5 import json
6 from optparse import OptionParser
7
8 try:
9     from itertools import ifilter as filter
10 except ImportError:
11     pass
12
13 import nhentai.constant as constant
14 from nhentai import __version__
15 from nhentai.utils import urlparse, generate_html, generate_main_html, DB
16 from nhentai.logger import logger
17
18
19 def banner():
20     logger.info(u'''nHentai ver %s: あなたも変態。 いいね?
21        _   _            _        _
22  _ __ | | | | ___ _ __ | |_ __ _(_)
23 | '_ \| |_| |/ _ \ '_ \| __/ _` | |
24 | | | |  _  |  __/ | | | || (_| | |
25 |_| |_|_| |_|\___|_| |_|\__\__,_|_|
26 ''' % __version__)
27
28
29 def load_config():
30     if not os.path.exists(constant.NHENTAI_CONFIG_FILE):
31         return
32
33     try:
34         with open(constant.NHENTAI_CONFIG_FILE, 'r') as f:
35             constant.CONFIG.update(json.load(f))
36     except json.JSONDecodeError:
37         logger.error('Failed to load config file.')
38         write_config()
39
40
41 def write_config():
42     if not os.path.exists(constant.NHENTAI_HOME):
43         os.mkdir(constant.NHENTAI_HOME)
44
45     with open(constant.NHENTAI_CONFIG_FILE, 'w') as f:
46         f.write(json.dumps(constant.CONFIG))
47
48
49 def cmd_parser():
50     load_config()
51
52     parser = OptionParser('\n  nhentai --search [keyword] --download'
53                           '\n  NHENTAI=http://h.loli.club nhentai --id [ID ...]'
54                           '\n  nhentai --file [filename]'
55                           '\n\nEnvironment Variable:\n'
56                           '  NHENTAI                 nhentai mirror url')
57     # operation options
58     parser.add_option('--download', '-D', dest='is_download', action='store_true',
59                       help='download doujinshi (for search results)')
60     parser.add_option('--show', '-S', dest='is_show', action='store_true', help='just show the doujinshi information')
61
62     # doujinshi options
63     parser.add_option('--id', type='string', dest='id', action='store', help='doujinshi ids set, e.g. 1,2,3')
64     parser.add_option('--search', '-s', type='string', dest='keyword', action='store',
65                       help='search doujinshi by keyword')
66     parser.add_option('--favorites', '-F', action='store_true', dest='favorites',
67                       help='list or download your favorites.')
68
69     # page options
70     parser.add_option('--page-all', dest='page_all', action='store_true', default=False,
71                       help='all search results')
72     parser.add_option('--page', '--page-range', type='string', dest='page', action='store', default='',
73                       help='page number of search results. e.g. 1,2-5,14')
74     parser.add_option('--sorting', dest='sorting', action='store', default='recent',
75                       help='sorting of doujinshi (recent / popular / popular-[today|week])',
76                       choices=['recent', 'popular', 'popular-today', 'popular-week'])
77
78     # download options
79     parser.add_option('--output', '-o', type='string', dest='output_dir', action='store', default='./',
80                       help='output dir')
81     parser.add_option('--threads', '-t', type='int', dest='threads', action='store', default=5,
82                       help='thread count for downloading doujinshi')
83     parser.add_option('--timeout', '-T', type='int', dest='timeout', action='store', default=30,
84                       help='timeout for downloading doujinshi')
85     parser.add_option('--delay', '-d', type='int', dest='delay', action='store', default=0,
86                       help='slow down between downloading every doujinshi')
87     parser.add_option('--proxy', type='string', dest='proxy', action='store',
88                       help='store a proxy, for example: -p \'http://127.0.0.1:1080\'')
89     parser.add_option('--file',  '-f', type='string', dest='file', action='store', help='read gallery IDs from file.')
90     parser.add_option('--format', type='string', dest='name_format', action='store',
91                       help='format the saved folder name', default='[%i][%a][%t]')
92     parser.add_option('--dry-run', '-r', action='store_true', dest='dryrun', help='Dry run, skip file download.')
93
94     # generate options
95     parser.add_option('--html', dest='html_viewer', action='store_true',
96                       help='generate a html viewer at current directory')
97     parser.add_option('--no-html', dest='is_nohtml', action='store_true',
98                       help='don\'t generate HTML after downloading')
99     parser.add_option('--gen-main', dest='main_viewer', action='store_true',
100                       help='generate a main viewer contain all the doujin in the folder')
101     parser.add_option('--cbz', '-C', dest='is_cbz', action='store_true',
102                       help='generate Comic Book CBZ File')
103     parser.add_option('--pdf', '-P', dest='is_pdf', action='store_true',
104                       help='generate PDF file')
105     parser.add_option('--rm-origin-dir', dest='rm_origin_dir', action='store_true', default=False,
106                       help='remove downloaded doujinshi dir when generated CBZ or PDF file.')
107
108     # nhentai options
109     parser.add_option('--cookie', type='str', dest='cookie', action='store',
110                       help='set cookie of nhentai to bypass Google recaptcha')
111     parser.add_option('--language', type='str', dest='language', action='store',
112                       help='set default language to parse doujinshis')
113     parser.add_option('--clean-language', dest='clean_language', action='store_true', default=False,
114                       help='set DEFAULT as language to parse doujinshis')
115     parser.add_option('--save-download-history', dest='is_save_download_history', action='store_true',
116                       default=False, help='save downloaded doujinshis, whose will be skipped if you re-download them')
117     parser.add_option('--clean-download-history', action='store_true', default=False, dest='clean_download_history',
118                       help='clean download history')
119     parser.add_option('--template', dest='viewer_template', action='store',
120                       help='set viewer template', default='')
121         
122     parser.add_option('--meta', dest='generate_metadata', action='store_true', help='generate a Metadata File in HDoujin Format')       
123                                           
124
125     try:
126         sys.argv = [unicode(i.decode(sys.stdin.encoding)) for i in sys.argv]
127     except (NameError, TypeError):
128         pass
129     except UnicodeDecodeError:
130         exit(0)
131
132     args, _ = parser.parse_args(sys.argv[1:])
133
134     if args.html_viewer:
135         generate_html()
136         exit(0)
137
138     if args.main_viewer and not args.id and not args.keyword and not args.favorites:
139         generate_main_html()
140         exit(0)
141
142     if args.clean_download_history:
143         with DB() as db:
144             db.clean_all()
145
146         logger.info('Download history cleaned.')
147         exit(0)
148
149     # --- set config ---
150     if args.cookie is not None:
151         constant.CONFIG['cookie'] = args.cookie
152         logger.info('Cookie saved.')
153         write_config()
154         exit(0)
155
156     if args.language is not None:
157         constant.CONFIG['language'] = args.language
158         logger.info('Default language now set to \'{0}\''.format(args.language))
159         write_config()
160         exit(0)
161         # TODO: search without language
162
163     if args.proxy is not None:
164         proxy_url = urlparse(args.proxy)
165         if not args.proxy == '' and proxy_url.scheme not in ('http', 'https'):
166             logger.error('Invalid protocol \'{0}\' of proxy, ignored'.format(proxy_url.scheme))
167             exit(0)
168         else:
169             constant.CONFIG['proxy'] = {
170                 'http': args.proxy,
171                 'https': args.proxy,
172             }
173             logger.info('Proxy now set to \'{0}\'.'.format(args.proxy))
174             write_config()
175             exit(0)
176
177     if args.viewer_template is not None:
178         if not args.viewer_template:
179             args.viewer_template = 'default'
180
181         if not os.path.exists(os.path.join(os.path.dirname(__file__),
182                                            'viewer/{}/index.html'.format(args.viewer_template))):
183             logger.error('Template \'{}\' does not exists'.format(args.viewer_template))
184             exit(1)
185         else:
186             constant.CONFIG['template'] = args.viewer_template
187             write_config()
188
189     # --- end set config ---
190
191     if args.favorites:
192         if not constant.CONFIG['cookie']:
193             logger.warning('Cookie has not been set, please use `nhentai --cookie \'COOKIE\'` to set it.')
194             exit(1)
195
196     if args.id:
197         _ = [i.strip() for i in args.id.split(',')]
198         args.id = set(int(i) for i in _ if i.isdigit())
199
200     if args.file:
201         with open(args.file, 'r') as f:
202             _ = [i.strip() for i in f.readlines()]
203             args.id = set(int(i) for i in _ if i.isdigit())
204
205     if (args.is_download or args.is_show) and not args.id and not args.keyword and not args.favorites:
206         logger.critical('Doujinshi id(s) are required for downloading')
207         parser.print_help()
208         exit(1)
209
210     if not args.keyword and not args.id and not  args.favorites:
211         parser.print_help()
212         exit(1)
213
214     if args.threads <= 0:
215         args.threads = 1
216
217     elif args.threads > 15:
218         logger.critical('Maximum number of used threads is 15')
219         exit(1)
220
221     if args.dryrun and (args.is_cbz or args.is_pdf):
222         logger.critical('Cannot generate PDF or CBZ during dry-run')
223         exit(1)
224
225     return args