jmcomic 2.5.12__py3-none-any.whl → 2.5.14__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
jmcomic/__init__.py CHANGED
@@ -2,7 +2,7 @@
2
2
  # 被依赖方 <--- 使用方
3
3
  # config <--- entity <--- toolkit <--- client <--- option <--- downloader
4
4
 
5
- __version__ = '2.5.12'
5
+ __version__ = '2.5.14'
6
6
 
7
7
  from .api import *
8
8
  from .jm_plugin import *
jmcomic/jm_client_impl.py CHANGED
@@ -237,6 +237,9 @@ class JmHtmlClient(AbstractJmClient):
237
237
 
238
238
  func_to_cache = ['search', 'fetch_detail_entity']
239
239
 
240
+ API_SEARCH = '/search/photos'
241
+ API_CATEGORY = '/albums'
242
+
240
243
  def add_favorite_album(self,
241
244
  album_id,
242
245
  folder_id='0',
@@ -304,7 +307,12 @@ class JmHtmlClient(AbstractJmClient):
304
307
  main_tag: int,
305
308
  order_by: str,
306
309
  time: str,
310
+ category: str,
311
+ sub_category: Optional[str],
307
312
  ) -> JmSearchPage:
313
+ """
314
+ 网页搜索API
315
+ """
308
316
  params = {
309
317
  'main_tag': main_tag,
310
318
  'search_query': search_query,
@@ -313,8 +321,10 @@ class JmHtmlClient(AbstractJmClient):
313
321
  't': time,
314
322
  }
315
323
 
324
+ url = self.build_search_url(self.API_SEARCH, category, sub_category)
325
+
316
326
  resp = self.get_jm_html(
317
- self.append_params_to_url('/search/photos', params),
327
+ self.append_params_to_url(url, params),
318
328
  allow_redirects=True,
319
329
  )
320
330
 
@@ -326,11 +336,31 @@ class JmHtmlClient(AbstractJmClient):
326
336
  else:
327
337
  return JmPageTool.parse_html_to_search_page(resp.text)
328
338
 
339
+ @classmethod
340
+ def build_search_url(cls, base: str, category: str, sub_category: Optional[str]):
341
+ """
342
+ 构建网页搜索/分类的URL
343
+
344
+ 示例:
345
+ :param base: "/search/photos"
346
+ :param category CATEGORY_DOUJIN
347
+ :param sub_category SUB_DOUJIN_CG
348
+ :return "/search/photos/doujin/sub/CG"
349
+ """
350
+ if category == JmMagicConstants.CATEGORY_ALL:
351
+ return base
352
+
353
+ if sub_category is None:
354
+ return f'{base}/{category}'
355
+ else:
356
+ return f'{base}/{category}/sub/{sub_category}'
357
+
329
358
  def categories_filter(self,
330
359
  page: int,
331
360
  time: str,
332
361
  category: str,
333
362
  order_by: str,
363
+ sub_category: Optional[str] = None,
334
364
  ) -> JmCategoryPage:
335
365
  params = {
336
366
  'page': page,
@@ -338,7 +368,7 @@ class JmHtmlClient(AbstractJmClient):
338
368
  't': time,
339
369
  }
340
370
 
341
- url = f'/albums/' + (category if category != JmMagicConstants.CATEGORY_ALL else '')
371
+ url = self.build_search_url(self.API_CATEGORY, category, sub_category)
342
372
 
343
373
  resp = self.get_jm_html(
344
374
  self.append_params_to_url(url, params),
@@ -573,7 +603,12 @@ class JmApiClient(AbstractJmClient):
573
603
  main_tag: int,
574
604
  order_by: str,
575
605
  time: str,
606
+ category: str,
607
+ sub_category: Optional[str],
576
608
  ) -> JmSearchPage:
