crawlo 1.4.6__py3-none-any.whl → 1.4.7__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 +90 -89
- 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 +186 -186
- crawlo/commands/help.py +140 -138
- crawlo/commands/list.py +155 -155
- crawlo/commands/run.py +379 -341
- crawlo/commands/startproject.py +460 -460
- crawlo/commands/stats.py +187 -187
- crawlo/commands/utils.py +196 -196
- crawlo/config.py +320 -312
- crawlo/config_validator.py +277 -277
- crawlo/core/__init__.py +52 -52
- crawlo/core/engine.py +451 -438
- crawlo/core/processor.py +47 -47
- crawlo/core/scheduler.py +290 -291
- crawlo/crawler.py +698 -657
- crawlo/data/__init__.py +5 -5
- crawlo/data/user_agents.py +194 -194
- crawlo/downloader/__init__.py +280 -276
- crawlo/downloader/aiohttp_downloader.py +233 -233
- crawlo/downloader/cffi_downloader.py +250 -247
- crawlo/downloader/httpx_downloader.py +265 -259
- crawlo/downloader/hybrid_downloader.py +212 -212
- crawlo/downloader/playwright_downloader.py +425 -402
- crawlo/downloader/selenium_downloader.py +486 -472
- crawlo/event.py +45 -11
- crawlo/exceptions.py +215 -82
- crawlo/extension/__init__.py +65 -64
- crawlo/extension/health_check.py +141 -141
- crawlo/extension/log_interval.py +94 -94
- crawlo/extension/log_stats.py +70 -70
- crawlo/extension/logging_extension.py +53 -61
- crawlo/extension/memory_monitor.py +104 -104
- crawlo/extension/performance_profiler.py +133 -133
- crawlo/extension/request_recorder.py +107 -107
- crawlo/factories/__init__.py +27 -27
- crawlo/factories/base.py +68 -68
- crawlo/factories/crawler.py +104 -103
- crawlo/factories/registry.py +84 -84
- crawlo/factories/utils.py +135 -0
- crawlo/filters/__init__.py +170 -153
- crawlo/filters/aioredis_filter.py +348 -264
- crawlo/filters/memory_filter.py +261 -276
- crawlo/framework.py +306 -292
- crawlo/initialization/__init__.py +44 -44
- crawlo/initialization/built_in.py +391 -434
- crawlo/initialization/context.py +141 -141
- crawlo/initialization/core.py +240 -194
- crawlo/initialization/phases.py +230 -149
- crawlo/initialization/registry.py +143 -145
- crawlo/initialization/utils.py +49 -0
- crawlo/interfaces.py +23 -23
- crawlo/items/__init__.py +23 -23
- crawlo/items/base.py +23 -23
- crawlo/items/fields.py +52 -52
- crawlo/items/items.py +104 -104
- crawlo/logging/__init__.py +42 -46
- crawlo/logging/config.py +277 -197
- crawlo/logging/factory.py +175 -171
- crawlo/logging/manager.py +104 -112
- crawlo/middleware/__init__.py +87 -24
- crawlo/middleware/default_header.py +132 -132
- crawlo/middleware/download_delay.py +104 -104
- crawlo/middleware/middleware_manager.py +142 -142
- crawlo/middleware/offsite.py +123 -123
- crawlo/middleware/proxy.py +209 -209
- crawlo/middleware/request_ignore.py +86 -86
- crawlo/middleware/response_code.py +150 -150
- crawlo/middleware/response_filter.py +136 -136
- crawlo/middleware/retry.py +124 -124
- crawlo/mode_manager.py +287 -253
- crawlo/network/__init__.py +21 -21
- crawlo/network/request.py +375 -379
- crawlo/network/response.py +569 -664
- crawlo/pipelines/__init__.py +53 -22
- crawlo/pipelines/base_pipeline.py +452 -0
- crawlo/pipelines/bloom_dedup_pipeline.py +146 -146
- crawlo/pipelines/console_pipeline.py +39 -39
- crawlo/pipelines/csv_pipeline.py +316 -316
- crawlo/pipelines/database_dedup_pipeline.py +197 -197
- crawlo/pipelines/json_pipeline.py +218 -218
- crawlo/pipelines/memory_dedup_pipeline.py +105 -105
- crawlo/pipelines/mongo_pipeline.py +140 -132
- crawlo/pipelines/mysql_pipeline.py +469 -476
- crawlo/pipelines/pipeline_manager.py +100 -100
- crawlo/pipelines/redis_dedup_pipeline.py +155 -156
- crawlo/project.py +347 -347
- crawlo/queue/__init__.py +10 -0
- crawlo/queue/pqueue.py +38 -38
- crawlo/queue/queue_manager.py +591 -525
- crawlo/queue/redis_priority_queue.py +519 -370
- crawlo/settings/__init__.py +7 -7
- crawlo/settings/default_settings.py +284 -277
- crawlo/settings/setting_manager.py +219 -219
- crawlo/spider/__init__.py +657 -657
- crawlo/stats_collector.py +81 -81
- crawlo/subscriber.py +129 -129
- crawlo/task_manager.py +138 -138
- crawlo/templates/crawlo.cfg.tmpl +10 -10
- crawlo/templates/project/__init__.py.tmpl +2 -4
- crawlo/templates/project/items.py.tmpl +13 -17
- crawlo/templates/project/middlewares.py.tmpl +38 -38
- crawlo/templates/project/pipelines.py.tmpl +35 -36
- crawlo/templates/project/settings.py.tmpl +109 -111
- crawlo/templates/project/settings_distributed.py.tmpl +156 -159
- crawlo/templates/project/settings_gentle.py.tmpl +170 -176
- crawlo/templates/project/settings_high_performance.py.tmpl +171 -177
- crawlo/templates/project/settings_minimal.py.tmpl +98 -100
- crawlo/templates/project/settings_simple.py.tmpl +168 -174
- crawlo/templates/project/spiders/__init__.py.tmpl +9 -9
- crawlo/templates/run.py.tmpl +23 -23
- crawlo/templates/spider/spider.py.tmpl +32 -40
- crawlo/templates/spiders_init.py.tmpl +5 -10
- crawlo/tools/__init__.py +86 -189
- crawlo/tools/date_tools.py +289 -289
- crawlo/tools/distributed_coordinator.py +384 -384
- crawlo/tools/scenario_adapter.py +262 -262
- crawlo/tools/text_cleaner.py +232 -232
- crawlo/utils/__init__.py +50 -50
- crawlo/utils/batch_processor.py +276 -259
- crawlo/utils/config_manager.py +442 -0
- crawlo/utils/controlled_spider_mixin.py +439 -439
- crawlo/utils/db_helper.py +250 -250
- crawlo/utils/error_handler.py +410 -410
- crawlo/utils/fingerprint.py +121 -121
- crawlo/utils/func_tools.py +82 -82
- crawlo/utils/large_scale_helper.py +344 -344
- crawlo/utils/leak_detector.py +335 -0
- crawlo/utils/log.py +79 -79
- crawlo/utils/misc.py +81 -81
- crawlo/utils/mongo_connection_pool.py +157 -0
- crawlo/utils/mysql_connection_pool.py +197 -0
- crawlo/utils/performance_monitor.py +285 -285
- crawlo/utils/queue_helper.py +175 -175
- crawlo/utils/redis_checker.py +91 -0
- crawlo/utils/redis_connection_pool.py +578 -388
- crawlo/utils/redis_key_validator.py +198 -198
- crawlo/utils/request.py +278 -256
- crawlo/utils/request_serializer.py +225 -225
- crawlo/utils/resource_manager.py +337 -0
- crawlo/utils/selector_helper.py +137 -137
- crawlo/utils/singleton.py +70 -0
- crawlo/utils/spider_loader.py +201 -201
- crawlo/utils/text_helper.py +94 -94
- crawlo/utils/{url.py → url_utils.py} +39 -39
- crawlo-1.4.7.dist-info/METADATA +689 -0
- crawlo-1.4.7.dist-info/RECORD +347 -0
- examples/__init__.py +7 -7
- tests/__init__.py +7 -7
- tests/advanced_tools_example.py +217 -275
- tests/authenticated_proxy_example.py +110 -110
- tests/baidu_performance_test.py +108 -108
- tests/baidu_test.py +59 -59
- tests/bug_check_test.py +250 -250
- tests/cleaners_example.py +160 -160
- tests/comprehensive_framework_test.py +212 -212
- tests/comprehensive_test.py +81 -81
- tests/comprehensive_testing_summary.md +186 -186
- tests/config_validation_demo.py +142 -142
- tests/controlled_spider_example.py +205 -205
- tests/date_tools_example.py +180 -180
- tests/debug_configure.py +69 -69
- tests/debug_framework_logger.py +84 -84
- tests/debug_log_config.py +126 -126
- tests/debug_log_levels.py +63 -63
- tests/debug_pipelines.py +66 -66
- tests/detailed_log_test.py +233 -233
- tests/direct_selector_helper_test.py +96 -96
- tests/distributed_dedup_test.py +467 -0
- tests/distributed_test.py +66 -66
- tests/distributed_test_debug.py +76 -76
- tests/dynamic_loading_example.py +523 -523
- tests/dynamic_loading_test.py +104 -104
- tests/error_handling_example.py +171 -171
- tests/explain_mysql_update_behavior.py +76 -76
- tests/final_comprehensive_test.py +151 -151
- tests/final_log_test.py +260 -260
- tests/final_validation_test.py +182 -182
- tests/fix_log_test.py +142 -142
- tests/framework_performance_test.py +202 -202
- tests/log_buffering_test.py +111 -111
- tests/log_generation_timing_test.py +153 -153
- tests/monitor_redis_dedup.sh +72 -0
- tests/ofweek_scrapy/ofweek_scrapy/items.py +12 -12
- tests/ofweek_scrapy/ofweek_scrapy/middlewares.py +100 -100
- tests/ofweek_scrapy/ofweek_scrapy/pipelines.py +13 -13
- tests/ofweek_scrapy/ofweek_scrapy/settings.py +84 -84
- tests/ofweek_scrapy/scrapy.cfg +11 -11
- tests/optimized_performance_test.py +211 -211
- tests/performance_comparison.py +244 -244
- tests/queue_blocking_test.py +113 -113
- tests/queue_test.py +89 -89
- tests/redis_key_validation_demo.py +130 -130
- tests/request_params_example.py +150 -150
- tests/response_improvements_example.py +144 -144
- tests/scrapy_comparison/ofweek_scrapy.py +138 -138
- tests/scrapy_comparison/scrapy_test.py +133 -133
- tests/simple_cli_test.py +55 -0
- tests/simple_command_test.py +119 -119
- tests/simple_crawlo_test.py +126 -126
- tests/simple_follow_test.py +38 -38
- tests/simple_log_test2.py +137 -137
- tests/simple_optimization_test.py +128 -128
- tests/simple_queue_type_test.py +41 -41
- tests/simple_response_selector_test.py +94 -94
- tests/simple_selector_helper_test.py +154 -154
- tests/simple_selector_test.py +207 -207
- tests/simple_spider_test.py +49 -49
- tests/simple_url_test.py +73 -73
- tests/simulate_mysql_update_test.py +139 -139
- tests/spider_log_timing_test.py +177 -177
- tests/test_advanced_tools.py +148 -148
- tests/test_all_commands.py +230 -230
- tests/test_all_pipeline_fingerprints.py +133 -133
- tests/test_all_redis_key_configs.py +145 -145
- tests/test_asyncmy_usage.py +56 -56
- tests/test_batch_processor.py +178 -178
- tests/test_cleaners.py +54 -54
- tests/test_cli_arguments.py +119 -0
- tests/test_component_factory.py +174 -174
- tests/test_config_consistency.py +80 -80
- tests/test_config_merge.py +152 -152
- tests/test_config_validator.py +182 -182
- tests/test_controlled_spider_mixin.py +79 -79
- tests/test_crawler_process_import.py +38 -38
- tests/test_crawler_process_spider_modules.py +47 -47
- tests/test_crawlo_proxy_integration.py +114 -114
- tests/test_date_tools.py +123 -123
- tests/test_dedup_fix.py +220 -220
- tests/test_dedup_pipeline_consistency.py +124 -124
- tests/test_default_header_middleware.py +313 -313
- tests/test_distributed.py +65 -65
- tests/test_double_crawlo_fix.py +204 -204
- tests/test_double_crawlo_fix_simple.py +124 -124
- tests/test_download_delay_middleware.py +221 -221
- tests/test_downloader_proxy_compatibility.py +272 -272
- tests/test_edge_cases.py +305 -305
- tests/test_encoding_core.py +56 -56
- tests/test_encoding_detection.py +126 -126
- tests/test_enhanced_error_handler.py +270 -270
- tests/test_enhanced_error_handler_comprehensive.py +245 -245
- tests/test_error_handler_compatibility.py +112 -112
- tests/test_factories.py +252 -252
- tests/test_factory_compatibility.py +196 -196
- tests/test_final_validation.py +153 -153
- tests/test_fingerprint_consistency.py +135 -135
- tests/test_fingerprint_simple.py +51 -51
- tests/test_get_component_logger.py +83 -83
- tests/test_hash_performance.py +99 -99
- tests/test_integration.py +169 -169
- tests/test_item_dedup_redis_key.py +122 -122
- tests/test_large_scale_helper.py +235 -235
- tests/test_logging_enhancements.py +374 -374
- tests/test_logging_final.py +184 -184
- tests/test_logging_integration.py +312 -312
- tests/test_logging_system.py +282 -282
- tests/test_middleware_debug.py +141 -141
- tests/test_mode_consistency.py +51 -51
- tests/test_multi_directory.py +67 -67
- tests/test_multiple_spider_modules.py +80 -80
- tests/test_mysql_pipeline_config.py +164 -164
- tests/test_mysql_pipeline_error.py +98 -98
- tests/test_mysql_pipeline_init_log.py +82 -82
- tests/test_mysql_pipeline_integration.py +132 -132
- tests/test_mysql_pipeline_refactor.py +143 -143
- tests/test_mysql_pipeline_refactor_simple.py +85 -85
- tests/test_mysql_pipeline_robustness.py +195 -195
- tests/test_mysql_pipeline_types.py +88 -88
- tests/test_mysql_update_columns.py +93 -93
- tests/test_offsite_middleware.py +244 -244
- tests/test_offsite_middleware_simple.py +203 -203
- tests/test_optimized_selector_naming.py +100 -100
- tests/test_parsel.py +29 -29
- tests/test_performance.py +327 -327
- tests/test_performance_monitor.py +115 -115
- tests/test_pipeline_fingerprint_consistency.py +86 -86
- tests/test_priority_behavior.py +211 -211
- tests/test_priority_consistency.py +151 -151
- tests/test_priority_consistency_fixed.py +249 -249
- tests/test_proxy_health_check.py +32 -32
- tests/test_proxy_middleware.py +217 -217
- tests/test_proxy_middleware_enhanced.py +212 -212
- tests/test_proxy_middleware_integration.py +142 -142
- tests/test_proxy_middleware_refactored.py +207 -207
- tests/test_proxy_only.py +83 -83
- tests/test_proxy_providers.py +56 -56
- tests/test_proxy_stats.py +19 -19
- tests/test_proxy_strategies.py +59 -59
- tests/test_proxy_with_downloader.py +152 -152
- tests/test_queue_empty_check.py +41 -41
- tests/test_queue_manager_double_crawlo.py +173 -173
- tests/test_queue_manager_redis_key.py +179 -179
- tests/test_queue_naming.py +154 -154
- tests/test_queue_type.py +106 -106
- tests/test_queue_type_redis_config_consistency.py +130 -130
- tests/test_random_headers_default.py +322 -322
- tests/test_random_headers_necessity.py +308 -308
- tests/test_random_user_agent.py +72 -72
- 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_redis_queue_name_fix.py +175 -175
- tests/test_redis_queue_type_fallback.py +129 -129
- tests/test_request_ignore_middleware.py +182 -182
- tests/test_request_params.py +111 -111
- 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_follow.py +104 -104
- tests/test_response_improvements.py +152 -152
- tests/test_response_selector_methods.py +92 -92
- tests/test_response_url_methods.py +70 -70
- tests/test_response_urljoin.py +86 -86
- tests/test_retry_middleware.py +333 -333
- tests/test_retry_middleware_realistic.py +273 -273
- tests/test_scheduler.py +252 -252
- tests/test_scheduler_config_update.py +133 -133
- tests/test_scrapy_style_encoding.py +112 -112
- tests/test_selector_helper.py +100 -100
- tests/test_selector_optimizations.py +146 -146
- tests/test_simple_response.py +61 -61
- tests/test_spider_loader.py +49 -49
- tests/test_spider_loader_comprehensive.py +69 -69
- tests/test_spider_modules.py +84 -84
- tests/test_spiders/test_spider.py +9 -9
- 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 -159
- tests/test_user_agent_randomness.py +176 -176
- tests/test_user_agents.py +96 -96
- tests/untested_features_report.md +138 -138
- tests/verify_debug.py +51 -51
- tests/verify_distributed.py +117 -117
- tests/verify_log_fix.py +111 -111
- tests/verify_mysql_warnings.py +109 -109
- crawlo/logging/async_handler.py +0 -181
- crawlo/logging/monitor.py +0 -153
- crawlo/logging/sampler.py +0 -167
- crawlo/tools/authenticated_proxy.py +0 -241
- crawlo/tools/data_formatter.py +0 -226
- crawlo/tools/data_validator.py +0 -181
- crawlo/tools/encoding_converter.py +0 -127
- crawlo/tools/network_diagnostic.py +0 -365
- crawlo/tools/request_tools.py +0 -83
- crawlo/tools/retry_mechanism.py +0 -224
- crawlo/utils/env_config.py +0 -143
- crawlo/utils/large_scale_config.py +0 -287
- crawlo/utils/system.py +0 -11
- crawlo/utils/tools.py +0 -5
- crawlo-1.4.6.dist-info/METADATA +0 -329
- crawlo-1.4.6.dist-info/RECORD +0 -361
- tests/env_config_example.py +0 -134
- tests/ofweek_scrapy/ofweek_scrapy/spiders/ofweek_spider.py +0 -162
- tests/test_authenticated_proxy.py +0 -142
- tests/test_comprehensive.py +0 -147
- tests/test_dynamic_downloaders_proxy.py +0 -125
- tests/test_dynamic_proxy.py +0 -93
- tests/test_dynamic_proxy_config.py +0 -147
- tests/test_dynamic_proxy_real.py +0 -110
- tests/test_env_config.py +0 -122
- tests/test_framework_env_usage.py +0 -104
- tests/test_large_scale_config.py +0 -113
- tests/test_proxy_api.py +0 -265
- tests/test_real_scenario_proxy.py +0 -196
- tests/tools_example.py +0 -261
- {crawlo-1.4.6.dist-info → crawlo-1.4.7.dist-info}/WHEEL +0 -0
- {crawlo-1.4.6.dist-info → crawlo-1.4.7.dist-info}/entry_points.txt +0 -0
- {crawlo-1.4.6.dist-info → crawlo-1.4.7.dist-info}/top_level.txt +0 -0
tests/test_proxy_providers.py
CHANGED
|
@@ -1,57 +1,57 @@
|
|
|
1
|
-
# tests/test_proxy_providers.py
|
|
2
|
-
import pytest
|
|
3
|
-
import pytest
|
|
4
|
-
import respx
|
|
5
|
-
from httpx import Response
|
|
6
|
-
from crawlo.proxy.providers import StaticProxyProvider, FileProxyProvider, APIProxyProvider
|
|
7
|
-
import tempfile
|
|
8
|
-
import os
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
@pytest.mark.asyncio
|
|
12
|
-
async def test_static_provider():
|
|
13
|
-
"""测试静态代理提供者"""
|
|
14
|
-
provider = StaticProxyProvider(['http://1.1.1.1:8080', 'http://2.2.2.2:8080'])
|
|
15
|
-
proxies = await provider.fetch_proxies()
|
|
16
|
-
assert len(proxies) == 2
|
|
17
|
-
assert 'http://1.1.1.1:8080' in proxies
|
|
18
|
-
assert 'http://2.2.2.2:8080' in proxies
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
@pytest.mark.asyncio
|
|
22
|
-
async def test_file_provider():
|
|
23
|
-
"""测试文件代理提供者"""
|
|
24
|
-
with tempfile.NamedTemporaryFile(mode='w', delete=False) as f:
|
|
25
|
-
f.write("http://a.com:8080\nhttp://b.com:8080\n")
|
|
26
|
-
temp_path = f.name
|
|
27
|
-
try:
|
|
28
|
-
provider = FileProxyProvider(temp_path)
|
|
29
|
-
proxies = await provider.fetch_proxies()
|
|
30
|
-
assert len(proxies) == 2
|
|
31
|
-
assert 'http://a.com:8080' in proxies
|
|
32
|
-
assert 'http://b.com:8080' in proxies
|
|
33
|
-
finally:
|
|
34
|
-
os.unlink(temp_path)
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
@pytest.mark.asyncio
|
|
38
|
-
@respx.mock
|
|
39
|
-
async def test_api_provider():
|
|
40
|
-
"""使用 respx 拦截 HTTP 请求,更简洁可靠"""
|
|
41
|
-
# 拦截 GET 请求
|
|
42
|
-
respx.get("https://api.example.com").mock(
|
|
43
|
-
return_value=Response(
|
|
44
|
-
200,
|
|
45
|
-
json=[
|
|
46
|
-
{"ip": "1.1.1.1", "port": 8080},
|
|
47
|
-
{"ip": "2.2.2.2", "port": 8080}
|
|
48
|
-
]
|
|
49
|
-
)
|
|
50
|
-
)
|
|
51
|
-
|
|
52
|
-
provider = APIProxyProvider(url="https://api.example.com")
|
|
53
|
-
proxies = await provider.fetch_proxies()
|
|
54
|
-
|
|
55
|
-
assert len(proxies) == 2
|
|
56
|
-
assert "http://1.1.1.1:8080" in proxies
|
|
1
|
+
# tests/test_proxy_providers.py
|
|
2
|
+
import pytest
|
|
3
|
+
import pytest
|
|
4
|
+
import respx
|
|
5
|
+
from httpx import Response
|
|
6
|
+
from crawlo.proxy.providers import StaticProxyProvider, FileProxyProvider, APIProxyProvider
|
|
7
|
+
import tempfile
|
|
8
|
+
import os
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
@pytest.mark.asyncio
|
|
12
|
+
async def test_static_provider():
|
|
13
|
+
"""测试静态代理提供者"""
|
|
14
|
+
provider = StaticProxyProvider(['http://1.1.1.1:8080', 'http://2.2.2.2:8080'])
|
|
15
|
+
proxies = await provider.fetch_proxies()
|
|
16
|
+
assert len(proxies) == 2
|
|
17
|
+
assert 'http://1.1.1.1:8080' in proxies
|
|
18
|
+
assert 'http://2.2.2.2:8080' in proxies
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
@pytest.mark.asyncio
|
|
22
|
+
async def test_file_provider():
|
|
23
|
+
"""测试文件代理提供者"""
|
|
24
|
+
with tempfile.NamedTemporaryFile(mode='w', delete=False) as f:
|
|
25
|
+
f.write("http://a.com:8080\nhttp://b.com:8080\n")
|
|
26
|
+
temp_path = f.name
|
|
27
|
+
try:
|
|
28
|
+
provider = FileProxyProvider(temp_path)
|
|
29
|
+
proxies = await provider.fetch_proxies()
|
|
30
|
+
assert len(proxies) == 2
|
|
31
|
+
assert 'http://a.com:8080' in proxies
|
|
32
|
+
assert 'http://b.com:8080' in proxies
|
|
33
|
+
finally:
|
|
34
|
+
os.unlink(temp_path)
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
@pytest.mark.asyncio
|
|
38
|
+
@respx.mock
|
|
39
|
+
async def test_api_provider():
|
|
40
|
+
"""使用 respx 拦截 HTTP 请求,更简洁可靠"""
|
|
41
|
+
# 拦截 GET 请求
|
|
42
|
+
respx.get("https://api.example.com").mock(
|
|
43
|
+
return_value=Response(
|
|
44
|
+
200,
|
|
45
|
+
json=[
|
|
46
|
+
{"ip": "1.1.1.1", "port": 8080},
|
|
47
|
+
{"ip": "2.2.2.2", "port": 8080}
|
|
48
|
+
]
|
|
49
|
+
)
|
|
50
|
+
)
|
|
51
|
+
|
|
52
|
+
provider = APIProxyProvider(url="https://api.example.com")
|
|
53
|
+
proxies = await provider.fetch_proxies()
|
|
54
|
+
|
|
55
|
+
assert len(proxies) == 2
|
|
56
|
+
assert "http://1.1.1.1:8080" in proxies
|
|
57
57
|
assert "http://2.2.2.2:8080" in proxies
|
tests/test_proxy_stats.py
CHANGED
|
@@ -1,20 +1,20 @@
|
|
|
1
|
-
# tests/test_proxy_stats.py
|
|
2
|
-
from crawlo.proxy.stats import ProxyStats
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
def test_proxy_stats():
|
|
6
|
-
"""测试代理统计功能"""
|
|
7
|
-
stats = ProxyStats()
|
|
8
|
-
url = 'http://proxy1:8080'
|
|
9
|
-
|
|
10
|
-
stats.record(url, 'success')
|
|
11
|
-
stats.record(url, 'success')
|
|
12
|
-
stats.record(url, 'failure')
|
|
13
|
-
|
|
14
|
-
assert stats.get(url)['success'] == 2
|
|
15
|
-
assert stats.get(url)['failure'] == 1
|
|
16
|
-
assert stats.get(url)['total'] == 3
|
|
17
|
-
|
|
18
|
-
all_data = stats.all()
|
|
19
|
-
assert url in all_data
|
|
1
|
+
# tests/test_proxy_stats.py
|
|
2
|
+
from crawlo.proxy.stats import ProxyStats
|
|
3
|
+
|
|
4
|
+
|
|
5
|
+
def test_proxy_stats():
|
|
6
|
+
"""测试代理统计功能"""
|
|
7
|
+
stats = ProxyStats()
|
|
8
|
+
url = 'http://proxy1:8080'
|
|
9
|
+
|
|
10
|
+
stats.record(url, 'success')
|
|
11
|
+
stats.record(url, 'success')
|
|
12
|
+
stats.record(url, 'failure')
|
|
13
|
+
|
|
14
|
+
assert stats.get(url)['success'] == 2
|
|
15
|
+
assert stats.get(url)['failure'] == 1
|
|
16
|
+
assert stats.get(url)['total'] == 3
|
|
17
|
+
|
|
18
|
+
all_data = stats.all()
|
|
19
|
+
assert url in all_data
|
|
20
20
|
assert all_data[url]['success'] == 2
|
tests/test_proxy_strategies.py
CHANGED
|
@@ -1,60 +1,60 @@
|
|
|
1
|
-
# tests/test_proxy_strategies.py
|
|
2
|
-
import pytest
|
|
3
|
-
from crawlo import Request
|
|
4
|
-
from crawlo.proxy.strategies import STRATEGIES
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
@pytest.fixture
|
|
8
|
-
def mock_proxies():
|
|
9
|
-
"""提供测试用的代理列表"""
|
|
10
|
-
return [
|
|
11
|
-
{'url': 'http://p1:8080'},
|
|
12
|
-
{'url': 'http://p2:8080'},
|
|
13
|
-
{'url': 'http://p3:8080'},
|
|
14
|
-
]
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
@pytest.fixture
|
|
18
|
-
def mock_stats():
|
|
19
|
-
"""提供测试用的统计信息"""
|
|
20
|
-
return {
|
|
21
|
-
'http://p1:8080': {'total': 10},
|
|
22
|
-
'http://p2:8080': {'total': 5},
|
|
23
|
-
'http://p3:8080': {'total': 1},
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
@pytest.fixture
|
|
28
|
-
def mock_request():
|
|
29
|
-
"""提供测试用的请求对象"""
|
|
30
|
-
return Request("https://example.com")
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
def test_random_strategy(mock_proxies, mock_request, mock_stats):
|
|
34
|
-
"""测试随机策略"""
|
|
35
|
-
strategy = STRATEGIES['random']
|
|
36
|
-
chosen = strategy(mock_proxies, mock_request, mock_stats)
|
|
37
|
-
assert chosen in [p['url'] for p in mock_proxies]
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
def test_least_used_strategy(mock_proxies, mock_request, mock_stats):
|
|
41
|
-
"""测试最少使用策略"""
|
|
42
|
-
strategy = STRATEGIES['least_used']
|
|
43
|
-
chosen = strategy(mock_proxies, mock_request, mock_stats)
|
|
44
|
-
assert chosen == 'http://p3:8080' # total=1
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
def test_domain_rule_strategy(mock_proxies, mock_request, mock_stats):
|
|
48
|
-
"""测试域名规则策略"""
|
|
49
|
-
from crawlo.proxy.strategies.domain_rule import domain_rule_strategy
|
|
50
|
-
request = Request("https://taobao.com/item/123")
|
|
51
|
-
rules = {'taobao.com': 'http://special:8080'}
|
|
52
|
-
|
|
53
|
-
# Monkey patch 确保有回退策略
|
|
54
|
-
old_strategy = STRATEGIES['least_used']
|
|
55
|
-
try:
|
|
56
|
-
STRATEGIES['least_used'] = lambda p, r, s: 'http://fallback:8080'
|
|
57
|
-
chosen = domain_rule_strategy(mock_proxies, request, mock_stats, rules)
|
|
58
|
-
assert chosen == 'http://special:8080'
|
|
59
|
-
finally:
|
|
1
|
+
# tests/test_proxy_strategies.py
|
|
2
|
+
import pytest
|
|
3
|
+
from crawlo import Request
|
|
4
|
+
from crawlo.proxy.strategies import STRATEGIES
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
@pytest.fixture
|
|
8
|
+
def mock_proxies():
|
|
9
|
+
"""提供测试用的代理列表"""
|
|
10
|
+
return [
|
|
11
|
+
{'url': 'http://p1:8080'},
|
|
12
|
+
{'url': 'http://p2:8080'},
|
|
13
|
+
{'url': 'http://p3:8080'},
|
|
14
|
+
]
|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
@pytest.fixture
|
|
18
|
+
def mock_stats():
|
|
19
|
+
"""提供测试用的统计信息"""
|
|
20
|
+
return {
|
|
21
|
+
'http://p1:8080': {'total': 10},
|
|
22
|
+
'http://p2:8080': {'total': 5},
|
|
23
|
+
'http://p3:8080': {'total': 1},
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
@pytest.fixture
|
|
28
|
+
def mock_request():
|
|
29
|
+
"""提供测试用的请求对象"""
|
|
30
|
+
return Request("https://example.com")
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
def test_random_strategy(mock_proxies, mock_request, mock_stats):
|
|
34
|
+
"""测试随机策略"""
|
|
35
|
+
strategy = STRATEGIES['random']
|
|
36
|
+
chosen = strategy(mock_proxies, mock_request, mock_stats)
|
|
37
|
+
assert chosen in [p['url'] for p in mock_proxies]
|
|
38
|
+
|
|
39
|
+
|
|
40
|
+
def test_least_used_strategy(mock_proxies, mock_request, mock_stats):
|
|
41
|
+
"""测试最少使用策略"""
|
|
42
|
+
strategy = STRATEGIES['least_used']
|
|
43
|
+
chosen = strategy(mock_proxies, mock_request, mock_stats)
|
|
44
|
+
assert chosen == 'http://p3:8080' # total=1
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
def test_domain_rule_strategy(mock_proxies, mock_request, mock_stats):
|
|
48
|
+
"""测试域名规则策略"""
|
|
49
|
+
from crawlo.proxy.strategies.domain_rule import domain_rule_strategy
|
|
50
|
+
request = Request("https://taobao.com/item/123")
|
|
51
|
+
rules = {'taobao.com': 'http://special:8080'}
|
|
52
|
+
|
|
53
|
+
# Monkey patch 确保有回退策略
|
|
54
|
+
old_strategy = STRATEGIES['least_used']
|
|
55
|
+
try:
|
|
56
|
+
STRATEGIES['least_used'] = lambda p, r, s: 'http://fallback:8080'
|
|
57
|
+
chosen = domain_rule_strategy(mock_proxies, request, mock_stats, rules)
|
|
58
|
+
assert chosen == 'http://special:8080'
|
|
59
|
+
finally:
|
|
60
60
|
STRATEGIES['least_used'] = old_strategy
|
|
@@ -1,153 +1,153 @@
|
|
|
1
|
-
#!/usr/bin/env python
|
|
2
|
-
# -*- coding: utf-8 -*-
|
|
3
|
-
"""
|
|
4
|
-
代理中间件与下载器配合测试脚本
|
|
5
|
-
测试指定的代理URL与下载器的兼容性
|
|
6
|
-
"""
|
|
7
|
-
|
|
8
|
-
import asyncio
|
|
9
|
-
import sys
|
|
10
|
-
import os
|
|
11
|
-
|
|
12
|
-
# 添加项目根目录到Python路径
|
|
13
|
-
sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..'))
|
|
14
|
-
|
|
15
|
-
from crawlo.settings.setting_manager import SettingManager
|
|
16
|
-
from crawlo.middleware.proxy import ProxyMiddleware
|
|
17
|
-
from crawlo.downloader.httpx_downloader import HttpXDownloader
|
|
18
|
-
from crawlo.network import Request
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
async def test_proxy_middleware():
|
|
22
|
-
"""测试代理中间件"""
|
|
23
|
-
print("=== 测试代理中间件 ===")
|
|
24
|
-
|
|
25
|
-
# 创建设置管理器
|
|
26
|
-
settings_manager = SettingManager()
|
|
27
|
-
settings = settings_manager # SettingManager实例本身就是设置对象
|
|
28
|
-
|
|
29
|
-
# 配置代理
|
|
30
|
-
settings.set('PROXY_API_URL', 'http://proxy-api.example.com/proxy/getitem/')
|
|
31
|
-
settings.set('LOG_LEVEL', 'DEBUG')
|
|
32
|
-
|
|
33
|
-
# 创建代理中间件
|
|
34
|
-
proxy_middleware = ProxyMiddleware(settings, "DEBUG")
|
|
35
|
-
|
|
36
|
-
print(f"代理中间件已创建")
|
|
37
|
-
print(f"模式: {proxy_middleware.mode}")
|
|
38
|
-
print(f"是否启用: {proxy_middleware.enabled}")
|
|
39
|
-
|
|
40
|
-
if proxy_middleware.enabled and proxy_middleware.mode == "dynamic":
|
|
41
|
-
# 测试从API获取代理
|
|
42
|
-
print("\n尝试从API获取代理...")
|
|
43
|
-
proxy = await proxy_middleware._fetch_proxy_from_api()
|
|
44
|
-
print(f"获取到的代理: {proxy}")
|
|
45
|
-
|
|
46
|
-
return proxy_middleware
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
async def test_downloader_with_proxy():
|
|
50
|
-
"""测试下载器与代理配合"""
|
|
51
|
-
print("\n=== 测试下载器与代理配合 ===")
|
|
52
|
-
|
|
53
|
-
# 创建设置管理器
|
|
54
|
-
settings_manager = SettingManager()
|
|
55
|
-
settings = settings_manager # SettingManager实例本身就是设置对象
|
|
56
|
-
|
|
57
|
-
# 配置代理
|
|
58
|
-
settings.set('PROXY_API_URL', 'http://proxy-api.example.com/proxy/getitem/')
|
|
59
|
-
settings.set('LOG_LEVEL', 'DEBUG')
|
|
60
|
-
|
|
61
|
-
# 创建代理中间件
|
|
62
|
-
proxy_middleware = ProxyMiddleware(settings, "DEBUG")
|
|
63
|
-
|
|
64
|
-
# 创建下载器
|
|
65
|
-
class MockStats:
|
|
66
|
-
def __init__(self):
|
|
67
|
-
pass
|
|
68
|
-
|
|
69
|
-
def inc_value(self, key, count=1):
|
|
70
|
-
pass
|
|
71
|
-
|
|
72
|
-
class MockSubscriber:
|
|
73
|
-
def __init__(self):
|
|
74
|
-
pass
|
|
75
|
-
|
|
76
|
-
def subscribe(self, callback, event):
|
|
77
|
-
pass
|
|
78
|
-
|
|
79
|
-
class MockSpider:
|
|
80
|
-
def __init__(self):
|
|
81
|
-
self.name = "test_spider"
|
|
82
|
-
|
|
83
|
-
class MockEngine:
|
|
84
|
-
def __init__(self):
|
|
85
|
-
pass
|
|
86
|
-
|
|
87
|
-
class MockCrawler:
|
|
88
|
-
def __init__(self, settings):
|
|
89
|
-
self.settings = settings
|
|
90
|
-
self.spider = MockSpider() # 添加spider属性
|
|
91
|
-
self.stats = MockStats() # 添加stats属性
|
|
92
|
-
self.subscriber = MockSubscriber() # 添加subscriber属性
|
|
93
|
-
self.engine = MockEngine() # 添加engine属性
|
|
94
|
-
|
|
95
|
-
crawler = MockCrawler(settings)
|
|
96
|
-
downloader = HttpXDownloader(crawler)
|
|
97
|
-
downloader.open()
|
|
98
|
-
|
|
99
|
-
# 创建测试请求
|
|
100
|
-
test_url = "https://httpbin.org/ip" # 返回客户端IP的测试站点
|
|
101
|
-
request = Request(url=test_url)
|
|
102
|
-
|
|
103
|
-
# 创建模拟爬虫
|
|
104
|
-
spider = MockSpider()
|
|
105
|
-
|
|
106
|
-
try:
|
|
107
|
-
# 通过代理中间件处理请求
|
|
108
|
-
print("通过代理中间件处理请求...")
|
|
109
|
-
await proxy_middleware.process_request(request, spider)
|
|
110
|
-
|
|
111
|
-
if request.proxy:
|
|
112
|
-
print(f"代理已设置: {request.proxy}")
|
|
113
|
-
else:
|
|
114
|
-
print("未设置代理")
|
|
115
|
-
|
|
116
|
-
# 使用下载器下载
|
|
117
|
-
print(f"开始下载: {test_url}")
|
|
118
|
-
response = await downloader.download(request)
|
|
119
|
-
|
|
120
|
-
if response:
|
|
121
|
-
print(f"下载成功,状态码: {response.status_code}")
|
|
122
|
-
print(f"响应内容: {response.text[:200]}...") # 只显示前200个字符
|
|
123
|
-
else:
|
|
124
|
-
print("下载失败,响应为空")
|
|
125
|
-
|
|
126
|
-
except Exception as e:
|
|
127
|
-
print(f"下载过程中出错: {e}")
|
|
128
|
-
import traceback
|
|
129
|
-
traceback.print_exc()
|
|
130
|
-
|
|
131
|
-
finally:
|
|
132
|
-
# 清理资源
|
|
133
|
-
try:
|
|
134
|
-
await downloader.close()
|
|
135
|
-
except:
|
|
136
|
-
pass
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
async def main():
|
|
140
|
-
"""主测试函数"""
|
|
141
|
-
print("开始测试代理中间件与下载器的配合...")
|
|
142
|
-
|
|
143
|
-
# 测试代理中间件
|
|
144
|
-
await test_proxy_middleware()
|
|
145
|
-
|
|
146
|
-
# 测试下载器与代理配合
|
|
147
|
-
await test_downloader_with_proxy()
|
|
148
|
-
|
|
149
|
-
print("\n测试完成")
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
if __name__ == "__main__":
|
|
1
|
+
#!/usr/bin/env python
|
|
2
|
+
# -*- coding: utf-8 -*-
|
|
3
|
+
"""
|
|
4
|
+
代理中间件与下载器配合测试脚本
|
|
5
|
+
测试指定的代理URL与下载器的兼容性
|
|
6
|
+
"""
|
|
7
|
+
|
|
8
|
+
import asyncio
|
|
9
|
+
import sys
|
|
10
|
+
import os
|
|
11
|
+
|
|
12
|
+
# 添加项目根目录到Python路径
|
|
13
|
+
sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..'))
|
|
14
|
+
|
|
15
|
+
from crawlo.settings.setting_manager import SettingManager
|
|
16
|
+
from crawlo.middleware.proxy import ProxyMiddleware
|
|
17
|
+
from crawlo.downloader.httpx_downloader import HttpXDownloader
|
|
18
|
+
from crawlo.network import Request
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
async def test_proxy_middleware():
|
|
22
|
+
"""测试代理中间件"""
|
|
23
|
+
print("=== 测试代理中间件 ===")
|
|
24
|
+
|
|
25
|
+
# 创建设置管理器
|
|
26
|
+
settings_manager = SettingManager()
|
|
27
|
+
settings = settings_manager # SettingManager实例本身就是设置对象
|
|
28
|
+
|
|
29
|
+
# 配置代理
|
|
30
|
+
settings.set('PROXY_API_URL', 'http://proxy-api.example.com/proxy/getitem/')
|
|
31
|
+
settings.set('LOG_LEVEL', 'DEBUG')
|
|
32
|
+
|
|
33
|
+
# 创建代理中间件
|
|
34
|
+
proxy_middleware = ProxyMiddleware(settings, "DEBUG")
|
|
35
|
+
|
|
36
|
+
print(f"代理中间件已创建")
|
|
37
|
+
print(f"模式: {proxy_middleware.mode}")
|
|
38
|
+
print(f"是否启用: {proxy_middleware.enabled}")
|
|
39
|
+
|
|
40
|
+
if proxy_middleware.enabled and proxy_middleware.mode == "dynamic":
|
|
41
|
+
# 测试从API获取代理
|
|
42
|
+
print("\n尝试从API获取代理...")
|
|
43
|
+
proxy = await proxy_middleware._fetch_proxy_from_api()
|
|
44
|
+
print(f"获取到的代理: {proxy}")
|
|
45
|
+
|
|
46
|
+
return proxy_middleware
|
|
47
|
+
|
|
48
|
+
|
|
49
|
+
async def test_downloader_with_proxy():
|
|
50
|
+
"""测试下载器与代理配合"""
|
|
51
|
+
print("\n=== 测试下载器与代理配合 ===")
|
|
52
|
+
|
|
53
|
+
# 创建设置管理器
|
|
54
|
+
settings_manager = SettingManager()
|
|
55
|
+
settings = settings_manager # SettingManager实例本身就是设置对象
|
|
56
|
+
|
|
57
|
+
# 配置代理
|
|
58
|
+
settings.set('PROXY_API_URL', 'http://proxy-api.example.com/proxy/getitem/')
|
|
59
|
+
settings.set('LOG_LEVEL', 'DEBUG')
|
|
60
|
+
|
|
61
|
+
# 创建代理中间件
|
|
62
|
+
proxy_middleware = ProxyMiddleware(settings, "DEBUG")
|
|
63
|
+
|
|
64
|
+
# 创建下载器
|
|
65
|
+
class MockStats:
|
|
66
|
+
def __init__(self):
|
|
67
|
+
pass
|
|
68
|
+
|
|
69
|
+
def inc_value(self, key, count=1):
|
|
70
|
+
pass
|
|
71
|
+
|
|
72
|
+
class MockSubscriber:
|
|
73
|
+
def __init__(self):
|
|
74
|
+
pass
|
|
75
|
+
|
|
76
|
+
def subscribe(self, callback, event):
|
|
77
|
+
pass
|
|
78
|
+
|
|
79
|
+
class MockSpider:
|
|
80
|
+
def __init__(self):
|
|
81
|
+
self.name = "test_spider"
|
|
82
|
+
|
|
83
|
+
class MockEngine:
|
|
84
|
+
def __init__(self):
|
|
85
|
+
pass
|
|
86
|
+
|
|
87
|
+
class MockCrawler:
|
|
88
|
+
def __init__(self, settings):
|
|
89
|
+
self.settings = settings
|
|
90
|
+
self.spider = MockSpider() # 添加spider属性
|
|
91
|
+
self.stats = MockStats() # 添加stats属性
|
|
92
|
+
self.subscriber = MockSubscriber() # 添加subscriber属性
|
|
93
|
+
self.engine = MockEngine() # 添加engine属性
|
|
94
|
+
|
|
95
|
+
crawler = MockCrawler(settings)
|
|
96
|
+
downloader = HttpXDownloader(crawler)
|
|
97
|
+
downloader.open()
|
|
98
|
+
|
|
99
|
+
# 创建测试请求
|
|
100
|
+
test_url = "https://httpbin.org/ip" # 返回客户端IP的测试站点
|
|
101
|
+
request = Request(url=test_url)
|
|
102
|
+
|
|
103
|
+
# 创建模拟爬虫
|
|
104
|
+
spider = MockSpider()
|
|
105
|
+
|
|
106
|
+
try:
|
|
107
|
+
# 通过代理中间件处理请求
|
|
108
|
+
print("通过代理中间件处理请求...")
|
|
109
|
+
await proxy_middleware.process_request(request, spider)
|
|
110
|
+
|
|
111
|
+
if request.proxy:
|
|
112
|
+
print(f"代理已设置: {request.proxy}")
|
|
113
|
+
else:
|
|
114
|
+
print("未设置代理")
|
|
115
|
+
|
|
116
|
+
# 使用下载器下载
|
|
117
|
+
print(f"开始下载: {test_url}")
|
|
118
|
+
response = await downloader.download(request)
|
|
119
|
+
|
|
120
|
+
if response:
|
|
121
|
+
print(f"下载成功,状态码: {response.status_code}")
|
|
122
|
+
print(f"响应内容: {response.text[:200]}...") # 只显示前200个字符
|
|
123
|
+
else:
|
|
124
|
+
print("下载失败,响应为空")
|
|
125
|
+
|
|
126
|
+
except Exception as e:
|
|
127
|
+
print(f"下载过程中出错: {e}")
|
|
128
|
+
import traceback
|
|
129
|
+
traceback.print_exc()
|
|
130
|
+
|
|
131
|
+
finally:
|
|
132
|
+
# 清理资源
|
|
133
|
+
try:
|
|
134
|
+
await downloader.close()
|
|
135
|
+
except:
|
|
136
|
+
pass
|
|
137
|
+
|
|
138
|
+
|
|
139
|
+
async def main():
|
|
140
|
+
"""主测试函数"""
|
|
141
|
+
print("开始测试代理中间件与下载器的配合...")
|
|
142
|
+
|
|
143
|
+
# 测试代理中间件
|
|
144
|
+
await test_proxy_middleware()
|
|
145
|
+
|
|
146
|
+
# 测试下载器与代理配合
|
|
147
|
+
await test_downloader_with_proxy()
|
|
148
|
+
|
|
149
|
+
print("\n测试完成")
|
|
150
|
+
|
|
151
|
+
|
|
152
|
+
if __name__ == "__main__":
|
|
153
153
|
asyncio.run(main())
|
tests/test_queue_empty_check.py
CHANGED
|
@@ -1,42 +1,42 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
|
-
# -*- coding: utf-8 -*-
|
|
3
|
-
"""
|
|
4
|
-
测试队列空检查功能
|
|
5
|
-
"""
|
|
6
|
-
import asyncio
|
|
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.queue.pqueue import SpiderPriorityQueue
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
async def test_queue_empty_check():
|
|
17
|
-
"""测试队列空检查功能"""
|
|
18
|
-
print("🚀 开始测试队列空检查功能...")
|
|
19
|
-
|
|
20
|
-
# 创建队列实例
|
|
21
|
-
queue = SpiderPriorityQueue()
|
|
22
|
-
|
|
23
|
-
# 检查空队列
|
|
24
|
-
print(f"空队列大小: {queue.qsize()}")
|
|
25
|
-
print(f"空队列是否为空: {queue.qsize() == 0}")
|
|
26
|
-
|
|
27
|
-
# 添加一个元素
|
|
28
|
-
await queue.put((1, "test"))
|
|
29
|
-
print(f"添加元素后队列大小: {queue.qsize()}")
|
|
30
|
-
print(f"添加元素后队列是否为空: {queue.qsize() == 0}")
|
|
31
|
-
|
|
32
|
-
# 获取元素
|
|
33
|
-
item = await queue.get()
|
|
34
|
-
print(f"获取元素: {item}")
|
|
35
|
-
print(f"获取元素后队列大小: {queue.qsize()}")
|
|
36
|
-
print(f"获取元素后队列是否为空: {queue.qsize() == 0}")
|
|
37
|
-
|
|
38
|
-
print("✅ 队列空检查功能测试完成!")
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
if __name__ == '__main__':
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
# -*- coding: utf-8 -*-
|
|
3
|
+
"""
|
|
4
|
+
测试队列空检查功能
|
|
5
|
+
"""
|
|
6
|
+
import asyncio
|
|
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.queue.pqueue import SpiderPriorityQueue
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
async def test_queue_empty_check():
|
|
17
|
+
"""测试队列空检查功能"""
|
|
18
|
+
print("🚀 开始测试队列空检查功能...")
|
|
19
|
+
|
|
20
|
+
# 创建队列实例
|
|
21
|
+
queue = SpiderPriorityQueue()
|
|
22
|
+
|
|
23
|
+
# 检查空队列
|
|
24
|
+
print(f"空队列大小: {queue.qsize()}")
|
|
25
|
+
print(f"空队列是否为空: {queue.qsize() == 0}")
|
|
26
|
+
|
|
27
|
+
# 添加一个元素
|
|
28
|
+
await queue.put((1, "test"))
|
|
29
|
+
print(f"添加元素后队列大小: {queue.qsize()}")
|
|
30
|
+
print(f"添加元素后队列是否为空: {queue.qsize() == 0}")
|
|
31
|
+
|
|
32
|
+
# 获取元素
|
|
33
|
+
item = await queue.get()
|
|
34
|
+
print(f"获取元素: {item}")
|
|
35
|
+
print(f"获取元素后队列大小: {queue.qsize()}")
|
|
36
|
+
print(f"获取元素后队列是否为空: {queue.qsize() == 0}")
|
|
37
|
+
|
|
38
|
+
print("✅ 队列空检查功能测试完成!")
|
|
39
|
+
|
|
40
|
+
|
|
41
|
+
if __name__ == '__main__':
|
|
42
42
|
asyncio.run(test_queue_empty_check())
|