crawlo 1.3.3__py3-none-any.whl → 1.3.4__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.

Potentially problematic release.


This version of crawlo might be problematic. Click here for more details.

Files changed (279) hide show
  1. crawlo/__init__.py +87 -63
  2. crawlo/__version__.py +1 -1
  3. crawlo/cli.py +75 -75
  4. crawlo/commands/__init__.py +14 -14
  5. crawlo/commands/check.py +594 -594
  6. crawlo/commands/genspider.py +151 -151
  7. crawlo/commands/help.py +138 -138
  8. crawlo/commands/list.py +155 -155
  9. crawlo/commands/run.py +341 -323
  10. crawlo/commands/startproject.py +436 -436
  11. crawlo/commands/stats.py +187 -187
  12. crawlo/commands/utils.py +196 -196
  13. crawlo/config.py +312 -312
  14. crawlo/config_validator.py +277 -277
  15. crawlo/core/__init__.py +46 -2
  16. crawlo/core/engine.py +439 -365
  17. crawlo/core/processor.py +40 -40
  18. crawlo/core/scheduler.py +257 -256
  19. crawlo/crawler.py +639 -1167
  20. crawlo/data/__init__.py +5 -5
  21. crawlo/data/user_agents.py +194 -194
  22. crawlo/downloader/__init__.py +273 -273
  23. crawlo/downloader/aiohttp_downloader.py +228 -226
  24. crawlo/downloader/cffi_downloader.py +245 -245
  25. crawlo/downloader/httpx_downloader.py +259 -259
  26. crawlo/downloader/hybrid_downloader.py +212 -212
  27. crawlo/downloader/playwright_downloader.py +402 -402
  28. crawlo/downloader/selenium_downloader.py +472 -472
  29. crawlo/event.py +11 -11
  30. crawlo/exceptions.py +81 -81
  31. crawlo/extension/__init__.py +39 -39
  32. crawlo/extension/health_check.py +141 -141
  33. crawlo/extension/log_interval.py +57 -57
  34. crawlo/extension/log_stats.py +81 -81
  35. crawlo/extension/logging_extension.py +61 -52
  36. crawlo/extension/memory_monitor.py +104 -104
  37. crawlo/extension/performance_profiler.py +133 -133
  38. crawlo/extension/request_recorder.py +107 -107
  39. crawlo/factories/__init__.py +28 -0
  40. crawlo/factories/base.py +69 -0
  41. crawlo/factories/crawler.py +104 -0
  42. crawlo/factories/registry.py +85 -0
  43. crawlo/filters/__init__.py +154 -154
  44. crawlo/filters/aioredis_filter.py +257 -234
  45. crawlo/filters/memory_filter.py +269 -269
  46. crawlo/framework.py +292 -0
  47. crawlo/initialization/__init__.py +40 -0
  48. crawlo/initialization/built_in.py +426 -0
  49. crawlo/initialization/context.py +142 -0
  50. crawlo/initialization/core.py +194 -0
  51. crawlo/initialization/phases.py +149 -0
  52. crawlo/initialization/registry.py +146 -0
  53. crawlo/items/__init__.py +23 -23
  54. crawlo/items/base.py +23 -22
  55. crawlo/items/fields.py +52 -52
  56. crawlo/items/items.py +104 -104
  57. crawlo/logging/__init__.py +38 -0
  58. crawlo/logging/config.py +97 -0
  59. crawlo/logging/factory.py +129 -0
  60. crawlo/logging/manager.py +112 -0
  61. crawlo/middleware/__init__.py +21 -21
  62. crawlo/middleware/default_header.py +132 -132
  63. crawlo/middleware/download_delay.py +104 -104
  64. crawlo/middleware/middleware_manager.py +135 -135
  65. crawlo/middleware/offsite.py +123 -123
  66. crawlo/middleware/proxy.py +386 -386
  67. crawlo/middleware/request_ignore.py +86 -86
  68. crawlo/middleware/response_code.py +163 -163
  69. crawlo/middleware/response_filter.py +136 -136
  70. crawlo/middleware/retry.py +124 -124
  71. crawlo/middleware/simple_proxy.py +65 -65
  72. crawlo/mode_manager.py +212 -187
  73. crawlo/network/__init__.py +21 -21
  74. crawlo/network/request.py +379 -379
  75. crawlo/network/response.py +359 -359
  76. crawlo/pipelines/__init__.py +21 -21
  77. crawlo/pipelines/bloom_dedup_pipeline.py +156 -156
  78. crawlo/pipelines/console_pipeline.py +39 -39
  79. crawlo/pipelines/csv_pipeline.py +316 -316
  80. crawlo/pipelines/database_dedup_pipeline.py +222 -222
  81. crawlo/pipelines/json_pipeline.py +218 -218
  82. crawlo/pipelines/memory_dedup_pipeline.py +115 -115
  83. crawlo/pipelines/mongo_pipeline.py +131 -131
  84. crawlo/pipelines/mysql_pipeline.py +318 -318
  85. crawlo/pipelines/pipeline_manager.py +76 -75
  86. crawlo/pipelines/redis_dedup_pipeline.py +166 -166
  87. crawlo/project.py +327 -325
  88. crawlo/queue/pqueue.py +43 -37
  89. crawlo/queue/queue_manager.py +503 -379
  90. crawlo/queue/redis_priority_queue.py +326 -306
  91. crawlo/settings/__init__.py +7 -7
  92. crawlo/settings/default_settings.py +321 -225
  93. crawlo/settings/setting_manager.py +214 -198
  94. crawlo/spider/__init__.py +657 -639
  95. crawlo/stats_collector.py +73 -59
  96. crawlo/subscriber.py +129 -129
  97. crawlo/task_manager.py +139 -30
  98. crawlo/templates/crawlo.cfg.tmpl +10 -10
  99. crawlo/templates/project/__init__.py.tmpl +3 -3
  100. crawlo/templates/project/items.py.tmpl +17 -17
  101. crawlo/templates/project/middlewares.py.tmpl +118 -118
  102. crawlo/templates/project/pipelines.py.tmpl +96 -96
  103. crawlo/templates/project/settings.py.tmpl +168 -267
  104. crawlo/templates/project/settings_distributed.py.tmpl +167 -180
  105. crawlo/templates/project/settings_gentle.py.tmpl +167 -61
  106. crawlo/templates/project/settings_high_performance.py.tmpl +168 -131
  107. crawlo/templates/project/settings_minimal.py.tmpl +66 -35
  108. crawlo/templates/project/settings_simple.py.tmpl +165 -102
  109. crawlo/templates/project/spiders/__init__.py.tmpl +10 -6
  110. crawlo/templates/run.py.tmpl +34 -38
  111. crawlo/templates/spider/spider.py.tmpl +143 -143
  112. crawlo/templates/spiders_init.py.tmpl +10 -0
  113. crawlo/tools/__init__.py +200 -200
  114. crawlo/tools/anti_crawler.py +268 -268
  115. crawlo/tools/authenticated_proxy.py +240 -240
  116. crawlo/tools/data_formatter.py +225 -225
  117. crawlo/tools/data_validator.py +180 -180
  118. crawlo/tools/date_tools.py +289 -289
  119. crawlo/tools/distributed_coordinator.py +388 -388
  120. crawlo/tools/encoding_converter.py +127 -127
  121. crawlo/tools/network_diagnostic.py +365 -0
  122. crawlo/tools/request_tools.py +82 -82
  123. crawlo/tools/retry_mechanism.py +224 -224
  124. crawlo/tools/scenario_adapter.py +262 -262
  125. crawlo/tools/text_cleaner.py +232 -232
  126. crawlo/utils/__init__.py +34 -34
  127. crawlo/utils/batch_processor.py +259 -259
  128. crawlo/utils/class_loader.py +26 -0
  129. crawlo/utils/controlled_spider_mixin.py +439 -439
  130. crawlo/utils/db_helper.py +343 -343
  131. crawlo/utils/enhanced_error_handler.py +356 -356
  132. crawlo/utils/env_config.py +142 -142
  133. crawlo/utils/error_handler.py +165 -124
  134. crawlo/utils/func_tools.py +82 -82
  135. crawlo/utils/large_scale_config.py +286 -286
  136. crawlo/utils/large_scale_helper.py +344 -344
  137. crawlo/utils/log.py +44 -200
  138. crawlo/utils/performance_monitor.py +285 -285
  139. crawlo/utils/queue_helper.py +175 -175
  140. crawlo/utils/redis_connection_pool.py +388 -351
  141. crawlo/utils/redis_key_validator.py +198 -198
  142. crawlo/utils/request.py +267 -267
  143. crawlo/utils/request_serializer.py +225 -218
  144. crawlo/utils/spider_loader.py +61 -61
  145. crawlo/utils/system.py +11 -11
  146. crawlo/utils/tools.py +4 -4
  147. crawlo/utils/url.py +39 -39
  148. {crawlo-1.3.3.dist-info → crawlo-1.3.4.dist-info}/METADATA +1126 -1020
  149. crawlo-1.3.4.dist-info/RECORD +278 -0
  150. examples/__init__.py +7 -7
  151. tests/__init__.py +7 -7
  152. tests/advanced_tools_example.py +275 -275
  153. tests/authenticated_proxy_example.py +107 -107
  154. tests/baidu_performance_test.py +109 -0
  155. tests/baidu_test.py +60 -0
  156. tests/cleaners_example.py +160 -160
  157. tests/comprehensive_framework_test.py +213 -0
  158. tests/comprehensive_test.py +82 -0
  159. tests/comprehensive_testing_summary.md +187 -0
  160. tests/config_validation_demo.py +142 -142
  161. tests/controlled_spider_example.py +205 -205
  162. tests/date_tools_example.py +180 -180
  163. tests/debug_configure.py +70 -0
  164. tests/debug_framework_logger.py +85 -0
  165. tests/debug_log_levels.py +64 -0
  166. tests/debug_pipelines.py +66 -66
  167. tests/distributed_test.py +67 -0
  168. tests/distributed_test_debug.py +77 -0
  169. tests/dynamic_loading_example.py +523 -523
  170. tests/dynamic_loading_test.py +104 -104
  171. tests/env_config_example.py +133 -133
  172. tests/error_handling_example.py +171 -171
  173. tests/final_command_test_report.md +0 -0
  174. tests/final_comprehensive_test.py +152 -0
  175. tests/final_validation_test.py +183 -0
  176. tests/framework_performance_test.py +203 -0
  177. tests/optimized_performance_test.py +212 -0
  178. tests/performance_comparison.py +246 -0
  179. tests/queue_blocking_test.py +114 -0
  180. tests/queue_test.py +90 -0
  181. tests/redis_key_validation_demo.py +130 -130
  182. tests/request_params_example.py +150 -150
  183. tests/response_improvements_example.py +144 -144
  184. tests/scrapy_comparison/ofweek_scrapy.py +139 -0
  185. tests/scrapy_comparison/scrapy_test.py +134 -0
  186. tests/simple_command_test.py +120 -0
  187. tests/simple_crawlo_test.py +128 -0
  188. tests/simple_log_test.py +58 -0
  189. tests/simple_optimization_test.py +129 -0
  190. tests/simple_spider_test.py +50 -0
  191. tests/simple_test.py +48 -0
  192. tests/test_advanced_tools.py +148 -148
  193. tests/test_all_commands.py +231 -0
  194. tests/test_all_redis_key_configs.py +145 -145
  195. tests/test_authenticated_proxy.py +141 -141
  196. tests/test_batch_processor.py +179 -0
  197. tests/test_cleaners.py +54 -54
  198. tests/test_component_factory.py +175 -0
  199. tests/test_comprehensive.py +146 -146
  200. tests/test_config_consistency.py +80 -80
  201. tests/test_config_merge.py +152 -152
  202. tests/test_config_validator.py +182 -182
  203. tests/test_controlled_spider_mixin.py +80 -0
  204. tests/test_crawlo_proxy_integration.py +108 -108
  205. tests/test_date_tools.py +123 -123
  206. tests/test_default_header_middleware.py +158 -158
  207. tests/test_distributed.py +65 -65
  208. tests/test_double_crawlo_fix.py +207 -207
  209. tests/test_double_crawlo_fix_simple.py +124 -124
  210. tests/test_download_delay_middleware.py +221 -221
  211. tests/test_downloader_proxy_compatibility.py +268 -268
  212. tests/test_dynamic_downloaders_proxy.py +124 -124
  213. tests/test_dynamic_proxy.py +92 -92
  214. tests/test_dynamic_proxy_config.py +146 -146
  215. tests/test_dynamic_proxy_real.py +109 -109
  216. tests/test_edge_cases.py +303 -303
  217. tests/test_enhanced_error_handler.py +270 -270
  218. tests/test_enhanced_error_handler_comprehensive.py +246 -0
  219. tests/test_env_config.py +121 -121
  220. tests/test_error_handler_compatibility.py +112 -112
  221. tests/test_factories.py +253 -0
  222. tests/test_final_validation.py +153 -153
  223. tests/test_framework_env_usage.py +103 -103
  224. tests/test_framework_logger.py +67 -0
  225. tests/test_framework_startup.py +65 -0
  226. tests/test_integration.py +169 -169
  227. tests/test_item_dedup_redis_key.py +122 -122
  228. tests/test_large_scale_config.py +113 -0
  229. tests/test_large_scale_helper.py +236 -0
  230. tests/test_mode_change.py +73 -0
  231. tests/test_mode_consistency.py +51 -51
  232. tests/test_offsite_middleware.py +221 -221
  233. tests/test_parsel.py +29 -29
  234. tests/test_performance.py +327 -327
  235. tests/test_performance_monitor.py +116 -0
  236. tests/test_proxy_api.py +264 -264
  237. tests/test_proxy_health_check.py +32 -32
  238. tests/test_proxy_middleware.py +121 -121
  239. tests/test_proxy_middleware_enhanced.py +216 -216
  240. tests/test_proxy_middleware_integration.py +136 -136
  241. tests/test_proxy_middleware_refactored.py +184 -184
  242. tests/test_proxy_providers.py +56 -56
  243. tests/test_proxy_stats.py +19 -19
  244. tests/test_proxy_strategies.py +59 -59
  245. tests/test_queue_empty_check.py +42 -0
  246. tests/test_queue_manager_double_crawlo.py +173 -173
  247. tests/test_queue_manager_redis_key.py +176 -176
  248. tests/test_random_user_agent.py +72 -72
  249. tests/test_real_scenario_proxy.py +195 -195
  250. tests/test_redis_config.py +28 -28
  251. tests/test_redis_connection_pool.py +294 -294
  252. tests/test_redis_key_naming.py +181 -181
  253. tests/test_redis_key_validator.py +123 -123
  254. tests/test_redis_queue.py +224 -224
  255. tests/test_request_ignore_middleware.py +182 -182
  256. tests/test_request_params.py +111 -111
  257. tests/test_request_serialization.py +70 -70
  258. tests/test_response_code_middleware.py +349 -349
  259. tests/test_response_filter_middleware.py +427 -427
  260. tests/test_response_improvements.py +152 -152
  261. tests/test_retry_middleware.py +241 -241
  262. tests/test_scheduler.py +252 -252
  263. tests/test_scheduler_config_update.py +133 -133
  264. tests/test_simple_response.py +61 -61
  265. tests/test_telecom_spider_redis_key.py +205 -205
  266. tests/test_template_content.py +87 -87
  267. tests/test_template_redis_key.py +134 -134
  268. tests/test_tools.py +159 -159
  269. tests/test_user_agents.py +96 -96
  270. tests/tools_example.py +260 -260
  271. tests/untested_features_report.md +139 -0
  272. tests/verify_debug.py +52 -0
  273. tests/verify_distributed.py +117 -117
  274. tests/verify_log_fix.py +112 -0
  275. crawlo-1.3.3.dist-info/RECORD +0 -219
  276. tests/DOUBLE_CRAWLO_PREFIX_FIX_REPORT.md +0 -82
  277. {crawlo-1.3.3.dist-info → crawlo-1.3.4.dist-info}/WHEEL +0 -0
  278. {crawlo-1.3.3.dist-info → crawlo-1.3.4.dist-info}/entry_points.txt +0 -0
  279. {crawlo-1.3.3.dist-info → crawlo-1.3.4.dist-info}/top_level.txt +0 -0