609
+ """
610
+ 移动端暂不支持 category和sub_category
611
+ """
577
612
  params = {
578
613
  'main_tag': main_tag,
579
614
  'search_query': search_query,
@@ -603,7 +638,11 @@ class JmApiClient(AbstractJmClient):
603
638
  time: str,
604
639
  category: str,
605
640
  order_by: str,
641
+ sub_category: Optional[str] = None,
606
642
  ):
643
+ """
644
+ 移动端不支持 sub_category
645
+ """
607
646
  # o: mv, mv_m, mv_w, mv_t
608
647
  o = f'{order_by}_{time}' if time != JmMagicConstants.TIME_ALL else order_by
609
648
 
@@ -308,9 +308,14 @@ class JmSearchAlbumClient:
308
308
  main_tag: int,
309
309
  order_by: str,
310
310
  time: str,
311
+ category: str,
312
+ sub_category: Optional[str],
311
313
  ) -> JmSearchPage:
312
314
  """
313
315
  搜索【成人A漫】
316
+ 网页端与移动端的搜索有差别:
317
+
318
+ - 移动端不支持 category, sub_category参数,网页端支持全部参数
314
319
  """
315
320
  raise NotImplementedError
316
321
 
@@ -319,55 +324,65 @@ class JmSearchAlbumClient:
319
324
  page: int = 1,
320
325
  order_by: str = JmMagicConstants.ORDER_BY_LATEST,
321
326
  time: str = JmMagicConstants.TIME_ALL,
327
+ category: str = JmMagicConstants.CATEGORY_ALL,
328
+ sub_category: Optional[str] = None,
322
329
  ):
323
330
  """
324
331
  对应禁漫的站内搜索
325
332
  """
326
- return self.search(search_query, page, 0, order_by, time)
333
+ return self.search(search_query, page, 0, order_by, time, category, sub_category)
327
334
 
328
335
  def search_work(self,
329
336
  search_query: str,
330
337
  page: int = 1,
331
338
  order_by: str = JmMagicConstants.ORDER_BY_LATEST,
332
339
  time: str = JmMagicConstants.TIME_ALL,
340
+ category: str = JmMagicConstants.CATEGORY_ALL,
341
+ sub_category: Optional[str] = None,
333
342
  ):
334
343
  """
335
344
  搜索album的作品 work
336
345
  """
337
- return self.search(search_query, page, 1, order_by, time)
346
+ return self.search(search_query, page, 1, order_by, time, category, sub_category)
338
347
 
339
348
  def search_author(self,
340
349
  search_query: str,
341
350
  page: int = 1,
342
351
  order_by: str = JmMagicConstants.ORDER_BY_LATEST,
343
352
  time: str = JmMagicConstants.TIME_ALL,
353
+ category: str = JmMagicConstants.CATEGORY_ALL,
354
+ sub_category: Optional[str] = None,
344
355
  ):
345
356
  """
346
357
  搜索album的作者 author
347
358
  """
348
- return self.search(search_query, page, 2, order_by, time)
359
+ return self.search(search_query, page, 2, order_by, time, category, sub_category)
349
360
 
350
361
  def search_tag(self,
351
362
  search_query: str,
352
363
  page: int = 1,
353
364
  order_by: str = JmMagicConstants.ORDER_BY_LATEST,
354
365
  time: str = JmMagicConstants.TIME_ALL,
366
+ category: str = JmMagicConstants.CATEGORY_ALL,
367
+ sub_category: Optional[str] = None,
355
368
  ):
356
369
  """
357
370
  搜索album的标签 tag
358
371
  """
359
- return self.search(search_query, page, 3, order_by, time)
372
+ return self.search(search_query, page, 3, order_by, time, category, sub_category)
360
373
 
361
374
  def search_actor(self,
362
375
  search_query: str,
363
376
  page: int = 1,
364
377
  order_by: str = JmMagicConstants.ORDER_BY_LATEST,
365
378
  time: str = JmMagicConstants.TIME_ALL,
379
+ category: str = JmMagicConstants.CATEGORY_ALL,
380
+ sub_category: Optional[str] = None,
366
381
  ):
