crawlo 1.2.8__py3-none-any.whl → 1.2.9__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.
- crawlo/__init__.py +63 -61
- crawlo/__version__.py +1 -1
- crawlo/cli.py +75 -75
- crawlo/commands/__init__.py +14 -14
- crawlo/commands/check.py +594 -594
- crawlo/commands/genspider.py +151 -151
- crawlo/commands/help.py +138 -138
- crawlo/commands/list.py +155 -155
- crawlo/commands/run.py +323 -323
- crawlo/commands/startproject.py +436 -436
- crawlo/commands/stats.py +187 -187
- crawlo/commands/utils.py +186 -186
- crawlo/config.py +312 -312
- crawlo/config_validator.py +277 -251
- crawlo/core/__init__.py +2 -2
- crawlo/core/engine.py +366 -365
- crawlo/core/processor.py +40 -40
- crawlo/core/scheduler.py +256 -251
- crawlo/crawler.py +1103 -1100
- crawlo/data/__init__.py +5 -5
- crawlo/data/user_agents.py +194 -107
- crawlo/downloader/__init__.py +273 -266
- crawlo/downloader/aiohttp_downloader.py +226 -228
- crawlo/downloader/cffi_downloader.py +245 -256
- crawlo/downloader/httpx_downloader.py +259 -259
- crawlo/downloader/hybrid_downloader.py +212 -212
- crawlo/downloader/playwright_downloader.py +402 -402
- crawlo/downloader/selenium_downloader.py +472 -472
- crawlo/event.py +11 -11
- crawlo/exceptions.py +81 -81
- crawlo/extension/__init__.py +39 -39
- crawlo/extension/health_check.py +141 -141
- crawlo/extension/log_interval.py +57 -57
- crawlo/extension/log_stats.py +81 -81
- crawlo/extension/logging_extension.py +43 -43
- crawlo/extension/memory_monitor.py +104 -104
- crawlo/extension/performance_profiler.py +133 -133
- crawlo/extension/request_recorder.py +107 -107
- crawlo/filters/__init__.py +154 -154
- crawlo/filters/aioredis_filter.py +234 -234
- crawlo/filters/memory_filter.py +269 -269
- crawlo/items/__init__.py +23 -23
- crawlo/items/base.py +21 -21
- crawlo/items/fields.py +52 -52
- crawlo/items/items.py +104 -104
- crawlo/middleware/__init__.py +21 -21
- crawlo/middleware/default_header.py +132 -132
- crawlo/middleware/download_delay.py +104 -104
- crawlo/middleware/middleware_manager.py +136 -136
- crawlo/middleware/offsite.py +114 -114
- crawlo/middleware/proxy.py +386 -368
- crawlo/middleware/request_ignore.py +86 -86
- crawlo/middleware/response_code.py +163 -163
- crawlo/middleware/response_filter.py +136 -136
- crawlo/middleware/retry.py +124 -124
- crawlo/middleware/simple_proxy.py +65 -0
- crawlo/mode_manager.py +211 -211
- crawlo/network/__init__.py +21 -21
- crawlo/network/request.py +379 -338
- crawlo/network/response.py +359 -359
- crawlo/pipelines/__init__.py +21 -21
- crawlo/pipelines/bloom_dedup_pipeline.py +157 -157
- crawlo/pipelines/console_pipeline.py +39 -39
- crawlo/pipelines/csv_pipeline.py +316 -316
- crawlo/pipelines/database_dedup_pipeline.py +223 -223
- crawlo/pipelines/json_pipeline.py +218 -218
- crawlo/pipelines/memory_dedup_pipeline.py +115 -115
- crawlo/pipelines/mongo_pipeline.py +131 -131
- crawlo/pipelines/mysql_pipeline.py +317 -317
- crawlo/pipelines/pipeline_manager.py +62 -62
- crawlo/pipelines/redis_dedup_pipeline.py +167 -167
- crawlo/project.py +290 -315
- crawlo/queue/pqueue.py +37 -37
- crawlo/queue/queue_manager.py +379 -378
- crawlo/queue/redis_priority_queue.py +306 -306
- crawlo/settings/__init__.py +7 -7
- crawlo/settings/default_settings.py +216 -220
- crawlo/settings/setting_manager.py +163 -122
- crawlo/spider/__init__.py +639 -639
- crawlo/stats_collector.py +59 -59
- crawlo/subscriber.py +129 -129
- crawlo/task_manager.py +30 -30
- crawlo/templates/crawlo.cfg.tmpl +10 -10
- crawlo/templates/project/__init__.py.tmpl +3 -3
- crawlo/templates/project/items.py.tmpl +17 -17
- crawlo/templates/project/middlewares.py.tmpl +118 -118
- crawlo/templates/project/pipelines.py.tmpl +96 -96
- crawlo/templates/project/settings.py.tmpl +261 -288
- crawlo/templates/project/settings_distributed.py.tmpl +174 -157
- crawlo/templates/project/settings_gentle.py.tmpl +95 -100
- crawlo/templates/project/settings_high_performance.py.tmpl +125 -134
- crawlo/templates/project/settings_minimal.py.tmpl +30 -0
- crawlo/templates/project/settings_simple.py.tmpl +96 -98
- crawlo/templates/project/spiders/__init__.py.tmpl +5 -5
- crawlo/templates/run.py.tmpl +47 -47
- crawlo/templates/spider/spider.py.tmpl +143 -143
- crawlo/tools/__init__.py +200 -182
- crawlo/tools/anti_crawler.py +268 -268
- crawlo/tools/authenticated_proxy.py +240 -240
- crawlo/{cleaners → tools}/data_formatter.py +225 -225
- crawlo/tools/data_validator.py +180 -180
- crawlo/tools/date_tools.py +290 -36
- crawlo/tools/distributed_coordinator.py +388 -387
- crawlo/{cleaners → tools}/encoding_converter.py +127 -126
- crawlo/tools/request_tools.py +83 -0
- crawlo/tools/retry_mechanism.py +224 -221
- crawlo/tools/scenario_adapter.py +262 -262
- crawlo/{cleaners → tools}/text_cleaner.py +232 -232
- crawlo/utils/__init__.py +35 -35
- crawlo/utils/batch_processor.py +259 -259
- crawlo/utils/controlled_spider_mixin.py +439 -439
- crawlo/utils/db_helper.py +343 -343
- crawlo/utils/enhanced_error_handler.py +356 -356
- crawlo/utils/env_config.py +142 -142
- crawlo/utils/error_handler.py +123 -123
- crawlo/utils/func_tools.py +82 -82
- crawlo/utils/large_scale_config.py +286 -286
- crawlo/utils/large_scale_helper.py +344 -344
- crawlo/utils/log.py +187 -128
- crawlo/utils/performance_monitor.py +285 -285
- crawlo/utils/queue_helper.py +175 -175
- crawlo/utils/redis_connection_pool.py +351 -351
- crawlo/utils/redis_key_validator.py +198 -198
- crawlo/utils/request.py +267 -267
- crawlo/utils/request_serializer.py +218 -218
- crawlo/utils/spider_loader.py +61 -61
- crawlo/utils/system.py +11 -11
- crawlo/utils/tools.py +4 -4
- crawlo/utils/url.py +39 -39
- {crawlo-1.2.8.dist-info → crawlo-1.2.9.dist-info}/METADATA +1011 -764
- crawlo-1.2.9.dist-info/RECORD +219 -0
- examples/__init__.py +7 -7
- tests/DOUBLE_CRAWLO_PREFIX_FIX_REPORT.md +81 -81
- tests/__init__.py +7 -7
- tests/advanced_tools_example.py +275 -275
- tests/authenticated_proxy_example.py +107 -237
- tests/cleaners_example.py +160 -160
- tests/config_validation_demo.py +143 -103
- tests/controlled_spider_example.py +205 -205
- tests/date_tools_example.py +180 -180
- tests/debug_pipelines.py +67 -0
- tests/dynamic_loading_example.py +523 -523
- tests/dynamic_loading_test.py +104 -104
- tests/env_config_example.py +133 -133
- tests/error_handling_example.py +171 -171
- tests/redis_key_validation_demo.py +130 -130
- tests/request_params_example.py +151 -0
- tests/response_improvements_example.py +144 -144
- tests/test_advanced_tools.py +148 -148
- tests/test_all_redis_key_configs.py +145 -145
- tests/test_authenticated_proxy.py +141 -141
- tests/test_cleaners.py +54 -54
- tests/test_comprehensive.py +146 -146
- tests/test_config_consistency.py +80 -80
- tests/test_config_merge.py +153 -0
- tests/test_config_validator.py +182 -193
- tests/test_crawlo_proxy_integration.py +109 -173
- tests/test_date_tools.py +123 -123
- tests/test_default_header_middleware.py +158 -158
- tests/test_distributed.py +65 -0
- tests/test_double_crawlo_fix.py +207 -207
- tests/test_double_crawlo_fix_simple.py +124 -124
- tests/test_download_delay_middleware.py +221 -221
- tests/test_downloader_proxy_compatibility.py +268 -268
- tests/test_dynamic_downloaders_proxy.py +124 -124
- tests/test_dynamic_proxy.py +92 -92
- tests/test_dynamic_proxy_config.py +146 -146
- tests/test_dynamic_proxy_real.py +109 -109
- tests/test_edge_cases.py +303 -303
- tests/test_enhanced_error_handler.py +270 -270
- tests/test_env_config.py +121 -121
- tests/test_error_handler_compatibility.py +112 -112
- tests/test_final_validation.py +153 -153
- tests/test_framework_env_usage.py +103 -103
- tests/test_integration.py +169 -357
- tests/test_item_dedup_redis_key.py +122 -122
- tests/test_mode_consistency.py +51 -51
- tests/test_offsite_middleware.py +221 -221
- tests/test_parsel.py +29 -29
- tests/test_performance.py +327 -327
- tests/test_proxy_api.py +264 -264
- tests/test_proxy_health_check.py +32 -32
- tests/test_proxy_middleware.py +121 -121
- tests/test_proxy_middleware_enhanced.py +216 -216
- tests/test_proxy_middleware_integration.py +136 -136
- tests/test_proxy_middleware_refactored.py +185 -0
- tests/test_proxy_providers.py +56 -56
- tests/test_proxy_stats.py +19 -19
- tests/test_proxy_strategies.py +59 -59
- tests/test_queue_manager_double_crawlo.py +173 -173
- tests/test_queue_manager_redis_key.py +176 -176
- tests/test_random_user_agent.py +73 -0
- tests/test_real_scenario_proxy.py +195 -195
- tests/test_redis_config.py +28 -28
- tests/test_redis_connection_pool.py +294 -294
- tests/test_redis_key_naming.py +181 -181
- tests/test_redis_key_validator.py +123 -123
- tests/test_redis_queue.py +224 -224
- tests/test_request_ignore_middleware.py +182 -182
- tests/test_request_params.py +112 -0
- tests/test_request_serialization.py +70 -70
- tests/test_response_code_middleware.py +349 -349
- tests/test_response_filter_middleware.py +427 -427
- tests/test_response_improvements.py +152 -152
- tests/test_retry_middleware.py +241 -241
- tests/test_scheduler.py +252 -252
- tests/test_scheduler_config_update.py +133 -133
- tests/test_simple_response.py +61 -61
- tests/test_telecom_spider_redis_key.py +205 -205
- tests/test_template_content.py +87 -87
- tests/test_template_redis_key.py +134 -134
- tests/test_tools.py +159 -153
- tests/test_user_agents.py +97 -0
- tests/tools_example.py +260 -257
- tests/verify_distributed.py +117 -0
- crawlo/cleaners/__init__.py +0 -61
- crawlo/utils/date_tools.py +0 -290
- crawlo-1.2.8.dist-info/RECORD +0 -209
- {crawlo-1.2.8.dist-info → crawlo-1.2.9.dist-info}/WHEEL +0 -0
- {crawlo-1.2.8.dist-info → crawlo-1.2.9.dist-info}/entry_points.txt +0 -0
- {crawlo-1.2.8.dist-info → crawlo-1.2.9.dist-info}/top_level.txt +0 -0
|
@@ -1,177 +1,177 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
|
-
# -*- coding: utf-8 -*-
|
|
3
|
-
"""
|
|
4
|
-
QueueManager Redis Key 测试脚本
|
|
5
|
-
用于验证QueueManager在创建RedisPriorityQueue时是否正确传递module_name参数
|
|
6
|
-
"""
|
|
7
|
-
import asyncio
|
|
8
|
-
import sys
|
|
9
|
-
import os
|
|
10
|
-
import traceback
|
|
11
|
-
|
|
12
|
-
# 添加项目根目录到路径
|
|
13
|
-
sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..'))
|
|
14
|
-
|
|
15
|
-
from crawlo.queue.queue_manager import QueueManager, QueueConfig, QueueType
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
class MockSettings:
|
|
19
|
-
"""模拟设置类"""
|
|
20
|
-
def __init__(self, project_name="test_project"):
|
|
21
|
-
self.project_name = project_name
|
|
22
|
-
self.REDIS_URL = "redis://127.0.0.1:6379/15" # 使用测试数据库
|
|
23
|
-
self.REDIS_TTL = 0
|
|
24
|
-
self.CLEANUP_FP = 0
|
|
25
|
-
self.FILTER_DEBUG = True
|
|
26
|
-
self.LOG_LEVEL = "INFO"
|
|
27
|
-
self.DECODE_RESPONSES = True
|
|
28
|
-
|
|
29
|
-
def get(self, key, default=None):
|
|
30
|
-
if key == 'PROJECT_NAME':
|
|
31
|
-
return self.project_name
|
|
32
|
-
elif key == 'REDIS_URL':
|
|
33
|
-
return self.REDIS_URL
|
|
34
|
-
elif key == 'FILTER_DEBUG':
|
|
35
|
-
return self.FILTER_DEBUG
|
|
36
|
-
elif key == 'LOG_LEVEL':
|
|
37
|
-
return self.LOG_LEVEL
|
|
38
|
-
elif key == 'DECODE_RESPONSES':
|
|
39
|
-
return self.DECODE_RESPONSES
|
|
40
|
-
return default
|
|
41
|
-
|
|
42
|
-
def get_bool(self, key, default=False):
|
|
43
|
-
if key == 'FILTER_DEBUG':
|
|
44
|
-
return self.FILTER_DEBUG
|
|
45
|
-
elif key == 'DECODE_RESPONSES':
|
|
46
|
-
return self.DECODE_RESPONSES
|
|
47
|
-
elif key == 'CLEANUP_FP':
|
|
48
|
-
return self.CLEANUP_FP
|
|
49
|
-
return default
|
|
50
|
-
|
|
51
|
-
def get_int(self, key, default=0):
|
|
52
|
-
if key == 'REDIS_TTL':
|
|
53
|
-
return self.REDIS_TTL
|
|
54
|
-
elif key == 'REDIS_PORT':
|
|
55
|
-
return 6379
|
|
56
|
-
elif key == 'REDIS_DB':
|
|
57
|
-
return 0
|
|
58
|
-
elif key == 'SCHEDULER_MAX_QUEUE_SIZE':
|
|
59
|
-
return 1000
|
|
60
|
-
elif key == 'QUEUE_MAX_RETRIES':
|
|
61
|
-
return 3
|
|
62
|
-
elif key == 'QUEUE_TIMEOUT':
|
|
63
|
-
return 300
|
|
64
|
-
return default
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
async def test_queue_manager_redis_key():
|
|
68
|
-
"""测试QueueManager创建Redis队列时的key命名"""
|
|
69
|
-
print("
|
|
70
|
-
|
|
71
|
-
try:
|
|
72
|
-
# 测试不同的队列名称配置
|
|
73
|
-
test_cases = [
|
|
74
|
-
{
|
|
75
|
-
"queue_name": "crawlo:books_distributed:queue:requests",
|
|
76
|
-
"expected_module_name": "books_distributed",
|
|
77
|
-
"description": "标准项目名称"
|
|
78
|
-
},
|
|
79
|
-
{
|
|
80
|
-
"queue_name": "crawlo:api_data_collection:queue:requests",
|
|
81
|
-
"expected_module_name": "api_data_collection",
|
|
82
|
-
"description": "API数据采集项目"
|
|
83
|
-
},
|
|
84
|
-
{
|
|
85
|
-
"queue_name": "crawlo:test_project:queue:requests",
|
|
86
|
-
"expected_module_name": "test_project",
|
|
87
|
-
"description": "测试项目"
|
|
88
|
-
},
|
|
89
|
-
{
|
|
90
|
-
"queue_name": "simple_queue_name",
|
|
91
|
-
"expected_module_name": "simple_queue_name",
|
|
92
|
-
"description": "简单队列名称"
|
|
93
|
-
},
|
|
94
|
-
{
|
|
95
|
-
"queue_name": "",
|
|
96
|
-
"expected_module_name": "default",
|
|
97
|
-
"description": "空队列名称"
|
|
98
|
-
}
|
|
99
|
-
]
|
|
100
|
-
|
|
101
|
-
for i, test_case in enumerate(test_cases, 1):
|
|
102
|
-
print(f" {i}. 测试 {test_case['description']}...")
|
|
103
|
-
|
|
104
|
-
# 创建QueueConfig
|
|
105
|
-
config = QueueConfig(
|
|
106
|
-
queue_type=QueueType.REDIS,
|
|
107
|
-
redis_url="redis://127.0.0.1:6379/15",
|
|
108
|
-
queue_name=test_case["queue_name"],
|
|
109
|
-
max_queue_size=1000,
|
|
110
|
-
max_retries=3,
|
|
111
|
-
timeout=300
|
|
112
|
-
)
|
|
113
|
-
|
|
114
|
-
# 创建QueueManager
|
|
115
|
-
queue_manager = QueueManager(config)
|
|
116
|
-
|
|
117
|
-
# 创建队列实例
|
|
118
|
-
queue = await queue_manager._create_queue(QueueType.REDIS)
|
|
119
|
-
|
|
120
|
-
# 验证module_name是否正确设置
|
|
121
|
-
assert hasattr(queue, 'module_name'), "RedisPriorityQueue缺少module_name属性"
|
|
122
|
-
assert queue.module_name == test_case["expected_module_name"], \
|
|
123
|
-
f"module_name不匹配: {queue.module_name} != {test_case['expected_module_name']}"
|
|
124
|
-
|
|
125
|
-
# 验证队列名称是否符合规范
|
|
126
|
-
expected_queue_name = f"crawlo:{queue.module_name}:queue:requests"
|
|
127
|
-
expected_processing_queue = f"crawlo:{queue.module_name}:queue:processing"
|
|
128
|
-
expected_failed_queue = f"crawlo:{queue.module_name}:queue:failed"
|
|
129
|
-
|
|
130
|
-
assert queue.queue_name == expected_queue_name, \
|
|
131
|
-
f"队列名称不匹配: {queue.queue_name} != {expected_queue_name}"
|
|
132
|
-
assert queue.processing_queue == expected_processing_queue, \
|
|
133
|
-
f"处理中队列名称不匹配: {queue.processing_queue} != {expected_processing_queue}"
|
|
134
|
-
assert queue.failed_queue == expected_failed_queue, \
|
|
135
|
-
f"失败队列名称不匹配: {queue.failed_queue} != {expected_failed_queue}"
|
|
136
|
-
|
|
137
|
-
print(f"
|
|
138
|
-
print(f"
|
|
139
|
-
print(f"
|
|
140
|
-
print(f"
|
|
141
|
-
|
|
142
|
-
print("
|
|
143
|
-
return True
|
|
144
|
-
|
|
145
|
-
except Exception as e:
|
|
146
|
-
print(f"
|
|
147
|
-
traceback.print_exc()
|
|
148
|
-
return False
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
async def main():
|
|
152
|
-
"""主测试函数"""
|
|
153
|
-
print("
|
|
154
|
-
print("=" * 50)
|
|
155
|
-
|
|
156
|
-
try:
|
|
157
|
-
success = await test_queue_manager_redis_key()
|
|
158
|
-
|
|
159
|
-
print("=" * 50)
|
|
160
|
-
if success:
|
|
161
|
-
print("
|
|
162
|
-
else:
|
|
163
|
-
print("
|
|
164
|
-
return 1
|
|
165
|
-
|
|
166
|
-
except Exception as e:
|
|
167
|
-
print("=" * 50)
|
|
168
|
-
print(f"
|
|
169
|
-
traceback.print_exc()
|
|
170
|
-
return 1
|
|
171
|
-
|
|
172
|
-
return 0
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
if __name__ == "__main__":
|
|
176
|
-
exit_code = asyncio.run(main())
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
# -*- coding: utf-8 -*-
|
|
3
|
+
"""
|
|
4
|
+
QueueManager Redis Key 测试脚本
|
|
5
|
+
用于验证QueueManager在创建RedisPriorityQueue时是否正确传递module_name参数
|
|
6
|
+
"""
|
|
7
|
+
import asyncio
|
|
8
|
+
import sys
|
|
9
|
+
import os
|
|
10
|
+
import traceback
|
|
11
|
+
|
|
12
|
+
# 添加项目根目录到路径
|
|
13
|
+
sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..'))
|
|
14
|
+
|
|
15
|
+
from crawlo.queue.queue_manager import QueueManager, QueueConfig, QueueType
|
|
16
|
+
|
|
17
|
+
|
|
18
|
+
class MockSettings:
|
|
19
|
+
"""模拟设置类"""
|
|
20
|
+
def __init__(self, project_name="test_project"):
|
|
21
|
+
self.project_name = project_name
|
|
22
|
+
self.REDIS_URL = "redis://127.0.0.1:6379/15" # 使用测试数据库
|
|
23
|
+
self.REDIS_TTL = 0
|
|
24
|
+
self.CLEANUP_FP = 0
|
|
25
|
+
self.FILTER_DEBUG = True
|
|
26
|
+
self.LOG_LEVEL = "INFO"
|
|
27
|
+
self.DECODE_RESPONSES = True
|
|
28
|
+
|
|
29
|
+
def get(self, key, default=None):
|
|
30
|
+
if key == 'PROJECT_NAME':
|
|
31
|
+
return self.project_name
|
|
32
|
+
elif key == 'REDIS_URL':
|
|
33
|
+
return self.REDIS_URL
|
|
34
|
+
elif key == 'FILTER_DEBUG':
|
|
35
|
+
return self.FILTER_DEBUG
|
|
36
|
+
elif key == 'LOG_LEVEL':
|
|
37
|
+
return self.LOG_LEVEL
|
|
38
|
+
elif key == 'DECODE_RESPONSES':
|
|
39
|
+
return self.DECODE_RESPONSES
|
|
40
|
+
return default
|
|
41
|
+
|
|
42
|
+
def get_bool(self, key, default=False):
|
|
43
|
+
if key == 'FILTER_DEBUG':
|
|
44
|
+
return self.FILTER_DEBUG
|
|
45
|
+
elif key == 'DECODE_RESPONSES':
|
|
46
|
+
return self.DECODE_RESPONSES
|
|
47
|
+
elif key == 'CLEANUP_FP':
|
|
48
|
+
return self.CLEANUP_FP
|
|
49
|
+
return default
|
|
50
|
+
|
|
51
|
+
def get_int(self, key, default=0):
|
|
52
|
+
if key == 'REDIS_TTL':
|
|
53
|
+
return self.REDIS_TTL
|
|
54
|
+
elif key == 'REDIS_PORT':
|
|
55
|
+
return 6379
|
|
56
|
+
elif key == 'REDIS_DB':
|
|
57
|
+
return 0
|
|
58
|
+
elif key == 'SCHEDULER_MAX_QUEUE_SIZE':
|
|
59
|
+
return 1000
|
|
60
|
+
elif key == 'QUEUE_MAX_RETRIES':
|
|
61
|
+
return 3
|
|
62
|
+
elif key == 'QUEUE_TIMEOUT':
|
|
63
|
+
return 300
|
|
64
|
+
return default
|
|
65
|
+
|
|
66
|
+
|
|
67
|
+
async def test_queue_manager_redis_key():
|
|
68
|
+
"""测试QueueManager创建Redis队列时的key命名"""
|
|
69
|
+
print("测试QueueManager创建Redis队列时的key命名...")
|
|
70
|
+
|
|
71
|
+
try:
|
|
72
|
+
# 测试不同的队列名称配置
|
|
73
|
+
test_cases = [
|
|
74
|
+
{
|
|
75
|
+
"queue_name": "crawlo:books_distributed:queue:requests",
|
|
76
|
+
"expected_module_name": "books_distributed",
|
|
77
|
+
"description": "标准项目名称"
|
|
78
|
+
},
|
|
79
|
+
{
|
|
80
|
+
"queue_name": "crawlo:api_data_collection:queue:requests",
|
|
81
|
+
"expected_module_name": "api_data_collection",
|
|
82
|
+
"description": "API数据采集项目"
|
|
83
|
+
},
|
|
84
|
+
{
|
|
85
|
+
"queue_name": "crawlo:test_project:queue:requests",
|
|
86
|
+
"expected_module_name": "test_project",
|
|
87
|
+
"description": "测试项目"
|
|
88
|
+
},
|
|
89
|
+
{
|
|
90
|
+
"queue_name": "simple_queue_name",
|
|
91
|
+
"expected_module_name": "simple_queue_name",
|
|
92
|
+
"description": "简单队列名称"
|
|
93
|
+
},
|
|
94
|
+
{
|
|
95
|
+
"queue_name": "",
|
|
96
|
+
"expected_module_name": "default",
|
|
97
|
+
"description": "空队列名称"
|
|
98
|
+
}
|
|
99
|
+
]
|
|
100
|
+
|
|
101
|
+
for i, test_case in enumerate(test_cases, 1):
|
|
102
|
+
print(f" {i}. 测试 {test_case['description']}...")
|
|
103
|
+
|
|
104
|
+
# 创建QueueConfig
|
|
105
|
+
config = QueueConfig(
|
|
106
|
+
queue_type=QueueType.REDIS,
|
|
107
|
+
redis_url="redis://127.0.0.1:6379/15",
|
|
108
|
+
queue_name=test_case["queue_name"],
|
|
109
|
+
max_queue_size=1000,
|
|
110
|
+
max_retries=3,
|
|
111
|
+
timeout=300
|
|
112
|
+
)
|
|
113
|
+
|
|
114
|
+
# 创建QueueManager
|
|
115
|
+
queue_manager = QueueManager(config)
|
|
116
|
+
|
|
117
|
+
# 创建队列实例
|
|
118
|
+
queue = await queue_manager._create_queue(QueueType.REDIS)
|
|
119
|
+
|
|
120
|
+
# 验证module_name是否正确设置
|
|
121
|
+
assert hasattr(queue, 'module_name'), "RedisPriorityQueue缺少module_name属性"
|
|
122
|
+
assert queue.module_name == test_case["expected_module_name"], \
|
|
123
|
+
f"module_name不匹配: {queue.module_name} != {test_case['expected_module_name']}"
|
|
124
|
+
|
|
125
|
+
# 验证队列名称是否符合规范
|
|
126
|
+
expected_queue_name = f"crawlo:{queue.module_name}:queue:requests"
|
|
127
|
+
expected_processing_queue = f"crawlo:{queue.module_name}:queue:processing"
|
|
128
|
+
expected_failed_queue = f"crawlo:{queue.module_name}:queue:failed"
|
|
129
|
+
|
|
130
|
+
assert queue.queue_name == expected_queue_name, \
|
|
131
|
+
f"队列名称不匹配: {queue.queue_name} != {expected_queue_name}"
|
|
132
|
+
assert queue.processing_queue == expected_processing_queue, \
|
|
133
|
+
f"处理中队列名称不匹配: {queue.processing_queue} != {expected_processing_queue}"
|
|
134
|
+
assert queue.failed_queue == expected_failed_queue, \
|
|
135
|
+
f"失败队列名称不匹配: {queue.failed_queue} != {expected_failed_queue}"
|
|
136
|
+
|
|
137
|
+
print(f" module_name: {queue.module_name}")
|
|
138
|
+
print(f" 队列名称: {queue.queue_name}")
|
|
139
|
+
print(f" 处理中队列名称: {queue.processing_queue}")
|
|
140
|
+
print(f" 失败队列名称: {queue.failed_queue}")
|
|
141
|
+
|
|
142
|
+
print("QueueManager Redis key命名测试通过!")
|
|
143
|
+
return True
|
|
144
|
+
|
|
145
|
+
except Exception as e:
|
|
146
|
+
print(f"QueueManager Redis key命名测试失败: {e}")
|
|
147
|
+
traceback.print_exc()
|
|
148
|
+
return False
|
|
149
|
+
|
|
150
|
+
|
|
151
|
+
async def main():
|
|
152
|
+
"""主测试函数"""
|
|
153
|
+
print("开始QueueManager Redis key命名测试...")
|
|
154
|
+
print("=" * 50)
|
|
155
|
+
|
|
156
|
+
try:
|
|
157
|
+
success = await test_queue_manager_redis_key()
|
|
158
|
+
|
|
159
|
+
print("=" * 50)
|
|
160
|
+
if success:
|
|
161
|
+
print("所有测试通过!QueueManager正确传递module_name参数")
|
|
162
|
+
else:
|
|
163
|
+
print("测试失败,请检查实现")
|
|
164
|
+
return 1
|
|
165
|
+
|
|
166
|
+
except Exception as e:
|
|
167
|
+
print("=" * 50)
|
|
168
|
+
print(f"测试过程中发生异常: {e}")
|
|
169
|
+
traceback.print_exc()
|
|
170
|
+
return 1
|
|
171
|
+
|
|
172
|
+
return 0
|
|
173
|
+
|
|
174
|
+
|
|
175
|
+
if __name__ == "__main__":
|
|
176
|
+
exit_code = asyncio.run(main())
|
|
177
177
|
sys.exit(exit_code)
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
#!/usr/bin/env python
|
|
2
|
+
# -*- coding: utf-8 -*-
|
|
3
|
+
"""
|
|
4
|
+
测试随机User-Agent功能
|
|
5
|
+
"""
|
|
6
|
+
|
|
7
|
+
import sys
|
|
8
|
+
import os
|
|
9
|
+
|
|
10
|
+
# 添加项目根目录到Python路径
|
|
11
|
+
sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..'))
|
|
12
|
+
|
|
13
|
+
from crawlo.data.user_agents import get_random_user_agent
|
|
14
|
+
from crawlo.middleware.default_header import DefaultHeaderMiddleware
|
|
15
|
+
from crawlo.settings.setting_manager import SettingManager
|
|
16
|
+
|
|
17
|
+
|
|
18
|
+
def test_random_user_agent_function():
|
|
19
|
+
"""测试随机User-Agent函数"""
|
|
20
|
+
print("=== 随机User-Agent函数测试 ===")
|
|
21
|
+
|
|
22
|
+
# 测试各种设备类型的随机User-Agent
|
|
23
|
+
device_types = ["desktop", "mobile", "all", "chrome", "firefox", "safari", "edge", "opera"]
|
|
24
|
+
|
|
25
|
+
for device_type in device_types:
|
|
26
|
+
print(f"\n{device_type}类型随机User-Agent:")
|
|
27
|
+
for i in range(3):
|
|
28
|
+
ua = get_random_user_agent(device_type)
|
|
29
|
+
print(f" {i+1}. {ua}")
|
|
30
|
+
|
|
31
|
+
print()
|
|
32
|
+
|
|
33
|
+
|
|
34
|
+
def test_default_header_middleware_with_random_ua():
|
|
35
|
+
"""测试DefaultHeaderMiddleware与随机User-Agent"""
|
|
36
|
+
print("=== DefaultHeaderMiddleware与随机User-Agent测试 ===")
|
|
37
|
+
|
|
38
|
+
# 创建设置管理器
|
|
39
|
+
settings = SettingManager()
|
|
40
|
+
settings.set("LOG_LEVEL", "INFO")
|
|
41
|
+
settings.set("RANDOM_USER_AGENT_ENABLED", True)
|
|
42
|
+
settings.set("USER_AGENT_DEVICE_TYPE", "all")
|
|
43
|
+
# 添加一个默认请求头以启用中间件
|
|
44
|
+
settings.set("DEFAULT_REQUEST_HEADERS", {"Accept": "text/html"})
|
|
45
|
+
|
|
46
|
+
# 创建中间件实例
|
|
47
|
+
middleware = DefaultHeaderMiddleware(settings, "INFO")
|
|
48
|
+
|
|
49
|
+
print(f"随机User-Agent功能已启用: {middleware.random_user_agent_enabled}")
|
|
50
|
+
print(f"User-Agent设备类型: {middleware.user_agent_device_type}")
|
|
51
|
+
print(f"内置User-Agent列表数量: {len(middleware.user_agents)}")
|
|
52
|
+
|
|
53
|
+
# 测试获取随机User-Agent
|
|
54
|
+
print("\n随机User-Agent测试:")
|
|
55
|
+
for i in range(5):
|
|
56
|
+
random_ua = middleware._get_random_user_agent()
|
|
57
|
+
print(f" {i+1}. {random_ua}")
|
|
58
|
+
|
|
59
|
+
print()
|
|
60
|
+
|
|
61
|
+
|
|
62
|
+
def main():
|
|
63
|
+
"""主测试函数"""
|
|
64
|
+
print("开始测试随机User-Agent功能...\n")
|
|
65
|
+
|
|
66
|
+
test_random_user_agent_function()
|
|
67
|
+
test_default_header_middleware_with_random_ua()
|
|
68
|
+
|
|
69
|
+
print("所有测试完成!")
|
|
70
|
+
|
|
71
|
+
|
|
72
|
+
if __name__ == "__main__":
|
|
73
|
+
main()
|