crawlo 1.4.1__py3-none-any.whl → 1.4.3__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 +93 -93
- 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 +341 -341
- crawlo/commands/startproject.py +436 -436
- crawlo/commands/stats.py +187 -187
- crawlo/commands/utils.py +196 -196
- crawlo/config.py +312 -312
- crawlo/config_validator.py +277 -277
- crawlo/core/__init__.py +52 -52
- crawlo/core/engine.py +438 -439
- crawlo/core/processor.py +47 -47
- crawlo/core/scheduler.py +291 -257
- crawlo/crawler.py +650 -650
- crawlo/data/__init__.py +5 -5
- crawlo/data/user_agents.py +194 -194
- crawlo/downloader/__init__.py +273 -273
- crawlo/downloader/aiohttp_downloader.py +233 -228
- crawlo/downloader/cffi_downloader.py +245 -245
- 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 +63 -63
- 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 +61 -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 +103 -103
- crawlo/factories/registry.py +84 -84
- crawlo/filters/__init__.py +154 -154
- crawlo/filters/aioredis_filter.py +257 -257
- crawlo/filters/memory_filter.py +269 -269
- crawlo/framework.py +292 -292
- crawlo/initialization/__init__.py +44 -44
- crawlo/initialization/built_in.py +425 -425
- crawlo/initialization/context.py +141 -141
- crawlo/initialization/core.py +193 -193
- crawlo/initialization/phases.py +148 -148
- crawlo/initialization/registry.py +145 -145
- 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 +45 -37
- crawlo/logging/async_handler.py +181 -0
- crawlo/logging/config.py +196 -96
- crawlo/logging/factory.py +171 -128
- crawlo/logging/manager.py +111 -111
- crawlo/logging/monitor.py +153 -0
- crawlo/logging/sampler.py +167 -0
- 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 +135 -135
- crawlo/middleware/offsite.py +123 -123
- crawlo/middleware/proxy.py +386 -386
- 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/middleware/simple_proxy.py +65 -65
- crawlo/mode_manager.py +219 -219
- crawlo/network/__init__.py +21 -21
- crawlo/network/request.py +379 -379
- crawlo/network/response.py +359 -359
- crawlo/pipelines/__init__.py +21 -21
- 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 +131 -131
- crawlo/pipelines/mysql_pipeline.py +325 -325
- crawlo/pipelines/pipeline_manager.py +100 -84
- crawlo/pipelines/redis_dedup_pipeline.py +156 -156
- crawlo/project.py +349 -338
- crawlo/queue/pqueue.py +42 -42
- crawlo/queue/queue_manager.py +526 -522
- crawlo/queue/redis_priority_queue.py +370 -367
- crawlo/settings/__init__.py +7 -7
- crawlo/settings/default_settings.py +284 -284
- crawlo/settings/setting_manager.py +219 -219
- crawlo/spider/__init__.py +657 -657
- crawlo/stats_collector.py +73 -73
- crawlo/subscriber.py +129 -129
- crawlo/task_manager.py +138 -138
- 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 +170 -170
- crawlo/templates/project/settings_distributed.py.tmpl +169 -169
- crawlo/templates/project/settings_gentle.py.tmpl +166 -166
- crawlo/templates/project/settings_high_performance.py.tmpl +167 -167
- crawlo/templates/project/settings_minimal.py.tmpl +65 -65
- crawlo/templates/project/settings_simple.py.tmpl +164 -164
- crawlo/templates/project/spiders/__init__.py.tmpl +9 -9
- crawlo/templates/run.py.tmpl +34 -34
- crawlo/templates/spider/spider.py.tmpl +143 -143
- crawlo/templates/spiders_init.py.tmpl +9 -9
- crawlo/tools/__init__.py +200 -200
- crawlo/tools/anti_crawler.py +268 -268
- crawlo/tools/authenticated_proxy.py +240 -240
- crawlo/tools/data_formatter.py +225 -225
- crawlo/tools/data_validator.py +180 -180
- crawlo/tools/date_tools.py +289 -289
- crawlo/tools/distributed_coordinator.py +384 -384
- crawlo/tools/encoding_converter.py +127 -127
- crawlo/tools/network_diagnostic.py +364 -364
- crawlo/tools/request_tools.py +82 -82
- crawlo/tools/retry_mechanism.py +224 -224
- crawlo/tools/scenario_adapter.py +262 -262
- crawlo/tools/text_cleaner.py +232 -232
- crawlo/utils/__init__.py +34 -34
- crawlo/utils/batch_processor.py +259 -259
- crawlo/utils/class_loader.py +25 -25
- 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 +165 -165
- crawlo/utils/fingerprint.py +122 -122
- 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 +79 -79
- crawlo/utils/performance_monitor.py +285 -285
- crawlo/utils/queue_helper.py +175 -175
- crawlo/utils/redis_connection_pool.py +388 -388
- crawlo/utils/redis_key_validator.py +198 -198
- crawlo/utils/request.py +267 -267
- crawlo/utils/request_serializer.py +225 -225
- 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.4.3.dist-info/METADATA +190 -0
- crawlo-1.4.3.dist-info/RECORD +326 -0
- examples/__init__.py +7 -7
- examples/test_project/__init__.py +7 -7
- examples/test_project/run.py +34 -34
- examples/test_project/test_project/__init__.py +3 -3
- examples/test_project/test_project/items.py +17 -17
- examples/test_project/test_project/middlewares.py +118 -118
- examples/test_project/test_project/pipelines.py +96 -96
- examples/test_project/test_project/settings.py +169 -169
- examples/test_project/test_project/spiders/__init__.py +9 -9
- examples/test_project/test_project/spiders/of_week_dis.py +143 -143
- tests/__init__.py +7 -7
- tests/advanced_tools_example.py +275 -275
- tests/authenticated_proxy_example.py +106 -106
- tests/baidu_performance_test.py +108 -108
- tests/baidu_test.py +59 -59
- 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/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/env_config_example.py +133 -133
- tests/error_handling_example.py +171 -171
- 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/optimized_performance_test.py +211 -211
- tests/performance_comparison.py +245 -245
- 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_command_test.py +119 -119
- tests/simple_crawlo_test.py +127 -127
- tests/simple_log_test.py +57 -57
- tests/simple_log_test2.py +137 -137
- tests/simple_optimization_test.py +128 -128
- tests/simple_queue_type_test.py +41 -41
- tests/simple_spider_test.py +49 -49
- tests/simple_test.py +47 -47
- 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_authenticated_proxy.py +141 -141
- tests/test_batch_processor.py +178 -178
- tests/test_cleaners.py +54 -54
- tests/test_component_factory.py +174 -174
- tests/test_comprehensive.py +146 -146
- 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_crawlo_proxy_integration.py +108 -108
- tests/test_date_tools.py +123 -123
- tests/test_dedup_fix.py +220 -220
- tests/test_dedup_pipeline_consistency.py +125 -0
- 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 +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_enhanced_error_handler_comprehensive.py +245 -245
- tests/test_env_config.py +121 -121
- tests/test_error_handler_compatibility.py +112 -112
- tests/test_factories.py +252 -252
- tests/test_final_validation.py +153 -153
- tests/test_fingerprint_consistency.py +135 -135
- tests/test_fingerprint_simple.py +51 -51
- tests/test_framework_env_usage.py +103 -103
- tests/test_framework_logger.py +66 -66
- tests/test_framework_startup.py +64 -64
- 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_config.py +112 -112
- tests/test_large_scale_helper.py +235 -235
- tests/test_logging_enhancements.py +375 -0
- tests/test_logging_final.py +185 -0
- tests/test_logging_integration.py +313 -0
- tests/test_logging_system.py +282 -282
- tests/test_middleware_debug.py +142 -0
- tests/test_mode_change.py +72 -72
- tests/test_mode_consistency.py +51 -51
- tests/test_offsite_middleware.py +244 -244
- tests/test_offsite_middleware_simple.py +203 -203
- 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 +212 -0
- tests/test_priority_consistency.py +152 -0
- tests/test_priority_consistency_fixed.py +250 -0
- 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 +184 -184
- tests/test_proxy_providers.py +56 -56
- tests/test_proxy_stats.py +19 -19
- tests/test_proxy_strategies.py +59 -59
- 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 +131 -0
- tests/test_random_headers_default.py +323 -0
- tests/test_random_headers_necessity.py +309 -0
- tests/test_random_user_agent.py +72 -72
- 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_redis_queue_name_fix.py +175 -175
- tests/test_redis_queue_type_fallback.py +130 -0
- 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_improvements.py +152 -152
- tests/test_retry_middleware.py +334 -242
- tests/test_retry_middleware_realistic.py +274 -0
- 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 -159
- tests/test_user_agent_randomness.py +177 -0
- tests/test_user_agents.py +96 -96
- tests/tools_example.py +260 -260
- 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
- crawlo-1.4.1.dist-info/METADATA +0 -1199
- crawlo-1.4.1.dist-info/RECORD +0 -309
- {crawlo-1.4.1.dist-info → crawlo-1.4.3.dist-info}/WHEEL +0 -0
- {crawlo-1.4.1.dist-info → crawlo-1.4.3.dist-info}/entry_points.txt +0 -0
- {crawlo-1.4.1.dist-info → crawlo-1.4.3.dist-info}/top_level.txt +0 -0
|
@@ -1,222 +1,222 @@
|
|
|
1
|
-
#!/usr/bin/python
|
|
2
|
-
# -*- coding:UTF-8 -*-
|
|
3
|
-
"""
|
|
4
|
-
DownloadDelayMiddleware 测试文件
|
|
5
|
-
用于测试下载延迟中间件的功能
|
|
6
|
-
"""
|
|
7
|
-
|
|
8
|
-
import asyncio
|
|
9
|
-
import unittest
|
|
10
|
-
from unittest.mock import Mock, patch
|
|
11
|
-
|
|
12
|
-
from crawlo.middleware.download_delay import DownloadDelayMiddleware
|
|
13
|
-
from crawlo.exceptions import NotConfiguredError
|
|
14
|
-
from crawlo.settings.setting_manager import SettingManager
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
class MockLogger:
|
|
18
|
-
"""Mock Logger 类,用于测试日志输出"""
|
|
19
|
-
def __init__(self, name, level=None):
|
|
20
|
-
self.name = name
|
|
21
|
-
self.level = level
|
|
22
|
-
self.logs = []
|
|
23
|
-
|
|
24
|
-
def debug(self, msg):
|
|
25
|
-
self.logs.append(('debug', msg))
|
|
26
|
-
|
|
27
|
-
def info(self, msg):
|
|
28
|
-
self.logs.append(('info', msg))
|
|
29
|
-
|
|
30
|
-
def warning(self, msg):
|
|
31
|
-
self.logs.append(('warning', msg))
|
|
32
|
-
|
|
33
|
-
def error(self, msg):
|
|
34
|
-
self.logs.append(('error', msg))
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
class MockStats:
|
|
38
|
-
"""Mock Stats 类,用于测试统计信息"""
|
|
39
|
-
def __init__(self):
|
|
40
|
-
self.stats = {}
|
|
41
|
-
|
|
42
|
-
def inc_value(self, key, value=1):
|
|
43
|
-
if key in self.stats:
|
|
44
|
-
self.stats[key] += value
|
|
45
|
-
else:
|
|
46
|
-
self.stats[key] = value
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
class TestDownloadDelayMiddleware(unittest.TestCase):
|
|
50
|
-
"""DownloadDelayMiddleware 测试类"""
|
|
51
|
-
|
|
52
|
-
def setUp(self):
|
|
53
|
-
"""测试前准备"""
|
|
54
|
-
# 创建设置管理器
|
|
55
|
-
self.settings = SettingManager()
|
|
56
|
-
|
|
57
|
-
# 创建爬虫模拟对象
|
|
58
|
-
self.crawler = Mock()
|
|
59
|
-
self.crawler.settings = self.settings
|
|
60
|
-
|
|
61
|
-
# 创建请求和爬虫模拟对象
|
|
62
|
-
self.request = Mock()
|
|
63
|
-
self.spider = Mock()
|
|
64
|
-
|
|
65
|
-
@patch('crawlo.utils.log.get_logger')
|
|
66
|
-
def test_middleware_initialization_without_delay(self, mock_get_logger):
|
|
67
|
-
"""测试没有设置DOWNLOAD_DELAY时中间件初始化"""
|
|
68
|
-
# 设置DOWNLOAD_DELAY为0
|
|
69
|
-
self.settings.set('DOWNLOAD_DELAY', 0)
|
|
70
|
-
mock_get_logger.return_value = MockLogger('DownloadDelayMiddleware')
|
|
71
|
-
|
|
72
|
-
# 应该抛出NotConfiguredError异常
|
|
73
|
-
with self.assertRaises(NotConfiguredError) as context:
|
|
74
|
-
DownloadDelayMiddleware.create_instance(self.crawler)
|
|
75
|
-
|
|
76
|
-
self.assertIn("DOWNLOAD_DELAY not set or is zero", str(context.exception))
|
|
77
|
-
|
|
78
|
-
@patch('crawlo.utils.log.get_logger')
|
|
79
|
-
def test_middleware_initialization_with_delay(self, mock_get_logger):
|
|
80
|
-
"""测试正确设置DOWNLOAD_DELAY时中间件初始化"""
|
|
81
|
-
# 设置DOWNLOAD_DELAY
|
|
82
|
-
self.settings.set('DOWNLOAD_DELAY', 2.0)
|
|
83
|
-
self.settings.set('RANDOMNESS', False)
|
|
84
|
-
self.settings.set('RANDOM_RANGE', [0.5, 1.5])
|
|
85
|
-
self.settings.set('LOG_LEVEL', 'INFO')
|
|
86
|
-
|
|
87
|
-
mock_get_logger.return_value = MockLogger('DownloadDelayMiddleware')
|
|
88
|
-
|
|
89
|
-
# 应该正常创建实例
|
|
90
|
-
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
91
|
-
self.assertIsInstance(middleware, DownloadDelayMiddleware)
|
|
92
|
-
self.assertEqual(middleware.delay, 2.0)
|
|
93
|
-
self.assertFalse(middleware.randomness)
|
|
94
|
-
|
|
95
|
-
@patch('crawlo.utils.log.get_logger')
|
|
96
|
-
def test_middleware_initialization_with_randomness(self, mock_get_logger):
|
|
97
|
-
"""测试启用随机延迟时中间件初始化"""
|
|
98
|
-
# 设置DOWNLOAD_DELAY和随机配置
|
|
99
|
-
self.settings.set('DOWNLOAD_DELAY', 1.0)
|
|
100
|
-
self.settings.set('RANDOMNESS', True)
|
|
101
|
-
self.settings.set('RANDOM_RANGE', [0.5, 2.0])
|
|
102
|
-
self.settings.set('LOG_LEVEL', 'INFO')
|
|
103
|
-
|
|
104
|
-
mock_get_logger.return_value = MockLogger('DownloadDelayMiddleware')
|
|
105
|
-
|
|
106
|
-
# 应该正常创建实例
|
|
107
|
-
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
108
|
-
self.assertIsInstance(middleware, DownloadDelayMiddleware)
|
|
109
|
-
self.assertEqual(middleware.delay, 1.0)
|
|
110
|
-
self.assertTrue(middleware.randomness)
|
|
111
|
-
self.assertEqual(middleware.floor, 0.5)
|
|
112
|
-
self.assertEqual(middleware.upper, 2.0)
|
|
113
|
-
|
|
114
|
-
@patch('crawlo.utils.log.get_logger')
|
|
115
|
-
def test_middleware_initialization_with_invalid_random_range(self, mock_get_logger):
|
|
116
|
-
"""测试随机范围配置无效时中间件初始化"""
|
|
117
|
-
# 设置DOWNLOAD_DELAY和无效的随机配置
|
|
118
|
-
self.settings.set('DOWNLOAD_DELAY', 1.0)
|
|
119
|
-
self.settings.set('RANDOMNESS', True)
|
|
120
|
-
self.settings.set('RANDOM_RANGE', ['invalid', 'range'])
|
|
121
|
-
self.settings.set('LOG_LEVEL', 'INFO')
|
|
122
|
-
|
|
123
|
-
mock_get_logger.return_value = MockLogger('DownloadDelayMiddleware')
|
|
124
|
-
|
|
125
|
-
# 应该正常创建实例,使用默认随机范围
|
|
126
|
-
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
127
|
-
self.assertIsInstance(middleware, DownloadDelayMiddleware)
|
|
128
|
-
self.assertEqual(middleware.floor, 0.5)
|
|
129
|
-
self.assertEqual(middleware.upper, 1.5)
|
|
130
|
-
|
|
131
|
-
@patch('crawlo.utils.log.get_logger')
|
|
132
|
-
def test_middleware_initialization_with_incomplete_random_range(self, mock_get_logger):
|
|
133
|
-
"""测试随机范围配置不完整时中间件初始化"""
|
|
134
|
-
# 设置DOWNLOAD_DELAY和不完整的随机配置
|
|
135
|
-
self.settings.set('DOWNLOAD_DELAY', 1.0)
|
|
136
|
-
self.settings.set('RANDOMNESS', True)
|
|
137
|
-
self.settings.set('RANDOM_RANGE', [0.8]) # 只有一个值
|
|
138
|
-
self.settings.set('LOG_LEVEL', 'INFO')
|
|
139
|
-
|
|
140
|
-
mock_get_logger.return_value = MockLogger('DownloadDelayMiddleware')
|
|
141
|
-
|
|
142
|
-
# 应该正常创建实例,使用默认随机范围
|
|
143
|
-
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
144
|
-
self.assertIsInstance(middleware, DownloadDelayMiddleware)
|
|
145
|
-
self.assertEqual(middleware.floor, 0.5)
|
|
146
|
-
self.assertEqual(middleware.upper, 1.5)
|
|
147
|
-
|
|
148
|
-
@patch('crawlo.middleware.download_delay.sleep')
|
|
149
|
-
@patch('crawlo.utils.log.get_logger')
|
|
150
|
-
def test_process_request_without_randomness(self, mock_get_logger, mock_sleep):
|
|
151
|
-
"""测试不启用随机延迟时的请求处理"""
|
|
152
|
-
# 设置DOWNLOAD_DELAY
|
|
153
|
-
self.settings.set('DOWNLOAD_DELAY', 1.5)
|
|
154
|
-
self.settings.set('RANDOMNESS', False)
|
|
155
|
-
self.settings.set('LOG_LEVEL', 'DEBUG') # 使用DEBUG级别以启用日志
|
|
156
|
-
|
|
157
|
-
mock_logger = MockLogger('DownloadDelayMiddleware')
|
|
158
|
-
mock_get_logger.return_value = mock_logger
|
|
159
|
-
|
|
160
|
-
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
161
|
-
|
|
162
|
-
# 执行请求处理
|
|
163
|
-
asyncio.run(middleware.process_request(self.request, self.spider))
|
|
164
|
-
|
|
165
|
-
# 验证sleep被调用且参数正确
|
|
166
|
-
mock_sleep.assert_called_once_with(1.5)
|
|
167
|
-
|
|
168
|
-
@patch('crawlo.middleware.download_delay.sleep')
|
|
169
|
-
@patch('crawlo.middleware.download_delay.uniform')
|
|
170
|
-
@patch('crawlo.utils.log.get_logger')
|
|
171
|
-
def test_process_request_with_randomness(self, mock_get_logger, mock_uniform, mock_sleep):
|
|
172
|
-
"""测试启用随机延迟时的请求处理"""
|
|
173
|
-
# 设置DOWNLOAD_DELAY和随机配置
|
|
174
|
-
self.settings.set('DOWNLOAD_DELAY', 2.0)
|
|
175
|
-
self.settings.set('RANDOMNESS', True)
|
|
176
|
-
self.settings.set('RANDOM_RANGE', [0.5, 1.5])
|
|
177
|
-
self.settings.set('LOG_LEVEL', 'DEBUG') # 使用DEBUG级别以启用日志
|
|
178
|
-
|
|
179
|
-
mock_logger = MockLogger('DownloadDelayMiddleware')
|
|
180
|
-
mock_get_logger.return_value = mock_logger
|
|
181
|
-
mock_uniform.return_value = 2.5 # 模拟随机数返回2.5
|
|
182
|
-
|
|
183
|
-
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
184
|
-
|
|
185
|
-
# 执行请求处理
|
|
186
|
-
asyncio.run(middleware.process_request(self.request, self.spider))
|
|
187
|
-
|
|
188
|
-
# 验证uniform被调用且参数正确
|
|
189
|
-
mock_uniform.assert_called_once_with(1.0, 3.0) # 2.0*0.5=1.0, 2.0*1.5=3.0
|
|
190
|
-
# 验证sleep被调用且参数正确
|
|
191
|
-
mock_sleep.assert_called_once_with(2.5)
|
|
192
|
-
|
|
193
|
-
@patch('crawlo.middleware.download_delay.sleep')
|
|
194
|
-
@patch('crawlo.utils.log.get_logger')
|
|
195
|
-
def test_process_request_with_stats(self, mock_get_logger, mock_sleep):
|
|
196
|
-
"""测试带统计信息的请求处理"""
|
|
197
|
-
# 设置DOWNLOAD_DELAY
|
|
198
|
-
self.settings.set('DOWNLOAD_DELAY', 1.0)
|
|
199
|
-
self.settings.set('RANDOMNESS', False)
|
|
200
|
-
self.settings.set('LOG_LEVEL', 'INFO')
|
|
201
|
-
|
|
202
|
-
# 添加统计收集器到爬虫
|
|
203
|
-
mock_stats = MockStats()
|
|
204
|
-
self.crawler.stats = mock_stats
|
|
205
|
-
|
|
206
|
-
mock_logger = MockLogger('DownloadDelayMiddleware')
|
|
207
|
-
mock_get_logger.return_value = mock_logger
|
|
208
|
-
|
|
209
|
-
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
210
|
-
|
|
211
|
-
# 执行请求处理
|
|
212
|
-
asyncio.run(middleware.process_request(self.request, self.spider))
|
|
213
|
-
|
|
214
|
-
# 验证统计信息
|
|
215
|
-
self.assertIn('download_delay/fixed_count', mock_stats.stats)
|
|
216
|
-
self.assertEqual(mock_stats.stats['download_delay/fixed_count'], 1)
|
|
217
|
-
self.assertIn('download_delay/fixed_total_time', mock_stats.stats)
|
|
218
|
-
self.assertEqual(mock_stats.stats['download_delay/fixed_total_time'], 1.0)
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
if __name__ == '__main__':
|
|
1
|
+
#!/usr/bin/python
|
|
2
|
+
# -*- coding:UTF-8 -*-
|
|
3
|
+
"""
|
|
4
|
+
DownloadDelayMiddleware 测试文件
|
|
5
|
+
用于测试下载延迟中间件的功能
|
|
6
|
+
"""
|
|
7
|
+
|
|
8
|
+
import asyncio
|
|
9
|
+
import unittest
|
|
10
|
+
from unittest.mock import Mock, patch
|
|
11
|
+
|
|
12
|
+
from crawlo.middleware.download_delay import DownloadDelayMiddleware
|
|
13
|
+
from crawlo.exceptions import NotConfiguredError
|
|
14
|
+
from crawlo.settings.setting_manager import SettingManager
|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
class MockLogger:
|
|
18
|
+
"""Mock Logger 类,用于测试日志输出"""
|
|
19
|
+
def __init__(self, name, level=None):
|
|
20
|
+
self.name = name
|
|
21
|
+
self.level = level
|
|
22
|
+
self.logs = []
|
|
23
|
+
|
|
24
|
+
def debug(self, msg):
|
|
25
|
+
self.logs.append(('debug', msg))
|
|
26
|
+
|
|
27
|
+
def info(self, msg):
|
|
28
|
+
self.logs.append(('info', msg))
|
|
29
|
+
|
|
30
|
+
def warning(self, msg):
|
|
31
|
+
self.logs.append(('warning', msg))
|
|
32
|
+
|
|
33
|
+
def error(self, msg):
|
|
34
|
+
self.logs.append(('error', msg))
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
class MockStats:
|
|
38
|
+
"""Mock Stats 类,用于测试统计信息"""
|
|
39
|
+
def __init__(self):
|
|
40
|
+
self.stats = {}
|
|
41
|
+
|
|
42
|
+
def inc_value(self, key, value=1):
|
|
43
|
+
if key in self.stats:
|
|
44
|
+
self.stats[key] += value
|
|
45
|
+
else:
|
|
46
|
+
self.stats[key] = value
|
|
47
|
+
|
|
48
|
+
|
|
49
|
+
class TestDownloadDelayMiddleware(unittest.TestCase):
|
|
50
|
+
"""DownloadDelayMiddleware 测试类"""
|
|
51
|
+
|
|
52
|
+
def setUp(self):
|
|
53
|
+
"""测试前准备"""
|
|
54
|
+
# 创建设置管理器
|
|
55
|
+
self.settings = SettingManager()
|
|
56
|
+
|
|
57
|
+
# 创建爬虫模拟对象
|
|
58
|
+
self.crawler = Mock()
|
|
59
|
+
self.crawler.settings = self.settings
|
|
60
|
+
|
|
61
|
+
# 创建请求和爬虫模拟对象
|
|
62
|
+
self.request = Mock()
|
|
63
|
+
self.spider = Mock()
|
|
64
|
+
|
|
65
|
+
@patch('crawlo.utils.log.get_logger')
|
|
66
|
+
def test_middleware_initialization_without_delay(self, mock_get_logger):
|
|
67
|
+
"""测试没有设置DOWNLOAD_DELAY时中间件初始化"""
|
|
68
|
+
# 设置DOWNLOAD_DELAY为0
|
|
69
|
+
self.settings.set('DOWNLOAD_DELAY', 0)
|
|
70
|
+
mock_get_logger.return_value = MockLogger('DownloadDelayMiddleware')
|
|
71
|
+
|
|
72
|
+
# 应该抛出NotConfiguredError异常
|
|
73
|
+
with self.assertRaises(NotConfiguredError) as context:
|
|
74
|
+
DownloadDelayMiddleware.create_instance(self.crawler)
|
|
75
|
+
|
|
76
|
+
self.assertIn("DOWNLOAD_DELAY not set or is zero", str(context.exception))
|
|
77
|
+
|
|
78
|
+
@patch('crawlo.utils.log.get_logger')
|
|
79
|
+
def test_middleware_initialization_with_delay(self, mock_get_logger):
|
|
80
|
+
"""测试正确设置DOWNLOAD_DELAY时中间件初始化"""
|
|
81
|
+
# 设置DOWNLOAD_DELAY
|
|
82
|
+
self.settings.set('DOWNLOAD_DELAY', 2.0)
|
|
83
|
+
self.settings.set('RANDOMNESS', False)
|
|
84
|
+
self.settings.set('RANDOM_RANGE', [0.5, 1.5])
|
|
85
|
+
self.settings.set('LOG_LEVEL', 'INFO')
|
|
86
|
+
|
|
87
|
+
mock_get_logger.return_value = MockLogger('DownloadDelayMiddleware')
|
|
88
|
+
|
|
89
|
+
# 应该正常创建实例
|
|
90
|
+
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
91
|
+
self.assertIsInstance(middleware, DownloadDelayMiddleware)
|
|
92
|
+
self.assertEqual(middleware.delay, 2.0)
|
|
93
|
+
self.assertFalse(middleware.randomness)
|
|
94
|
+
|
|
95
|
+
@patch('crawlo.utils.log.get_logger')
|
|
96
|
+
def test_middleware_initialization_with_randomness(self, mock_get_logger):
|
|
97
|
+
"""测试启用随机延迟时中间件初始化"""
|
|
98
|
+
# 设置DOWNLOAD_DELAY和随机配置
|
|
99
|
+
self.settings.set('DOWNLOAD_DELAY', 1.0)
|
|
100
|
+
self.settings.set('RANDOMNESS', True)
|
|
101
|
+
self.settings.set('RANDOM_RANGE', [0.5, 2.0])
|
|
102
|
+
self.settings.set('LOG_LEVEL', 'INFO')
|
|
103
|
+
|
|
104
|
+
mock_get_logger.return_value = MockLogger('DownloadDelayMiddleware')
|
|
105
|
+
|
|
106
|
+
# 应该正常创建实例
|
|
107
|
+
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
108
|
+
self.assertIsInstance(middleware, DownloadDelayMiddleware)
|
|
109
|
+
self.assertEqual(middleware.delay, 1.0)
|
|
110
|
+
self.assertTrue(middleware.randomness)
|
|
111
|
+
self.assertEqual(middleware.floor, 0.5)
|
|
112
|
+
self.assertEqual(middleware.upper, 2.0)
|
|
113
|
+
|
|
114
|
+
@patch('crawlo.utils.log.get_logger')
|
|
115
|
+
def test_middleware_initialization_with_invalid_random_range(self, mock_get_logger):
|
|
116
|
+
"""测试随机范围配置无效时中间件初始化"""
|
|
117
|
+
# 设置DOWNLOAD_DELAY和无效的随机配置
|
|
118
|
+
self.settings.set('DOWNLOAD_DELAY', 1.0)
|
|
119
|
+
self.settings.set('RANDOMNESS', True)
|
|
120
|
+
self.settings.set('RANDOM_RANGE', ['invalid', 'range'])
|
|
121
|
+
self.settings.set('LOG_LEVEL', 'INFO')
|
|
122
|
+
|
|
123
|
+
mock_get_logger.return_value = MockLogger('DownloadDelayMiddleware')
|
|
124
|
+
|
|
125
|
+
# 应该正常创建实例,使用默认随机范围
|
|
126
|
+
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
127
|
+
self.assertIsInstance(middleware, DownloadDelayMiddleware)
|
|
128
|
+
self.assertEqual(middleware.floor, 0.5)
|
|
129
|
+
self.assertEqual(middleware.upper, 1.5)
|
|
130
|
+
|
|
131
|
+
@patch('crawlo.utils.log.get_logger')
|
|
132
|
+
def test_middleware_initialization_with_incomplete_random_range(self, mock_get_logger):
|
|
133
|
+
"""测试随机范围配置不完整时中间件初始化"""
|
|
134
|
+
# 设置DOWNLOAD_DELAY和不完整的随机配置
|
|
135
|
+
self.settings.set('DOWNLOAD_DELAY', 1.0)
|
|
136
|
+
self.settings.set('RANDOMNESS', True)
|
|
137
|
+
self.settings.set('RANDOM_RANGE', [0.8]) # 只有一个值
|
|
138
|
+
self.settings.set('LOG_LEVEL', 'INFO')
|
|
139
|
+
|
|
140
|
+
mock_get_logger.return_value = MockLogger('DownloadDelayMiddleware')
|
|
141
|
+
|
|
142
|
+
# 应该正常创建实例,使用默认随机范围
|
|
143
|
+
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
144
|
+
self.assertIsInstance(middleware, DownloadDelayMiddleware)
|
|
145
|
+
self.assertEqual(middleware.floor, 0.5)
|
|
146
|
+
self.assertEqual(middleware.upper, 1.5)
|
|
147
|
+
|
|
148
|
+
@patch('crawlo.middleware.download_delay.sleep')
|
|
149
|
+
@patch('crawlo.utils.log.get_logger')
|
|
150
|
+
def test_process_request_without_randomness(self, mock_get_logger, mock_sleep):
|
|
151
|
+
"""测试不启用随机延迟时的请求处理"""
|
|
152
|
+
# 设置DOWNLOAD_DELAY
|
|
153
|
+
self.settings.set('DOWNLOAD_DELAY', 1.5)
|
|
154
|
+
self.settings.set('RANDOMNESS', False)
|
|
155
|
+
self.settings.set('LOG_LEVEL', 'DEBUG') # 使用DEBUG级别以启用日志
|
|
156
|
+
|
|
157
|
+
mock_logger = MockLogger('DownloadDelayMiddleware')
|
|
158
|
+
mock_get_logger.return_value = mock_logger
|
|
159
|
+
|
|
160
|
+
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
161
|
+
|
|
162
|
+
# 执行请求处理
|
|
163
|
+
asyncio.run(middleware.process_request(self.request, self.spider))
|
|
164
|
+
|
|
165
|
+
# 验证sleep被调用且参数正确
|
|
166
|
+
mock_sleep.assert_called_once_with(1.5)
|
|
167
|
+
|
|
168
|
+
@patch('crawlo.middleware.download_delay.sleep')
|
|
169
|
+
@patch('crawlo.middleware.download_delay.uniform')
|
|
170
|
+
@patch('crawlo.utils.log.get_logger')
|
|
171
|
+
def test_process_request_with_randomness(self, mock_get_logger, mock_uniform, mock_sleep):
|
|
172
|
+
"""测试启用随机延迟时的请求处理"""
|
|
173
|
+
# 设置DOWNLOAD_DELAY和随机配置
|
|
174
|
+
self.settings.set('DOWNLOAD_DELAY', 2.0)
|
|
175
|
+
self.settings.set('RANDOMNESS', True)
|
|
176
|
+
self.settings.set('RANDOM_RANGE', [0.5, 1.5])
|
|
177
|
+
self.settings.set('LOG_LEVEL', 'DEBUG') # 使用DEBUG级别以启用日志
|
|
178
|
+
|
|
179
|
+
mock_logger = MockLogger('DownloadDelayMiddleware')
|
|
180
|
+
mock_get_logger.return_value = mock_logger
|
|
181
|
+
mock_uniform.return_value = 2.5 # 模拟随机数返回2.5
|
|
182
|
+
|
|
183
|
+
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
184
|
+
|
|
185
|
+
# 执行请求处理
|
|
186
|
+
asyncio.run(middleware.process_request(self.request, self.spider))
|
|
187
|
+
|
|
188
|
+
# 验证uniform被调用且参数正确
|
|
189
|
+
mock_uniform.assert_called_once_with(1.0, 3.0) # 2.0*0.5=1.0, 2.0*1.5=3.0
|
|
190
|
+
# 验证sleep被调用且参数正确
|
|
191
|
+
mock_sleep.assert_called_once_with(2.5)
|
|
192
|
+
|
|
193
|
+
@patch('crawlo.middleware.download_delay.sleep')
|
|
194
|
+
@patch('crawlo.utils.log.get_logger')
|
|
195
|
+
def test_process_request_with_stats(self, mock_get_logger, mock_sleep):
|
|
196
|
+
"""测试带统计信息的请求处理"""
|
|
197
|
+
# 设置DOWNLOAD_DELAY
|
|
198
|
+
self.settings.set('DOWNLOAD_DELAY', 1.0)
|
|
199
|
+
self.settings.set('RANDOMNESS', False)
|
|
200
|
+
self.settings.set('LOG_LEVEL', 'INFO')
|
|
201
|
+
|
|
202
|
+
# 添加统计收集器到爬虫
|
|
203
|
+
mock_stats = MockStats()
|
|
204
|
+
self.crawler.stats = mock_stats
|
|
205
|
+
|
|
206
|
+
mock_logger = MockLogger('DownloadDelayMiddleware')
|
|
207
|
+
mock_get_logger.return_value = mock_logger
|
|
208
|
+
|
|
209
|
+
middleware = DownloadDelayMiddleware.create_instance(self.crawler)
|
|
210
|
+
|
|
211
|
+
# 执行请求处理
|
|
212
|
+
asyncio.run(middleware.process_request(self.request, self.spider))
|
|
213
|
+
|
|
214
|
+
# 验证统计信息
|
|
215
|
+
self.assertIn('download_delay/fixed_count', mock_stats.stats)
|
|
216
|
+
self.assertEqual(mock_stats.stats['download_delay/fixed_count'], 1)
|
|
217
|
+
self.assertIn('download_delay/fixed_total_time', mock_stats.stats)
|
|
218
|
+
self.assertEqual(mock_stats.stats['download_delay/fixed_total_time'], 1.0)
|
|
219
|
+
|
|
220
|
+
|
|
221
|
+
if __name__ == '__main__':
|
|
222
222
|
unittest.main()
|