367
382
  """
368
383
  搜索album的登场角色 actor
369
384
  """
370
- return self.search(search_query, page, 4, order_by, time)
385
+ return self.search(search_query, page, 4, order_by, time, category, sub_category)
371
386
 
372
387
 
373
388
  class JmCategoryClient:
@@ -384,6 +399,7 @@ class JmCategoryClient:
384
399
  time: str,
385
400
  category: str,
386
401
  order_by: str,
402
+ sub_category: Optional[str] = None,
387
403
  ) -> JmCategoryPage:
388
404
  """
389
405
  分类
@@ -391,6 +407,7 @@ class JmCategoryClient:
391
407
  :param page: 页码
392
408
  :param time: 时间范围,默认是全部时间
393
409
  :param category: 类别,默认是最新,即显示最新的禁漫本子
410
+ :param sub_category: 副分类,仅网页端有这功能
394
411
  :param order_by: 排序方式,默认是观看数
395
412
  """
396
413
  raise NotImplementedError
@@ -522,6 +539,8 @@ class JmcomicClient(
522
539
  page: int = 1,
523
540
  order_by: str = JmMagicConstants.ORDER_BY_LATEST,
524
541
  time: str = JmMagicConstants.TIME_ALL,
542
+ category: str = JmMagicConstants.CATEGORY_ALL,
543
+ sub_category: Optional[str] = None,
525
544
  ) -> Generator[JmSearchPage, Dict, None]:
526
545
  """
527
546
  搜索结果的生成器,支持下面这种调用方式:
@@ -552,6 +571,8 @@ class JmcomicClient(
552
571
  'main_tag': main_tag,
553
572
  'order_by': order_by,
554
573
  'time': time,
574
+ 'category': category,
575
+ 'sub_category': sub_category,
555
576
  }
556
577
 
557
578
  yield from self.do_page_iter(params, page, self.search)
@@ -561,6 +582,7 @@ class JmcomicClient(
561
582
  time: str = JmMagicConstants.TIME_ALL,
562
583
  category: str = JmMagicConstants.CATEGORY_ALL,
563
584
  order_by: str = JmMagicConstants.ORDER_BY_LATEST,
585
+ sub_category: Optional[str] = None,
564
586
  ) -> Generator[JmCategoryPage, Dict, None]:
565
587
  """
566
588
  见 search_gen
@@ -569,6 +591,7 @@ class JmcomicClient(
569
591
  'time': time,
570
592
  'category': category,
571
593
  'order_by': order_by,
594
+ 'sub_category': sub_category,
572
595
  }
573
596
 
574
597
  yield from self.do_page_iter(params, page, self.categories_filter)
jmcomic/jm_config.py CHANGED
@@ -35,7 +35,29 @@ class JmMagicConstants:
35
35
  CATEGORY_DOUJIN_COSPLAY = 'doujin_cosplay' # cosplay
36
36
  CATEGORY_3D = '3D' # 3D
37
37
  CATEGORY_ENGLISH_SITE = 'english_site' # 英文站
38
- CATEGORY_JM_TEAM = '禁漫漢化組'
38
+
39
+ # 副分类
40
+ SUB_CHINESE = 'chinese' # 汉化,通用副分类
41
+ SUB_JAPANESE = 'japanese' # 日语,通用副分类
42
+
43
+ # 其他类(CATEGORY_ANOTHER)的副分类
44
+ SUB_ANOTHER_OTHER = 'other' # 其他漫画
45
+ SUB_ANOTHER_3D = '3d' # 3D
46
+ SUB_ANOTHER_COSPLAY = 'cosplay' # cosplay
47
+
48
+ # 同人(SUB_CHINESE)的副分类
49
+ SUB_DOUJIN_CG = 'CG' # CG
50
+ SUB_DOUJIN_CHINESE = SUB_CHINESE
51
+ SUB_DOUJIN_JAPANESE = SUB_JAPANESE
52
+
53
+ # 短篇(CATEGORY_SHORT)的副分类
54
+ SUB_SHORT_CHINESE = SUB_CHINESE
55
+ SUB_SHORT_JAPANESE = SUB_JAPANESE
56
+
57
+ # 单本(CATEGORY_SINGLE)的副分类
58
+ SUB_SINGLE_CHINESE = SUB_CHINESE
59
+ SUB_SINGLE_JAPANESE = SUB_JAPANESE
60
+ SUB_SINGLE_YOUTH = 'youth'
39
61
 
