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,276 +1,276 @@
1
- #!/usr/bin/python
2
- # -*- coding: UTF-8 -*-
3
- """
4
- Crawlo框架高级工具使用示例
5
- """
6
- from crawlo.tools import (
7
- # 数据处理工具
8
- clean_text,
9
- format_currency,
10
- validate_email,
11
- validate_url,
12
- check_data_integrity,
13
-
14
- # 重试机制
15
- RetryMechanism,
16
- retry,
17
- exponential_backoff,
18
-
19
- # 反爬虫应对工具
20
- AntiCrawler,
21
- rotate_proxy,
22
- handle_captcha,
23
- detect_rate_limiting,
24
-
25
- # 带认证代理工具
26
- AuthenticatedProxy,
27
- create_proxy_config,
28
- format_proxy_for_request,
29
-
30
- # 分布式协调工具
31
- generate_pagination_tasks,
32
- distribute_tasks,
33
- DistributedCoordinator
34
- )
35
-
36
-
37
- def demo_data_processing_tools():
38
- """演示数据处理工具的使用"""
39
- print("=== 数据处理工具演示 ===\n")
40
-
41
- # 数据清洗
42
- dirty_text = "<p>这是一个&nbsp;<b>测试</b>&amp;文本</p>"
43
- clean_result = clean_text(dirty_text)
44
- print(f"清洗文本: {dirty_text} -> {clean_result}")
45
-
46
- # 数据格式化
47
- price = 1234.567
48
- formatted_price = format_currency(price, "¥", 2)
49
- print(f"格式化货币: {price} -> {formatted_price}")
50
-
51
- # 字段验证
52
- email = "test@example.com"
53
- is_valid_email = validate_email(email)
54
- print(f"验证邮箱: {email} -> {'有效' if is_valid_email else '无效'}")
55
-
56
- url = "https://example.com"
57
- is_valid_url = validate_url(url)
58
- print(f"验证URL: {url} -> {'有效' if is_valid_url else '无效'}")
59
-
60
- # 数据完整性检查
61
- data = {
62
- "name": "张三",
63
- "email": "test@example.com",
64
- "phone": "13812345678"
65
- }
66
- required_fields = ["name", "email", "phone", "address"]
67
- integrity_result = check_data_integrity(data, required_fields)
68
- print(f"数据完整性检查: {integrity_result}")
69
-
70
- print()
71
-
72
-
73
- def demo_retry_mechanism():
74
- """演示重试机制的使用"""
75
- print("=== 重试机制演示 ===\n")
76
-
77
- # 指数退避
78
- for attempt in range(5):
79
- delay = exponential_backoff(attempt)
80
- print(f"重试次数 {attempt}: 延迟 {delay:.2f} 秒")
81
-
82
- # 重试装饰器示例
83
- @retry(max_retries=3)
84
- def unreliable_function():
85
- import random
86
- if random.random() < 0.7: # 70%概率失败
87
- raise ConnectionError("网络连接失败")
88
- return "成功执行"
89
-
90
- try:
91
- result = unreliable_function()
92
- print(f"函数执行结果: {result}")
93
- except Exception as e:
94
- print(f"函数执行失败: {e}")
95
-
96
- print()
97
-
98
-
99
- def demo_anti_crawler_tools():
100
- """演示反爬虫应对工具的使用"""
101
- print("=== 反爬虫应对工具演示 ===\n")
102
-
103
- # 反爬虫工具
104
- anti_crawler = AntiCrawler()
105
-
106
- # 获取随机User-Agent
107
- user_agent = anti_crawler.get_random_user_agent()
108
- print(f"随机User-Agent: {user_agent[:50]}...")
109
-
110
- # 轮换代理
111
- proxy = anti_crawler.rotate_proxy()
112
- print(f"轮换代理: {proxy}")
113
-
114
- # 检测验证码
115
- response_with_captcha = "请输入验证码进行验证"
116
- has_captcha = anti_crawler.handle_captcha(response_with_captcha)
117
- print(f"检测验证码: {response_with_captcha} -> {'需要验证码' if has_captcha else '无需验证码'}")
118
-
119
- # 检测频率限制
120
- status_code = 429 # Too Many Requests
121
- response_headers = {"Retry-After": "60"}
122
- is_rate_limited = anti_crawler.detect_rate_limiting(status_code, response_headers)
123
- print(f"检测频率限制: 状态码{status_code} -> {'被限制' if is_rate_limited else '未限制'}")
124
-
125
- print()
126
-
127
-
128
- def demo_authenticated_proxy_tools():
129
- """演示带认证代理工具的使用"""
130
- print("=== 带认证代理工具演示 ===\n")
131
-
132
- # 不同类型的代理URL
133
- proxy_urls = [
134
- "http://user:pass@proxy1.example.com:8080", # 带认证HTTP代理
135
- "https://username:password@proxy2.example.com:443", # 带认证HTTPS代理
136
- "http://proxy3.example.com:8080" # 不带认证代理
137
- ]
138
-
139
- for proxy_url in proxy_urls:
140
- print(f"处理代理: {proxy_url}")
141
-
142
- # 创建代理对象
143
- proxy = AuthenticatedProxy(proxy_url)
144
-
145
- # 为不同下载器格式化代理配置
146
- for downloader in ["aiohttp", "httpx", "curl_cffi"]:
147
- config = create_proxy_config(proxy_url)
148
- formatted = format_proxy_for_request(config, downloader)
149
- print(f" {downloader}格式: {formatted}")
150
-
151
- print()
152
-
153
-
154
- def demo_distributed_coordinator_tools():
155
- """演示分布式协调工具的使用"""
156
- print("=== 分布式协调工具演示 ===\n")
157
-
158
- # 生成分页任务
159
- base_url = "https://example.com/products"
160
- pagination_tasks = generate_pagination_tasks(base_url, 1, 5)
161
- print(f"生成分页任务 ({len(pagination_tasks)} 个):")
162
- for i, task in enumerate(pagination_tasks[:3]): # 只显示前3个
163
- print(f" {i+1}. {task}")
164
- if len(pagination_tasks) > 3:
165
- print(f" ... 还有 {len(pagination_tasks) - 3} 个任务")
166
-
167
- # 任务分发
168
- tasks = list(range(1, 21)) # 20个任务
169
- distributed = distribute_tasks(tasks, 4) # 分发给4个工作节点
170
- print(f"\n任务分发 (20个任务分发给4个工作节点):")
171
- for i, worker_tasks in enumerate(distributed):
172
- print(f" 工作节点 {i+1}: {len(worker_tasks)} 个任务 -> {worker_tasks}")
173
-
174
- # 分布式协调器
175
- coordinator = DistributedCoordinator()
176
- cluster_info = coordinator.get_cluster_info()
177
- print(f"\n集群信息: {cluster_info}")
178
-
179
- print()
180
-
181
-
182
- def demo_in_spider():
183
- """演示在爬虫中使用高级工具"""
184
- print("=== 在爬虫中使用高级工具 ===\n")
185
- print("在爬虫项目中,您可以这样使用高级工具:")
186
- print("""
187
- import asyncio
188
- from crawlo import Spider, Request
189
- from crawlo.tools import (
190
- clean_text,
191
- validate_email,
192
- AntiCrawler,
193
- DistributedCoordinator,
194
- retry,
195
- AuthenticatedProxy
196
- )
197
-
198
- class AdvancedSpider(Spider):
199
- def __init__(self):
200
- super().__init__()
201
- self.anti_crawler = AntiCrawler()
202
- self.coordinator = DistributedCoordinator()
203
- # 代理列表
204
- self.proxy_urls = [
205
- "http://user1:pass1@proxy1.example.com:8080",
206
- "http://user2:pass2@proxy2.example.com:8080",
207
- "http://proxy3.example.com:8080" # 不带认证
208
- ]
209
-
210
- def start_requests(self):
211
- # 生成分页任务
212
- base_url = "https://api.example.com/products"
213
- pagination_tasks = self.coordinator.generate_pagination_tasks(base_url, 1, 100)
214
-
215
- for i, url in enumerate(pagination_tasks):
216
- # 轮换使用代理
217
- proxy_url = self.proxy_urls[i % len(self.proxy_urls)]
218
- proxy = AuthenticatedProxy(proxy_url)
219
-
220
- request = Request(url, callback=self.parse)
221
-
222
- # 根据下载器类型设置代理
223
- downloader_type = self.crawler.settings.get("DOWNLOADER_TYPE", "aiohttp")
224
- if downloader_type == "aiohttp":
225
- request.proxy = proxy.clean_url
226
- auth = proxy.get_auth_credentials()
227
- if auth:
228
- request.meta["proxy_auth"] = auth
229
- elif downloader_type == "httpx":
230
- request.proxy = proxy.clean_url
231
- elif downloader_type == "curl_cffi":
232
- request.proxy = proxy.proxy_dict
233
- auth_header = proxy.get_auth_header()
234
- if auth_header:
235
- request.headers["Proxy-Authorization"] = auth_header
236
-
237
- yield request
238
-
239
- @retry(max_retries=3)
240
- async def parse(self, response):
241
- # 检查是否遇到验证码
242
- if self.anti_crawler.handle_captcha(response.text):
243
- # 处理验证码逻辑
244
- print("遇到验证码,需要处理")
245
- return
246
-
247
- # 提取数据
248
- products = response.css('.product-item')
249
- for product in products:
250
- name = product.css('.product-name::text').get()
251
- price_text = product.css('.price::text').get()
252
- email = product.css('.contact-email::text').get()
253
-
254
- # 数据清洗和验证
255
- clean_name = clean_text(name) if name else None
256
- clean_price = clean_text(price_text) if price_text else None
257
- is_valid_email = validate_email(email) if email else False
258
-
259
- # 检查数据是否重复
260
- if not await self.coordinator.is_duplicate({"name": clean_name, "price": clean_price}):
261
- # 添加到去重集合
262
- await self.coordinator.add_to_dedup({"name": clean_name, "price": clean_price})
263
-
264
- # 处理产品数据...
265
- pass
266
- """)
267
-
268
-
269
- if __name__ == '__main__':
270
- # 运行演示
271
- demo_data_processing_tools()
272
- demo_retry_mechanism()
273
- demo_anti_crawler_tools()
274
- demo_authenticated_proxy_tools()
275
- demo_distributed_coordinator_tools()
1
+ #!/usr/bin/python
2
+ # -*- coding: UTF-8 -*-
3
+ """
4
+ Crawlo框架高级工具使用示例
5
+ """
6
+ from crawlo.tools import (
7
+ # 数据处理工具
8
+ clean_text,
9
+ format_currency,
10
+ validate_email,
11
+ validate_url,
12
+ check_data_integrity,
13
+
14
+ # 重试机制
15
+ RetryMechanism,
16
+ retry,
17
+ exponential_backoff,
18
+
19
+ # 反爬虫应对工具
20
+ AntiCrawler,
21
+ rotate_proxy,
22
+ handle_captcha,
23
+ detect_rate_limiting,
24
+
25
+ # 带认证代理工具
26
+ AuthenticatedProxy,
27
+ create_proxy_config,
28
+ format_proxy_for_request,
29
+
30
+ # 分布式协调工具
31
+ generate_pagination_tasks,
32
+ distribute_tasks,
33
+ DistributedCoordinator
34
+ )
35
+
36
+
37
+ def demo_data_processing_tools():
38
+ """演示数据处理工具的使用"""
39
+ print("=== 数据处理工具演示 ===\n")
40
+
41
+ # 数据清洗
42
+ dirty_text = "<p>这是一个&nbsp;<b>测试</b>&amp;文本</p>"
43
+ clean_result = clean_text(dirty_text)
44
+ print(f"清洗文本: {dirty_text} -> {clean_result}")
45
+
46
+ # 数据格式化
47
+ price = 1234.567
48
+ formatted_price = format_currency(price, "¥", 2)
49
+ print(f"格式化货币: {price} -> {formatted_price}")
50
+
51
+ # 字段验证
52
+ email = "test@example.com"
53
+ is_valid_email = validate_email(email)
54
+ print(f"验证邮箱: {email} -> {'有效' if is_valid_email else '无效'}")
55
+
56
+ url = "https://example.com"
57
+ is_valid_url = validate_url(url)
58
+ print(f"验证URL: {url} -> {'有效' if is_valid_url else '无效'}")
59
+
60
+ # 数据完整性检查
61
+ data = {
62
+ "name": "张三",
63
+ "email": "test@example.com",
64
+ "phone": "13812345678"
65
+ }
66
+ required_fields = ["name", "email", "phone", "address"]
67
+ integrity_result = check_data_integrity(data, required_fields)
68
+ print(f"数据完整性检查: {integrity_result}")
69
+
70
+ print()
71
+
72
+
73
+ def demo_retry_mechanism():
74
+ """演示重试机制的使用"""
75
+ print("=== 重试机制演示 ===\n")
76
+
77
+ # 指数退避
78
+ for attempt in range(5):
79
+ delay = exponential_backoff(attempt)
80
+ print(f"重试次数 {attempt}: 延迟 {delay:.2f} 秒")
81
+
82
+ # 重试装饰器示例
83
+ @retry(max_retries=3)
84
+ def unreliable_function():
85
+ import random
86
+ if random.random() < 0.7: # 70%概率失败
87
+ raise ConnectionError("网络连接失败")
88
+ return "成功执行"
89
+
90
+ try:
91
+ result = unreliable_function()
92
+ print(f"函数执行结果: {result}")
93
+ except Exception as e:
94
+ print(f"函数执行失败: {e}")
95
+
96
+ print()
97
+
98
+
99
+ def demo_anti_crawler_tools():
100
+ """演示反爬虫应对工具的使用"""
101
+ print("=== 反爬虫应对工具演示 ===\n")
102
+
103
+ # 反爬虫工具
104
+ anti_crawler = AntiCrawler()
105
+
106
+ # 获取随机User-Agent
107
+ user_agent = anti_crawler.get_random_user_agent()
108
+ print(f"随机User-Agent: {user_agent[:50]}...")
109
+
110
+ # 轮换代理
111
+ proxy = anti_crawler.rotate_proxy()
112
+ print(f"轮换代理: {proxy}")
113
+
114
+ # 检测验证码
115
+ response_with_captcha = "请输入验证码进行验证"
116
+ has_captcha = anti_crawler.handle_captcha(response_with_captcha)
117
+ print(f"检测验证码: {response_with_captcha} -> {'需要验证码' if has_captcha else '无需验证码'}")
118
+
119
+ # 检测频率限制
120
+ status_code = 429 # Too Many Requests
121
+ response_headers = {"Retry-After": "60"}
122
+ is_rate_limited = anti_crawler.detect_rate_limiting(status_code, response_headers)
123
+ print(f"检测频率限制: 状态码{status_code} -> {'被限制' if is_rate_limited else '未限制'}")
124
+
125
+ print()
126
+
127
+
128
+ def demo_authenticated_proxy_tools():
129
+ """演示带认证代理工具的使用"""
130
+ print("=== 带认证代理工具演示 ===\n")
131
+
132
+ # 不同类型的代理URL
133
+ proxy_urls = [
134
+ "http://user:pass@proxy1.example.com:8080", # 带认证HTTP代理
135
+ "https://username:password@proxy2.example.com:443", # 带认证HTTPS代理
136
+ "http://proxy3.example.com:8080" # 不带认证代理
137
+ ]
138
+
139
+ for proxy_url in proxy_urls:
140
+ print(f"处理代理: {proxy_url}")
141
+
142
+ # 创建代理对象
143
+ proxy = AuthenticatedProxy(proxy_url)
144
+
145
+ # 为不同下载器格式化代理配置
146
+ for downloader in ["aiohttp", "httpx", "curl_cffi"]:
147
+ config = create_proxy_config(proxy_url)
148
+ formatted = format_proxy_for_request(config, downloader)
149
+ print(f" {downloader}格式: {formatted}")
150
+
151
+ print()
152
+
153
+
154
+ def demo_distributed_coordinator_tools():
155
+ """演示分布式协调工具的使用"""
156
+ print("=== 分布式协调工具演示 ===\n")
157
+
158
+ # 生成分页任务
159
+ base_url = "https://example.com/products"
160
+ pagination_tasks = generate_pagination_tasks(base_url, 1, 5)
161
+ print(f"生成分页任务 ({len(pagination_tasks)} 个):")
162
+ for i, task in enumerate(pagination_tasks[:3]): # 只显示前3个
163
+ print(f" {i+1}. {task}")
164
+ if len(pagination_tasks) > 3:
165
+ print(f" ... 还有 {len(pagination_tasks) - 3} 个任务")
166
+
167
+ # 任务分发
168
+ tasks = list(range(1, 21)) # 20个任务
169
+ distributed = distribute_tasks(tasks, 4) # 分发给4个工作节点
170
+ print(f"\n任务分发 (20个任务分发给4个工作节点):")
171
+ for i, worker_tasks in enumerate(distributed):
172
+ print(f" 工作节点 {i+1}: {len(worker_tasks)} 个任务 -> {worker_tasks}")
173
+
174
+ # 分布式协调器
175
+ coordinator = DistributedCoordinator()
176
+ cluster_info = coordinator.get_cluster_info()
177
+ print(f"\n集群信息: {cluster_info}")
178
+
179
+ print()
180
+
181
+
182
+ def demo_in_spider():
183
+ """演示在爬虫中使用高级工具"""
184
+ print("=== 在爬虫中使用高级工具 ===\n")
185
+ print("在爬虫项目中,您可以这样使用高级工具:")
186
+ print("""
187
+ import asyncio
188
+ from crawlo import Spider, Request
189
+ from crawlo.tools import (
190
+ clean_text,
191
+ validate_email,
192
+ AntiCrawler,
193
+ DistributedCoordinator,
194
+ retry,
195
+ AuthenticatedProxy
196
+ )
197
+
198
+ class AdvancedSpider(Spider):
199
+ def __init__(self):
200
+ super().__init__()
201
+ self.anti_crawler = AntiCrawler()
202
+ self.coordinator = DistributedCoordinator()
203
+ # 代理列表
204
+ self.proxy_urls = [
205
+ "http://user1:pass1@proxy1.example.com:8080",
206
+ "http://user2:pass2@proxy2.example.com:8080",
207
+ "http://proxy3.example.com:8080" # 不带认证
208
+ ]
209
+
210
+ def start_requests(self):
211
+ # 生成分页任务
212
+ base_url = "https://api.example.com/products"
213
+ pagination_tasks = self.coordinator.generate_pagination_tasks(base_url, 1, 100)
214
+
215
+ for i, url in enumerate(pagination_tasks):
216
+ # 轮换使用代理
217
+ proxy_url = self.proxy_urls[i % len(self.proxy_urls)]
218
+ proxy = AuthenticatedProxy(proxy_url)
219
+
220
+ request = Request(url, callback=self.parse)
221
+
222
+ # 根据下载器类型设置代理
223
+ downloader_type = self.crawler.settings.get("DOWNLOADER_TYPE", "aiohttp")
224
+ if downloader_type == "aiohttp":
225
+ request.proxy = proxy.clean_url
226
+ auth = proxy.get_auth_credentials()
227
+ if auth:
228
+ request.meta["proxy_auth"] = auth
229
+ elif downloader_type == "httpx":
230
+ request.proxy = proxy.clean_url
231
+ elif downloader_type == "curl_cffi":
232
+ request.proxy = proxy.proxy_dict
233
+ auth_header = proxy.get_auth_header()
234
+ if auth_header:
235
+ request.headers["Proxy-Authorization"] = auth_header
236
+
237
+ yield request
238
+
239
+ @retry(max_retries=3)
240
+ async def parse(self, response):
241
+ # 检查是否遇到验证码
242
+ if self.anti_crawler.handle_captcha(response.text):
243
+ # 处理验证码逻辑
244
+ print("遇到验证码,需要处理")
245
+ return
246
+
247
+ # 提取数据
248
+ products = response.css('.product-item')
249
+ for product in products:
250
+ name = product.css('.product-name::text').get()
251
+ price_text = product.css('.price::text').get()
252
+ email = product.css('.contact-email::text').get()
253
+
254
+ # 数据清洗和验证
255
+ clean_name = clean_text(name) if name else None
256
+ clean_price = clean_text(price_text) if price_text else None
257
+ is_valid_email = validate_email(email) if email else False
258
+
259
+ # 检查数据是否重复
260
+ if not await self.coordinator.is_duplicate({"name": clean_name, "price": clean_price}):
261
+ # 添加到去重集合
262
+ await self.coordinator.add_to_dedup({"name": clean_name, "price": clean_price})
263
+
264
+ # 处理产品数据...
265
+ pass
266
+ """)
267
+
268
+
269
+ if __name__ == '__main__':
270
+ # 运行演示
271
+ demo_data_processing_tools()
272
+ demo_retry_mechanism()
273
+ demo_anti_crawler_tools()
274
+ demo_authenticated_proxy_tools()
275
+ demo_distributed_coordinator_tools()
276
276
  demo_in_spider()