@@ -1,273 +1,273 @@
1
- #!/usr/bin/python
2
- # -*- coding:UTF-8 -*-
3
- """
4
- Crawlo Downloader Module
5
- ========================
6
- 提供多种高性能异步下载器实现。
7
-
8
- 下载器类型:
9
- - AioHttpDownloader: 基于aiohttp的高性能下载器
10
- - CurlCffiDownloader: 支持浏览器指纹模拟的curl-cffi下载器
11
- - HttpXDownloader: 支持HTTP/2的httpx下载器
12
-
13
- 核心类:
14
- - DownloaderBase: 下载器基类
15
- - ActivateRequestManager: 活跃请求管理器
16
- """
17
- from abc import abstractmethod, ABCMeta
18
- from typing import Final, Set, Optional
19
- from contextlib import asynccontextmanager
20
-
21
- from crawlo.utils.log import get_logger
22
- from crawlo.middleware.middleware_manager import MiddlewareManager
23
-
24
-
25
- class ActivateRequestManager:
26
- """活跃请求管理器 - 跟踪和管理正在处理的请求"""
27
-
28
- def __init__(self):
29
- self._active: Final[Set] = set()
30
- self._total_requests: int = 0
31
- self._completed_requests: int = 0
32
- self._failed_requests: int = 0
33
-
34
- def add(self, request):
35
- """添加活跃请求"""
36
- self._active.add(request)
37
- self._total_requests += 1
38
- return request
39
-
40
- def remove(self, request, success: bool = True):
41
- """移除活跃请求并更新统计"""
42
- self._active.discard(request) # 使用discard避免KeyError
43
- if success:
44
- self._completed_requests += 1
45
- else:
46
- self._failed_requests += 1
47
-
48
- @asynccontextmanager
49
- async def __call__(self, request):
50
- """上下文管理器用法"""
51
- self.add(request)
52
- success = False
53
- try:
54
- yield request
55
- success = True
56
- except Exception:
57
- success = False
58
- raise
59
- finally:
60
- self.remove(request, success)
61
-
62
- def __len__(self):
63
- """返回当前活跃请求数"""
64
- return len(self._active)
65
-
66
- def get_stats(self) -> dict:
67
- """获取请求统计信息"""
68
- return {
69
- 'active_requests': len(self._active),
70
- 'total_requests': self._total_requests,
71
- 'completed_requests': self._completed_requests,
72
- 'failed_requests': self._failed_requests,
73
- 'success_rate': self._completed_requests / max(1, self._total_requests - len(self._active))
74
- }
75
-
76
- def reset_stats(self):
77
- """重置统计信息"""
78
- self._total_requests = 0
79
- self._completed_requests = 0
80
- self._failed_requests = 0
81
- # 注意:不清空 _active,因为可能有正在进行的请求
82
-
83
-
84
- class DownloaderMeta(ABCMeta):
85
- def __subclasscheck__(self, subclass):
86
- required_methods = ('fetch', 'download', 'create_instance', 'close')
87
- is_subclass = all(
88
- hasattr(subclass, method) and callable(getattr(subclass, method, None)) for method in required_methods
89
- )
90
- return is_subclass
91
-
92
-
93
- class DownloaderBase(metaclass=DownloaderMeta):
94
- """
95
- 下载器基类 - 提供通用的下载器功能和接口
96
-
97
- 所有下载器实现都应该继承此基类。
98
- """
99
-
100
- def __init__(self, crawler):
101
- self.crawler = crawler
102
- self._active = ActivateRequestManager()
103
- self.middleware: Optional[MiddlewareManager] = None
104
- self.logger = get_logger(self.__class__.__name__, crawler.settings.get("LOG_LEVEL"))
105
- self._closed = False
106
- self._stats_enabled = crawler.settings.get_bool("DOWNLOADER_STATS", True)
107
-
108
- @classmethod
109
- def create_instance(cls, *args, **kwargs):
110
- """创建下载器实例"""
111
- return cls(*args, **kwargs)
112
-
113
- def open(self) -> None:
114
- """初始化下载器"""
115
- if self._closed:
116
- raise RuntimeError(f"{self.__class__.__name__} 已关闭,无法重新打开")
117
-
118
- # 获取下载器类的完整路径
119
- downloader_class = f"{type(self).__module__}.{type(self).__name__}"
120
-
121
- # 输出启用的下载器信息(类似MiddlewareManager的格式)
122
- self.logger.info(f"enabled downloader: \n {downloader_class}")
123
-
124
- # 输出下载器配置摘要
125
- self.logger.debug(
126
- f"{self.crawler.spider} <下载器类:{downloader_class}> "
127
- f"<并发数:{self.crawler.settings.get_int('CONCURRENCY')}>"
128
- )
129
-
130
- try:
131
- self.middleware = MiddlewareManager.create_instance(self.crawler)
132
- self.logger.debug(f"{self.__class__.__name__} 中间件初始化完成")
133
- except Exception as e:
134
- self.logger.error(f"中间件初始化失败: {e}")
135
- raise
136
-
137
- async def fetch(self, request) -> Optional['Response']:
138
- """获取请求响应(经过中间件处理)"""
139
- if self._closed:
140
- raise RuntimeError(f"{self.__class__.__name__} 已关闭")
141
-
142
- if not self.middleware:
143
- raise RuntimeError("中间件未初始化")
144
-
145
- async with self._active(request):
146
- try:
147
- response = await self.middleware.download(request)
148
- return response
149
- except Exception as e:
150
- self.logger.error(f"下载请求 {request.url} 失败: {e}")
151
- raise
152
-
153
- @abstractmethod
154
- async def download(self, request) -> 'Response':
155
- """子类必须实现的下载方法"""
156
- pass
157
-
158
- async def close(self) -> None:
159
- """关闭下载器并清理资源"""
160
- if not self._closed:
161
- self._closed = True
162
- if self._stats_enabled:
163
- stats = self.get_stats()
164
- self.logger.info(f"{self.__class__.__name__} 统计: {stats}")
165
- self.logger.debug(f"{self.__class__.__name__} 已关闭")
166
-
167
- def idle(self) -> bool:
168
- """检查是否空闲(无活跃请求)"""
169
- return len(self._active) == 0
170
-
171
- def __len__(self) -> int:
172
- """返回活跃请求数"""
173
- return len(self._active)
174
-
175
- def get_stats(self) -> dict:
176
- """获取下载器统计信息"""
177
- base_stats = {
178
- 'downloader_class': self.__class__.__name__,
179
- 'is_idle': self.idle(),
180
- 'is_closed': self._closed
181
- }
182
-
183
- if self._stats_enabled:
184
- base_stats.update(self._active.get_stats())
185
-
186
- return base_stats
187
-
188
- def reset_stats(self):
189
- """重置统计信息"""
190
- if self._stats_enabled:
191
- self._active.reset_stats()
192
-
193
- def health_check(self) -> dict:
194
- """健康检查"""
195
- return {
196
- 'status': 'healthy' if not self._closed and self.middleware else 'unhealthy',
197
- 'active_requests': len(self._active),
198
- 'middleware_ready': self.middleware is not None,
199
- 'closed': self._closed
200
- }
201
-
202
-
203
- # 导入具体的下载器实现
204
- try:
205
- from .aiohttp_downloader import AioHttpDownloader
206
- except ImportError:
207
- AioHttpDownloader = None
208
-
209
- try:
210
- from .cffi_downloader import CurlCffiDownloader
211
- except ImportError:
212
- CurlCffiDownloader = None
213
-
214
- try:
215
- from .httpx_downloader import HttpXDownloader
216
- except ImportError:
217
- HttpXDownloader = None
218
-
219
- try:
220
- from .selenium_downloader import SeleniumDownloader
221
- except ImportError:
222
- SeleniumDownloader = None
223
-
224
- try:
225
- from .playwright_downloader import PlaywrightDownloader
226
- except ImportError:
227
- PlaywrightDownloader = None
228
-
229
- try:
230
- from .hybrid_downloader import HybridDownloader
231
- except ImportError:
232
- HybridDownloader = None
233
-
234
- # 导出所有可用的类
235
- __all__ = [
236
- 'DownloaderBase',
237
- 'DownloaderMeta',
238
- 'ActivateRequestManager',
239
- ]
240
-
241
- # 添加可用的下载器
242
- if AioHttpDownloader:
243
- __all__.append('AioHttpDownloader')
244
- if CurlCffiDownloader:
245
- __all__.append('CurlCffiDownloader')
246
- if HttpXDownloader:
247
- __all__.append('HttpXDownloader')
248
- if SeleniumDownloader:
249
- __all__.append('SeleniumDownloader')
250
- if PlaywrightDownloader:
251
- __all__.append('PlaywrightDownloader')
252
- if HybridDownloader:
253
- __all__.append('HybridDownloader')
254
-
255
- # 提供便捷的下载器映射
256
- DOWNLOADER_MAP = {
257
- 'aiohttp': AioHttpDownloader,
258
- 'httpx': HttpXDownloader,
259
- 'curl_cffi': CurlCffiDownloader,
260
- 'cffi': CurlCffiDownloader, # 别名
261
- 'selenium': SeleniumDownloader,
262
- 'playwright': PlaywrightDownloader,
263
- 'hybrid': HybridDownloader,
264
- }
265
-
266
- # 过滤掉不可用的下载器
267
- DOWNLOADER_MAP = {k: v for k, v in DOWNLOADER_MAP.items() if v is not None}
268
-
269
- def get_downloader_class(name: str):
270
- """根据名称获取下载器类"""
271
- if name in DOWNLOADER_MAP:
272
- return DOWNLOADER_MAP[name]
273
- raise ValueError(f"未知的下载器类型: {name}。可用类型: {list(DOWNLOADER_MAP.keys())}")
1
+ #!/usr/bin/python
2
+ # -*- coding:UTF-8 -*-
3
+ """
4
+ Crawlo Downloader Module
5
+ ========================
6
+ 提供多种高性能异步下载器实现。
7
+
8
+ 下载器类型:
9
+ - AioHttpDownloader: 基于aiohttp的高性能下载器
10
+ - CurlCffiDownloader: 支持浏览器指纹模拟的curl-cffi下载器
11
+ - HttpXDownloader: 支持HTTP/2的httpx下载器
12
+
13
+ 核心类:
14
+ - DownloaderBase: 下载器基类
15
+ - ActivateRequestManager: 活跃请求管理器
16
+ """
17
+ from abc import abstractmethod, ABCMeta
18
+ from typing import Final, Set, Optional
19
+ from contextlib import asynccontextmanager
20
+
21
+ from crawlo.utils.log import get_logger
22
+ from crawlo.middleware.middleware_manager import MiddlewareManager
23
+
24
+
25
+ class ActivateRequestManager:
26
+ """活跃请求管理器 - 跟踪和管理正在处理的请求"""
27
+
28
+ def __init__(self):
29
+ self._active: Final[Set] = set()
30
+ self._total_requests: int = 0
31
+ self._completed_requests: int = 0
32
+ self._failed_requests: int = 0
33
+
34
+ def add(self, request):
35
+ """添加活跃请求"""
36
+ self._active.add(request)
37
+ self._total_requests += 1
38
+ return request
39
+
40
+ def remove(self, request, success: bool = True):
41
+ """移除活跃请求并更新统计"""
42
+ self._active.discard(request) # 使用discard避免KeyError
43
+ if success:
44
+ self._completed_requests += 1
45
+ else:
46
+ self._failed_requests += 1
47
+
48
+ @asynccontextmanager
49
+ async def __call__(self, request):
50
+ """上下文管理器用法"""
51
+ self.add(request)
52
+ success = False
53
+ try:
54
+ yield request
55
+ success = True
56
+ except Exception:
57
+ success = False
58
+ raise
59
+ finally:
60
+ self.remove(request, success)
61
+
62
+ def __len__(self):
63
+ """返回当前活跃请求数"""
64
+ return len(self._active)
65
+
66
+ def get_stats(self) -> dict:
67
+ """获取请求统计信息"""
68
+ return {
69
+ 'active_requests': len(self._active),
70
+ 'total_requests': self._total_requests,
71
+ 'completed_requests': self._completed_requests,
72
+ 'failed_requests': self._failed_requests,
73
+ 'success_rate': self._completed_requests / max(1, self._total_requests - len(self._active))
74
+ }
75
+
76
+ def reset_stats(self):
77
+ """重置统计信息"""
78
+ self._total_requests = 0
79
+ self._completed_requests = 0
80
+ self._failed_requests = 0
81
+ # 注意:不清空 _active,因为可能有正在进行的请求
82
+
83
+
84
+ class DownloaderMeta(ABCMeta):
85
+ def __subclasscheck__(self, subclass):
86
+ required_methods = ('fetch', 'download', 'create_instance', 'close')
87
+ is_subclass = all(
88
+ hasattr(subclass, method) and callable(getattr(subclass, method, None)) for method in required_methods
89
+ )
90
+ return is_subclass
91
+
92
+
93
+ class DownloaderBase(metaclass=DownloaderMeta):
94
+ """
95
+ 下载器基类 - 提供通用的下载器功能和接口
96
+
97
+ 所有下载器实现都应该继承此基类。
98
+ """
99
+
100
+ def __init__(self, crawler):
101
+ self.crawler = crawler
102
+ self._active = ActivateRequestManager()
103
+ self.middleware: Optional[MiddlewareManager] = None
104
+ self.logger = get_logger(self.__class__.__name__, crawler.settings.get("LOG_LEVEL"))
105
+ self._closed = False
106
+ self._stats_enabled = crawler.settings.get_bool("DOWNLOADER_STATS", True)
107
+
108
+ @classmethod
109
+ def create_instance(cls, *args, **kwargs):
110
+ """创建下载器实例"""
111
+ return cls(*args, **kwargs)
112
+
113
+ def open(self) -> None:
114
+ """初始化下载器"""
115
+ if self._closed:
116
+ raise RuntimeError(f"{self.__class__.__name__} 已关闭,无法重新打开")
117
+
118
+ # 获取下载器类的完整路径
119
+ downloader_class = f"{type(self).__module__}.{type(self).__name__}"
120
+
121
+ # 输出启用的下载器信息(类似MiddlewareManager的格式)
122
+ self.logger.info(f"enabled downloader: \n {downloader_class}")
123
+
124
+ # 输出下载器配置摘要
125
+ self.logger.debug(
126
+ f"{self.crawler.spider} <下载器类:{downloader_class}> "
127
+ f"<并发数:{self.crawler.settings.get_int('CONCURRENCY')}>"
128
+ )
129
+
130
+ try:
131
+ self.middleware = MiddlewareManager.create_instance(self.crawler)
132
+ self.logger.debug(f"{self.__class__.__name__} 中间件初始化完成")
133
+ except Exception as e:
134
+ self.logger.error(f"中间件初始化失败: {e}")
135
+ raise
136
+
137
+ async def fetch(self, request) -> Optional['Response']:
138
+ """获取请求响应(经过中间件处理)"""
139
+ if self._closed:
140
+ raise RuntimeError(f"{self.__class__.__name__} 已关闭")
141
+
142
+ if not self.middleware:
143
+ raise RuntimeError("中间件未初始化")
144
+
145
+ async with self._active(request):
146
+ try:
147
+ response = await self.middleware.download(request)
148
+ return response
149
+ except Exception as e:
150
+ self.logger.error(f"下载请求 {request.url} 失败: {e}")
151
+ raise
152
+
153
+ @abstractmethod
154
+ async def download(self, request) -> 'Response':
155
+ """子类必须实现的下载方法"""
156
+ pass
157
+
158
+ async def close(self) -> None:
159
+ """关闭下载器并清理资源"""
160
+ if not self._closed:
161
+ self._closed = True
162
+ if self._stats_enabled:
163
+ stats = self.get_stats()
164
+ self.logger.info(f"{self.__class__.__name__} 统计: {stats}")
165
+ self.logger.debug(f"{self.__class__.__name__} 已关闭")
166
+
167
+ def idle(self) -> bool:
168
+ """检查是否空闲(无活跃请求)"""
169
+ return len(self._active) == 0
170
+
171
+ def __len__(self) -> int:
172
+ """返回活跃请求数"""
173
+ return len(self._active)
174
+
175
+ def get_stats(self) -> dict:
176
+ """获取下载器统计信息"""
177
+ base_stats = {
178
+ 'downloader_class': self.__class__.__name__,
179
+ 'is_idle': self.idle(),
180
+ 'is_closed': self._closed
181
+ }
182
+
183
+ if self._stats_enabled:
184
+ base_stats.update(self._active.get_stats())
185
+
186
+ return base_stats
187
+
188
+ def reset_stats(self):
189
+ """重置统计信息"""
190
+ if self._stats_enabled:
191
+ self._active.reset_stats()
192
+
193
+ def health_check(self) -> dict:
194
+ """健康检查"""
195
+ return {
196
+ 'status': 'healthy' if not self._closed and self.middleware else 'unhealthy',
197
+ 'active_requests': len(self._active),
198
+ 'middleware_ready': self.middleware is not None,
199
+ 'closed': self._closed
200
+ }
201
+
202
+
203
+ # 导入具体的下载器实现
204
+ try:
205
+ from .aiohttp_downloader import AioHttpDownloader
206
+ except ImportError:
207
+ AioHttpDownloader = None
208
+
209
+ try:
210
+ from .cffi_downloader import CurlCffiDownloader
211
+ except ImportError:
212
+ CurlCffiDownloader = None
213
+
214
+ try:
215
+ from .httpx_downloader import HttpXDownloader
216
+ except ImportError:
217
+ HttpXDownloader = None
218
+
219
+ try:
220
+ from .selenium_downloader import SeleniumDownloader
221
+ except ImportError:
222
+ SeleniumDownloader = None
223
+
224
+ try:
225
+ from .playwright_downloader import PlaywrightDownloader
226
+ except ImportError:
227
+ PlaywrightDownloader = None
228
+
229
+ try:
230
+ from .hybrid_downloader import HybridDownloader
231
+ except ImportError:
232
+ HybridDownloader = None
233
+
234
+ # 导出所有可用的类
235
+ __all__ = [
236
+ 'DownloaderBase',
237
+ 'DownloaderMeta',
238
+ 'ActivateRequestManager',
239
+ ]
240
+
241
+ # 添加可用的下载器
242
+ if AioHttpDownloader:
243
+ __all__.append('AioHttpDownloader')
244
+ if CurlCffiDownloader:
245
+ __all__.append('CurlCffiDownloader')
246
+ if HttpXDownloader:
247
+ __all__.append('HttpXDownloader')
248
+ if SeleniumDownloader:
249
+ __all__.append('SeleniumDownloader')
250
+ if PlaywrightDownloader:
251
+ __all__.append('PlaywrightDownloader')
252
+ if HybridDownloader:
253
+ __all__.append('HybridDownloader')
254
+
255
+ # 提供便捷的下载器映射
256
+ DOWNLOADER_MAP = {
257
+ 'aiohttp': AioHttpDownloader,
258
+ 'httpx': HttpXDownloader,
259
+ 'curl_cffi': CurlCffiDownloader,
260
+ 'cffi': CurlCffiDownloader, # 别名
261
+ 'selenium': SeleniumDownloader,
262
+ 'playwright': PlaywrightDownloader,
263
+ 'hybrid': HybridDownloader,
264
+ }
265
+
266
+ # 过滤掉不可用的下载器
267
+ DOWNLOADER_MAP = {k: v for k, v in DOWNLOADER_MAP.items() if v is not None}
268
+
269
+ def get_downloader_class(name: str):
270
+ """根据名称获取下载器类"""
271
+ if name in DOWNLOADER_MAP:
272
+ return DOWNLOADER_MAP[name]
273
+ raise ValueError(f"未知的下载器类型: {name}。可用类型: {list(DOWNLOADER_MAP.keys())}")