crawlo 1.3.3__py3-none-any.whl → 1.3.5__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 (289) 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 +325 -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 +80 -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.5.dist-info}/METADATA +1126 -1020
  149. crawlo-1.3.5.dist-info/RECORD +288 -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_config.py +127 -0
  166. tests/debug_log_levels.py +64 -0
  167. tests/debug_pipelines.py +66 -66
  168. tests/detailed_log_test.py +234 -0
  169. tests/distributed_test.py +67 -0
  170. tests/distributed_test_debug.py +77 -0
  171. tests/dynamic_loading_example.py +523 -523
  172. tests/dynamic_loading_test.py +104 -104
  173. tests/env_config_example.py +133 -133
  174. tests/error_handling_example.py +171 -171
  175. tests/final_command_test_report.md +0 -0
  176. tests/final_comprehensive_test.py +152 -0
  177. tests/final_log_test.py +261 -0
  178. tests/final_validation_test.py +183 -0
  179. tests/fix_log_test.py +143 -0
  180. tests/framework_performance_test.py +203 -0
  181. tests/log_buffering_test.py +112 -0
  182. tests/log_generation_timing_test.py +154 -0
  183. tests/optimized_performance_test.py +212 -0
  184. tests/performance_comparison.py +246 -0
  185. tests/queue_blocking_test.py +114 -0
  186. tests/queue_test.py +90 -0
  187. tests/redis_key_validation_demo.py +130 -130
  188. tests/request_params_example.py +150 -150
  189. tests/response_improvements_example.py +144 -144
  190. tests/scrapy_comparison/ofweek_scrapy.py +139 -0
  191. tests/scrapy_comparison/scrapy_test.py +134 -0
  192. tests/simple_command_test.py +120 -0
  193. tests/simple_crawlo_test.py +128 -0
  194. tests/simple_log_test.py +58 -0
  195. tests/simple_log_test2.py +138 -0
  196. tests/simple_optimization_test.py +129 -0
  197. tests/simple_spider_test.py +50 -0
  198. tests/simple_test.py +48 -0
  199. tests/spider_log_timing_test.py +178 -0
  200. tests/test_advanced_tools.py +148 -148
  201. tests/test_all_commands.py +231 -0
  202. tests/test_all_redis_key_configs.py +145 -145
  203. tests/test_authenticated_proxy.py +141 -141
  204. tests/test_batch_processor.py +179 -0
  205. tests/test_cleaners.py +54 -54
  206. tests/test_component_factory.py +175 -0
  207. tests/test_comprehensive.py +146 -146
  208. tests/test_config_consistency.py +80 -80
  209. tests/test_config_merge.py +152 -152
  210. tests/test_config_validator.py +182 -182
  211. tests/test_controlled_spider_mixin.py +80 -0
  212. tests/test_crawlo_proxy_integration.py +108 -108
  213. tests/test_date_tools.py +123 -123
  214. tests/test_default_header_middleware.py +158 -158
  215. tests/test_distributed.py +65 -65
  216. tests/test_double_crawlo_fix.py +207 -207
  217. tests/test_double_crawlo_fix_simple.py +124 -124
  218. tests/test_download_delay_middleware.py +221 -221
  219. tests/test_downloader_proxy_compatibility.py +268 -268
  220. tests/test_dynamic_downloaders_proxy.py +124 -124
  221. tests/test_dynamic_proxy.py +92 -92
  222. tests/test_dynamic_proxy_config.py +146 -146
  223. tests/test_dynamic_proxy_real.py +109 -109
  224. tests/test_edge_cases.py +303 -303
  225. tests/test_enhanced_error_handler.py +270 -270
  226. tests/test_enhanced_error_handler_comprehensive.py +246 -0
  227. tests/test_env_config.py +121 -121
  228. tests/test_error_handler_compatibility.py +112 -112
  229. tests/test_factories.py +253 -0
  230. tests/test_final_validation.py +153 -153
  231. tests/test_framework_env_usage.py +103 -103
  232. tests/test_framework_logger.py +67 -0
  233. tests/test_framework_startup.py +65 -0
  234. tests/test_get_component_logger.py +84 -0
  235. tests/test_integration.py +169 -169
  236. tests/test_item_dedup_redis_key.py +122 -122
  237. tests/test_large_scale_config.py +113 -0
  238. tests/test_large_scale_helper.py +236 -0
  239. tests/test_logging_system.py +283 -0
  240. tests/test_mode_change.py +73 -0
  241. tests/test_mode_consistency.py +51 -51
  242. tests/test_offsite_middleware.py +221 -221
  243. tests/test_parsel.py +29 -29
  244. tests/test_performance.py +327 -327
  245. tests/test_performance_monitor.py +116 -0
  246. tests/test_proxy_api.py +264 -264
  247. tests/test_proxy_health_check.py +32 -32
  248. tests/test_proxy_middleware.py +121 -121
  249. tests/test_proxy_middleware_enhanced.py +216 -216
  250. tests/test_proxy_middleware_integration.py +136 -136
  251. tests/test_proxy_middleware_refactored.py +184 -184
  252. tests/test_proxy_providers.py +56 -56
  253. tests/test_proxy_stats.py +19 -19
  254. tests/test_proxy_strategies.py +59 -59
  255. tests/test_queue_empty_check.py +42 -0
  256. tests/test_queue_manager_double_crawlo.py +173 -173
  257. tests/test_queue_manager_redis_key.py +176 -176
  258. tests/test_random_user_agent.py +72 -72
  259. tests/test_real_scenario_proxy.py +195 -195
  260. tests/test_redis_config.py +28 -28
  261. tests/test_redis_connection_pool.py +294 -294
  262. tests/test_redis_key_naming.py +181 -181
  263. tests/test_redis_key_validator.py +123 -123
  264. tests/test_redis_queue.py +224 -224
  265. tests/test_request_ignore_middleware.py +182 -182
  266. tests/test_request_params.py +111 -111
  267. tests/test_request_serialization.py +70 -70
  268. tests/test_response_code_middleware.py +349 -349
  269. tests/test_response_filter_middleware.py +427 -427
  270. tests/test_response_improvements.py +152 -152
  271. tests/test_retry_middleware.py +241 -241
  272. tests/test_scheduler.py +252 -252
  273. tests/test_scheduler_config_update.py +133 -133
  274. tests/test_simple_response.py +61 -61
  275. tests/test_telecom_spider_redis_key.py +205 -205
  276. tests/test_template_content.py +87 -87
  277. tests/test_template_redis_key.py +134 -134
  278. tests/test_tools.py +159 -159
  279. tests/test_user_agents.py +96 -96
  280. tests/tools_example.py +260 -260
  281. tests/untested_features_report.md +139 -0
  282. tests/verify_debug.py +52 -0
  283. tests/verify_distributed.py +117 -117
  284. tests/verify_log_fix.py +112 -0
  285. crawlo-1.3.3.dist-info/RECORD +0 -219
  286. tests/DOUBLE_CRAWLO_PREFIX_FIX_REPORT.md +0 -82
  287. {crawlo-1.3.3.dist-info → crawlo-1.3.5.dist-info}/WHEEL +0 -0
  288. {crawlo-1.3.3.dist-info → crawlo-1.3.5.dist-info}/entry_points.txt +0 -0
  289. {crawlo-1.3.3.dist-info → crawlo-1.3.5.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())}")