40
62
  # 分页大小
41
63
  PAGE_SIZE_SEARCH = 80
@@ -53,7 +75,7 @@ class JmMagicConstants:
53
75
  APP_TOKEN_SECRET = '18comicAPP'
54
76
  APP_TOKEN_SECRET_2 = '18comicAPPContent'
55
77
  APP_DATA_SECRET = '185Hcomic3PAPP7R'
56
- APP_VERSION = '1.6.7'
78
+ APP_VERSION = '1.7.0'
57
79
  APP_HEADERS_TEMPLATE = {
58
80
  'Accept-Encoding': 'gzip',
59
81
  'user-agent': 'Mozilla/5.0 (Linux; Android 9; V1938CT Build/PQ3A.190705.11211812; wv) AppleWebKit/537.36 (KHTML, '
@@ -65,14 +87,20 @@ class JmMagicConstants:
65
87
  'accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,'
66
88
  'application/signed-exchange;v=b3;q=0.7',
67
89
  'accept-language': 'zh-CN,zh;q=0.9',
68
- 'sec-ch-ua': '"Not.A/Brand";v="8", "Chromium";v="114", "Google Chrome";v="114"',
90
+ 'cache-control': 'no-cache',
91
+ 'dnt': '1',
92
+ 'pragma': 'no-cache',
93
+ 'priority': 'u=0, i',
94
+ 'referer': 'https://18comic.vip/',
95
+ 'sec-ch-ua': '"Chromium";v="124", "Google Chrome";v="124", "Not-A.Brand";v="99"',
69
96
  'sec-ch-ua-mobile': '?0',
70
97
  'sec-ch-ua-platform': '"Windows"',
71
98
  'sec-fetch-dest': 'document',
72
99
  'sec-fetch-mode': 'navigate',
73
100
  'sec-fetch-site': 'none',
74
101
  'sec-fetch-user': '?1',
75
- 'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 '
102
+ 'upgrade-insecure-requests': '1',
103
+ 'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/124.0.0.0 '
76
104
  'Safari/537.36',
77
105
  }
78
106
 
jmcomic/jm_plugin.py CHANGED
@@ -1,6 +1,7 @@
1
1
  """
2
2
  该文件存放的是option插件
3
3
  """
4
+ import os.path
4
5
 
5
6
  from .jm_option import *
6
7
 
@@ -296,91 +297,63 @@ class ZipPlugin(JmOptionPlugin):
296
297
  zip_dir = JmcomicText.parse_to_abspath(zip_dir)
297
298
  mkdir_if_not_exists(zip_dir)
298
299
 
299
- # 原文件夹 -> zip文件
300
- dir_zip_dict: Dict[str, Optional[str]] = {}
300
+ path_to_delete = []
301
301
  photo_dict = downloader.download_success_dict[album]
302
302
 
303
303
  if level == 'album':
304
304
  zip_path = self.get_zip_path(album, None, filename_rule, suffix, zip_dir)
305
- self.zip_album(album, photo_dict, zip_path, dir_zip_dict)
305
+ self.zip_album(album, photo_dict, zip_path, path_to_delete)
306
306
 
307
307
  elif level == 'photo':
308
308
  for photo, image_list in photo_dict.items():
309
309
  zip_path = self.get_zip_path(None, photo, filename_rule, suffix, zip_dir)
310
- self.zip_photo(photo, image_list, zip_path, dir_zip_dict)
310
+ self.zip_photo(photo, image_list, zip_path, path_to_delete)
311
311
 
312
312
  else:
313
313
  ExceptionTool.raises(f'Not Implemented Zip Level: {level}')
314
314
 
315
- self.after_zip(dir_zip_dict)
315
+ self.after_zip(path_to_delete)
316
316
 
317
- def zip_photo(self, photo, image_list: list, zip_path: str, dir_zip_dict) -> Optional[str]:
317
+ def zip_photo(self, photo, image_list: list, zip_path: str, path_to_delete):
318
318
  """
319
319
  压缩photo文件夹
320
- :returns: photo文件夹路径
321
320
  """
322
321
  photo_dir = self.option.decide_image_save_dir(photo) \
323
322
  if len(image_list) == 0 \
324
323
  else os.path.dirname(image_list[0][0])
325
324
 
326
- all_filepath = set(map(lambda t: self.unified_path(t[0]), image_list))
327
-
328
- if len(all_filepath) == 0:
329
- self.log('无下载文件,无需压缩', 'skip')
330
- return None
331
-
332
325
  from common import backup_dir_to_zip
333
- backup_dir_to_zip(
334
- photo_dir,
335
- zip_path,
336
- acceptor=lambda f: os.path.isdir(f) or self.unified_path(f) in all_filepath
337
- ).close()
326
+ backup_dir_to_zip(photo_dir, zip_path)
338
327
 
339
328
  self.log(f'压缩章节[{photo.photo_id}]成功 → {zip_path}', 'finish')
340
- dir_zip_dict[self.unified_path(photo_dir)] = zip_path
329
+ path_to_delete.append(self.unified_path(photo_dir))
341
330
 
342
331
  @staticmethod
343
332
  def unified_path(f):
344
333
  return fix_filepath(f, os.path.isdir(f))
345
334
 
346
- def zip_album(self, album, photo_dict: dict, zip_path, dir_zip_dict) -> Optional[str]:
335
+ def zip_album(self, album, photo_dict: dict, zip_path, path_to_delete):
347
336
  """
348
337
  压缩album文件夹
349
- :returns: album文件夹路径
350
338
  """
351
339
 
352
- # 所有下载了的图片文件的路径
353
- all_filepath: Set[str] = set(path for ls in photo_dict.values() for path, _ in ls)
354
-
355
- if len(all_filepath) == 0:
356
- self.log('无下载文件,无需压缩', 'skip')
357
- return
358
-
359
- # 该本子的所有章节的图片所在文件夹
360
- photo_dir_list = [self.option.decide_image_save_dir(photo) for photo in photo_dict.keys()]
361
-
362
- # 压缩文件对象
363
- from common import backup_dir_to_zip
340
+ album_dir = self.option.dir_rule.decide_album_root_dir(album)
364
341
  import zipfile
365
- zfile = zipfile.ZipFile(zip_path, 'w')
366
-
367
- for photo_dir in photo_dir_list:
368
- photo_dir = self.unified_path(photo_dir)
369
- backup_dir_to_zip(
370
- photo_dir,
371
- zip_path,
372
- zfile=zfile,
373
- prefix=os.path.basename(photo_dir.rstrip('/')),
374
- acceptor=lambda f: os.path.isdir(f) or self.unified_path(f) in all_filepath
375
- )
376
- dir_zip_dict[photo_dir] = zip_path
377
-
378
- zfile.close()
342
+ with zipfile.ZipFile(zip_path, 'w', zipfile.ZIP_DEFLATED) as f:
343
+ for photo in photo_dict.keys():
344
+ # 定位到章节所在文件夹
345
+ photo_dir = self.unified_path(self.option.decide_image_save_dir(photo))
346
+ # 章节文件夹标记为删除
347
+ path_to_delete.append(photo_dir)
348
+ for file in files_of_dir(photo_dir):
349
+ abspath = os.path.join(photo_dir, file)
350
+ relpath = os.path.relpath(abspath, album_dir)
351
+ f.write(abspath, relpath)
379
352
  self.log(f'压缩本子[{album.album_id}]成功 → {zip_path}', 'finish')
380
353
 
381
- def after_zip(self, dir_zip_dict: Dict[str, Optional[str]]):
354
+ def after_zip(self, path_to_delete: List[str]):
382
355
  # 删除所有原文件
383
- dirs = sorted(dir_zip_dict.keys(), reverse=True)
356
+ dirs = sorted(path_to_delete, reverse=True)
384
357
  image_paths = [
385
358
  path
386
359
  for photo_dict in self.downloader.download_success_dict.values()
@@ -650,7 +623,7 @@ class FavoriteFolderExportPlugin(JmOptionPlugin):
650
623
  """
651
624
  import zipfile
652
625
 
653
- with zipfile.ZipFile(zip_path, 'w') as zipf:
626
+ with zipfile.ZipFile(zip_path, 'w', zipfile.ZIP_DEFLATED) as zipf:
654
627
  # 获取文件夹中的文件列表并将其添加到 ZIP 文件中
655
628
  for file in files:
656
629
  zipf.write(file, arcname=of_file_name(file))
jmcomic/jm_toolkit.py CHANGED
@@ -135,7 +135,7 @@ class JmcomicText:
135
135
  last_pattern = pattern[len(pattern) - 1]
136
136
  # 缩小文本
137
137
  for i in range(0, len(pattern) - 1):
138
- match = pattern[i].search(text)
138
+ match: Match = pattern[i].search(text)
139
139
  if match is None:
140
140
  return None
141
141
  text = match[0]
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: jmcomic
3
- Version: 2.5.12
3
+ Version: 2.5.14
4
4
  Summary: Python API For JMComic (禁漫天堂)
5
5
  Home-page: https://github.com/hect0x7/JMComic-Crawler-Python
6
6
  Author: hect0x7
@@ -0,0 +1,18 @@
1
+ jmcomic/__init__.py,sha256=M5LrJ_IvaYPMjPvydKmvbgueRZj0xVIzi2DQ6eppfBc,903
2
+ jmcomic/api.py,sha256=uLHtSof7ZaiWZcNDDAz6grD1NYs4VxsoCOrsge3G7v8,3864
3
+ jmcomic/cl.py,sha256=PBSh0JndNFZw3B7WJPj5Y8SeFdKzHE00jIwYo9An-K0,3475
4
+ jmcomic/jm_client_impl.py,sha256=Ab1b_l_NQSNp2__hjes6RRbMsOETmjlkf6O8ddNfHlg,38968
5
+ jmcomic/jm_client_interface.py,sha256=5JFSZEvAffDPh7oo8GQNVbm_Hdlmy_KYvTwrkY-OSA4,19074
6
+ jmcomic/jm_config.py,sha256=7Ms2SK5FdECJ5tolWXIRMzTlrgflgLX3b-3bL-Yq_38,16254
7
+ jmcomic/jm_downloader.py,sha256=0r4z7FRnow6xkRy_WTv7nLQOhYdtZmoouw0BNrhngco,10397
8
+ jmcomic/jm_entity.py,sha256=-WUKYGkNToQv5Hja3RkI5abHF1g46WpLyVjoyXMs-wI,18974
9
+ jmcomic/jm_exception.py,sha256=B9APE1jw23JBzCT12eV_imCZny3mNrSXju1p2IquaHA,4801
10
+ jmcomic/jm_option.py,sha256=4J7RJi1D2plVjNhPXE2QSjMuBeQzlpeAwRCojWqNqnw,22025
11
+ jmcomic/jm_plugin.py,sha256=xluJnzBugFMFUdz5yZ8Pb3BnIjKOECqIPjJD1rIod3Q,34911
12
+ jmcomic/jm_toolkit.py,sha256=hcg5qqpFlZilUVN2CmT2jtUK0ufRNCx7SuWYNT4BEdk,28662
13
+ jmcomic-2.5.14.dist-info/LICENSE,sha256=kz4coTxZxuGxisK3W00tjK57Zh3RcMGq-EnbXrK7-xA,1064
14
+ jmcomic-2.5.14.dist-info/METADATA,sha256=VO1j3KaCPYeTlXwDty6nOcVxn8YKgKjTzC5CRsFK0B8,7075
15
+ jmcomic-2.5.14.dist-info/WHEEL,sha256=GJ7t_kWBFywbagK5eo9IoUwLW6oyOeTKmQ-9iHFVNxQ,92
16
+ jmcomic-2.5.14.dist-info/entry_points.txt,sha256=tRbQltaGSBjejI0c9jYt-4SXQMd5nSDHcMvHmuTy4ow,44
17
+ jmcomic-2.5.14.dist-info/top_level.txt,sha256=puvVMFYJqIbd6NOTMEvOyugMTT8woBfSQyxEBan3zY4,8
18
+ jmcomic-2.5.14.dist-info/RECORD,,
@@ -1,18 +0,0 @@
1
- jmcomic/__init__.py,sha256=1QCB9drWLz_G92PcVXTxmj-hCjpl4qW4NKNwvTxc0pA,903
2
- jmcomic/api.py,sha256=uLHtSof7ZaiWZcNDDAz6grD1NYs4VxsoCOrsge3G7v8,3864
3
- jmcomic/cl.py,sha256=PBSh0JndNFZw3B7WJPj5Y8SeFdKzHE00jIwYo9An-K0,3475
4
- jmcomic/jm_client_impl.py,sha256=Oszkw7dKfg8DHzmjyBVD1DrHp9OlvLK33IQHB7uXo48,37811
5
- jmcomic/jm_client_interface.py,sha256=UmwIZt-7-zHfBHq8sNFnfNf-C5kkCQW8_yKKI4cfJYU,17688
6
- jmcomic/jm_config.py,sha256=z512nMjxJldxRbn8Kt02DrBZ89wLF3cR5_OtY1pemuA,15359
7
- jmcomic/jm_downloader.py,sha256=0r4z7FRnow6xkRy_WTv7nLQOhYdtZmoouw0BNrhngco,10397
8
- jmcomic/jm_entity.py,sha256=-WUKYGkNToQv5Hja3RkI5abHF1g46WpLyVjoyXMs-wI,18974
9
- jmcomic/jm_exception.py,sha256=B9APE1jw23JBzCT12eV_imCZny3mNrSXju1p2IquaHA,4801
10
- jmcomic/jm_option.py,sha256=4J7RJi1D2plVjNhPXE2QSjMuBeQzlpeAwRCojWqNqnw,22025
11
- jmcomic/jm_plugin.py,sha256=F5mvqDNU0ir4XUNWpYroV4EdVR3bDBB-FOS15_l1T4k,35732
12
- jmcomic/jm_toolkit.py,sha256=Skd7bp008Brr3hiXUlm3lgGpPsB1UmmX9XU2Dbx3jps,28655
13
- jmcomic-2.5.12.dist-info/LICENSE,sha256=kz4coTxZxuGxisK3W00tjK57Zh3RcMGq-EnbXrK7-xA,1064
14
- jmcomic-2.5.12.dist-info/METADATA,sha256=8A6vYjiV74XxGq-B0NKwN8U4N7104yye6Ri8kSGt-HM,7075
15
- jmcomic-2.5.12.dist-info/WHEEL,sha256=GJ7t_kWBFywbagK5eo9IoUwLW6oyOeTKmQ-9iHFVNxQ,92
16
- jmcomic-2.5.12.dist-info/entry_points.txt,sha256=tRbQltaGSBjejI0c9jYt-4SXQMd5nSDHcMvHmuTy4ow,44
17
- jmcomic-2.5.12.dist-info/top_level.txt,sha256=puvVMFYJqIbd6NOTMEvOyugMTT8woBfSQyxEBan3zY4,8
18
- jmcomic-2.5.12.dist-info/RECORD,,