mirror of
https://github.com/NanmiCoder/MediaCrawler.git
synced 2025-11-25 03:15:17 +08:00
chore: add copyright to every file
This commit is contained in:
@@ -1,12 +1,21 @@
|
||||
# 声明:本代码仅供学习和研究目的使用。使用者应遵守以下原则:
|
||||
# 1. 不得用于任何商业用途。
|
||||
# 2. 使用时应遵守目标平台的使用条款和robots.txt规则。
|
||||
# 3. 不得进行大规模爬取或对平台造成运营干扰。
|
||||
# 4. 应合理控制请求频率,避免给目标平台带来不必要的负担。
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright (c) 2025 relakkes@gmail.com
|
||||
#
|
||||
# This file is part of MediaCrawler project.
|
||||
# Repository: https://github.com/NanmiCoder/MediaCrawler/blob/main/test/__init__.py
|
||||
# GitHub: https://github.com/NanmiCoder
|
||||
# Licensed under NON-COMMERCIAL LEARNING LICENSE 1.1
|
||||
#
|
||||
|
||||
# 声明:本代码仅供学习和研究目的使用。使用者应遵守以下原则:
|
||||
# 1. 不得用于任何商业用途。
|
||||
# 2. 使用时应遵守目标平台的使用条款和robots.txt规则。
|
||||
# 3. 不得进行大规模爬取或对平台造成运营干扰。
|
||||
# 4. 应合理控制请求频率,避免给目标平台带来不必要的负担。
|
||||
# 5. 不得用于任何非法或不当的用途。
|
||||
#
|
||||
# 详细许可条款请参阅项目根目录下的LICENSE文件。
|
||||
# 使用本代码即表示您同意遵守上述原则和LICENSE中的所有条款。
|
||||
#
|
||||
# 详细许可条款请参阅项目根目录下的LICENSE文件。
|
||||
# 使用本代码即表示您同意遵守上述原则和LICENSE中的所有条款。
|
||||
|
||||
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
@@ -1,4 +1,21 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright (c) 2025 relakkes@gmail.com
|
||||
#
|
||||
# This file is part of MediaCrawler project.
|
||||
# Repository: https://github.com/NanmiCoder/MediaCrawler/blob/main/test/test_db_sync.py
|
||||
# GitHub: https://github.com/NanmiCoder
|
||||
# Licensed under NON-COMMERCIAL LEARNING LICENSE 1.1
|
||||
#
|
||||
# 声明:本代码仅供学习和研究目的使用。使用者应遵守以下原则:
|
||||
# 1. 不得用于任何商业用途。
|
||||
# 2. 使用时应遵守目标平台的使用条款和robots.txt规则。
|
||||
# 3. 不得进行大规模爬取或对平台造成运营干扰。
|
||||
# 4. 应合理控制请求频率,避免给目标平台带来不必要的负担。
|
||||
# 5. 不得用于任何非法或不当的用途。
|
||||
#
|
||||
# 详细许可条款请参阅项目根目录下的LICENSE文件。
|
||||
# 使用本代码即表示您同意遵守上述原则和LICENSE中的所有条款。
|
||||
|
||||
# @Author : persist-1<persist1@126.com>
|
||||
# @Time : 2025/9/8 00:02
|
||||
# @Desc : 用于将orm映射模型(database/models.py)与两种数据库实际结构进行对比,并进行更新操作(连接数据库->结构比对->差异报告->交互式同步)
|
||||
@@ -62,7 +79,7 @@ def compare_schemas(db_schema, orm_schema):
|
||||
orm_cols = set(orm_schema[table].keys())
|
||||
added_cols = orm_cols - db_cols
|
||||
deleted_cols = db_cols - orm_cols
|
||||
|
||||
|
||||
modified_cols = {}
|
||||
for col in db_cols.intersection(orm_cols):
|
||||
if db_schema[table][col] != orm_schema[table][col]:
|
||||
@@ -116,7 +133,7 @@ def print_diff(db_name, diff):
|
||||
def sync_database(engine, diff):
|
||||
"""将ORM模型同步到数据库"""
|
||||
metadata = Base.metadata
|
||||
|
||||
|
||||
# Alembic的上下文配置
|
||||
from alembic.migration import MigrationContext
|
||||
from alembic.operations import Operations
|
||||
@@ -230,4 +247,4 @@ if __name__ == "__main__":
|
||||
# 在表 tieba_note 中已修改字段: publish_time (类型变为 VARCHAR(255))
|
||||
# 在表 tieba_note 中已修改字段: tieba_id (类型变为 VARCHAR(255))
|
||||
# 在表 tieba_note 中已修改字段: note_id (类型变为 VARCHAR(644))
|
||||
# MySQL数据库同步完成。
|
||||
# MySQL数据库同步完成。
|
||||
|
||||
@@ -1,12 +1,21 @@
|
||||
# 声明:本代码仅供学习和研究目的使用。使用者应遵守以下原则:
|
||||
# 1. 不得用于任何商业用途。
|
||||
# 2. 使用时应遵守目标平台的使用条款和robots.txt规则。
|
||||
# 3. 不得进行大规模爬取或对平台造成运营干扰。
|
||||
# 4. 应合理控制请求频率,避免给目标平台带来不必要的负担。
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright (c) 2025 relakkes@gmail.com
|
||||
#
|
||||
# This file is part of MediaCrawler project.
|
||||
# Repository: https://github.com/NanmiCoder/MediaCrawler/blob/main/test/test_expiring_local_cache.py
|
||||
# GitHub: https://github.com/NanmiCoder
|
||||
# Licensed under NON-COMMERCIAL LEARNING LICENSE 1.1
|
||||
#
|
||||
|
||||
# 声明:本代码仅供学习和研究目的使用。使用者应遵守以下原则:
|
||||
# 1. 不得用于任何商业用途。
|
||||
# 2. 使用时应遵守目标平台的使用条款和robots.txt规则。
|
||||
# 3. 不得进行大规模爬取或对平台造成运营干扰。
|
||||
# 4. 应合理控制请求频率,避免给目标平台带来不必要的负担。
|
||||
# 5. 不得用于任何非法或不当的用途。
|
||||
#
|
||||
# 详细许可条款请参阅项目根目录下的LICENSE文件。
|
||||
# 使用本代码即表示您同意遵守上述原则和LICENSE中的所有条款。
|
||||
#
|
||||
# 详细许可条款请参阅项目根目录下的LICENSE文件。
|
||||
# 使用本代码即表示您同意遵守上述原则和LICENSE中的所有条款。
|
||||
|
||||
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
@@ -1,4 +1,20 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright (c) 2025 relakkes@gmail.com
|
||||
#
|
||||
# This file is part of MediaCrawler project.
|
||||
# Repository: https://github.com/NanmiCoder/MediaCrawler/blob/main/test/test_mongodb_integration.py
|
||||
# GitHub: https://github.com/NanmiCoder
|
||||
# Licensed under NON-COMMERCIAL LEARNING LICENSE 1.1
|
||||
#
|
||||
# 声明:本代码仅供学习和研究目的使用。使用者应遵守以下原则:
|
||||
# 1. 不得用于任何商业用途。
|
||||
# 2. 使用时应遵守目标平台的使用条款和robots.txt规则。
|
||||
# 3. 不得进行大规模爬取或对平台造成运营干扰。
|
||||
# 4. 应合理控制请求频率,避免给目标平台带来不必要的负担。
|
||||
# 5. 不得用于任何非法或不当的用途。
|
||||
#
|
||||
# 详细许可条款请参阅项目根目录下的LICENSE文件。
|
||||
# 使用本代码即表示您同意遵守上述原则和LICENSE中的所有条款。
|
||||
|
||||
import asyncio
|
||||
import unittest
|
||||
@@ -30,7 +46,7 @@ class TestMongoDBRealConnection(unittest.TestCase):
|
||||
def setUp(self):
|
||||
if not self.mongodb_available:
|
||||
self.skipTest("MongoDB不可用")
|
||||
|
||||
|
||||
MongoDBConnection._instance = None
|
||||
MongoDBConnection._client = None
|
||||
MongoDBConnection._db = None
|
||||
@@ -46,7 +62,7 @@ class TestMongoDBRealConnection(unittest.TestCase):
|
||||
async def cleanup():
|
||||
conn = MongoDBConnection()
|
||||
db = await conn.get_db()
|
||||
|
||||
|
||||
test_collections = [
|
||||
"test_xhs_contents",
|
||||
"test_xhs_comments",
|
||||
@@ -55,15 +71,15 @@ class TestMongoDBRealConnection(unittest.TestCase):
|
||||
"test_douyin_comments",
|
||||
"test_douyin_creators"
|
||||
]
|
||||
|
||||
|
||||
for collection_name in test_collections:
|
||||
try:
|
||||
await db[collection_name].drop()
|
||||
except:
|
||||
pass
|
||||
|
||||
|
||||
await conn.close()
|
||||
|
||||
|
||||
try:
|
||||
asyncio.run(cleanup())
|
||||
print("\n✓ 测试数据清理完成")
|
||||
@@ -75,86 +91,86 @@ class TestMongoDBRealConnection(unittest.TestCase):
|
||||
conn = MongoDBConnection()
|
||||
client = await conn.get_client()
|
||||
db = await conn.get_db()
|
||||
|
||||
|
||||
self.assertIsNotNone(client)
|
||||
self.assertIsNotNone(db)
|
||||
|
||||
|
||||
result = await db.command("ping")
|
||||
self.assertEqual(result.get("ok"), 1.0)
|
||||
|
||||
|
||||
asyncio.run(test())
|
||||
|
||||
def test_real_save_and_query(self):
|
||||
async def test():
|
||||
store = MongoDBStoreBase(collection_prefix="test_xhs")
|
||||
|
||||
|
||||
test_data = {
|
||||
"note_id": "test_note_001",
|
||||
"title": "测试笔记",
|
||||
"content": "这是一条测试内容",
|
||||
"created_at": datetime.now().isoformat()
|
||||
}
|
||||
|
||||
|
||||
result = await store.save_or_update(
|
||||
"contents",
|
||||
{"note_id": "test_note_001"},
|
||||
test_data
|
||||
)
|
||||
self.assertTrue(result)
|
||||
|
||||
|
||||
found = await store.find_one(
|
||||
"contents",
|
||||
{"note_id": "test_note_001"}
|
||||
)
|
||||
|
||||
|
||||
self.assertIsNotNone(found)
|
||||
self.assertEqual(found["note_id"], "test_note_001")
|
||||
self.assertEqual(found["title"], "测试笔记")
|
||||
|
||||
|
||||
asyncio.run(test())
|
||||
|
||||
def test_real_update(self):
|
||||
async def test():
|
||||
store = MongoDBStoreBase(collection_prefix="test_xhs")
|
||||
|
||||
|
||||
initial_data = {
|
||||
"note_id": "test_note_002",
|
||||
"title": "初始标题",
|
||||
"likes": 10
|
||||
}
|
||||
|
||||
|
||||
await store.save_or_update(
|
||||
"contents",
|
||||
{"note_id": "test_note_002"},
|
||||
initial_data
|
||||
)
|
||||
|
||||
|
||||
updated_data = {
|
||||
"note_id": "test_note_002",
|
||||
"title": "更新后的标题",
|
||||
"likes": 100
|
||||
}
|
||||
|
||||
|
||||
await store.save_or_update(
|
||||
"contents",
|
||||
{"note_id": "test_note_002"},
|
||||
updated_data
|
||||
)
|
||||
|
||||
|
||||
found = await store.find_one(
|
||||
"contents",
|
||||
{"note_id": "test_note_002"}
|
||||
)
|
||||
|
||||
|
||||
self.assertEqual(found["title"], "更新后的标题")
|
||||
self.assertEqual(found["likes"], 100)
|
||||
|
||||
|
||||
asyncio.run(test())
|
||||
|
||||
def test_real_find_many(self):
|
||||
async def test():
|
||||
store = MongoDBStoreBase(collection_prefix="test_xhs")
|
||||
|
||||
|
||||
test_user_id = "test_user_123"
|
||||
for i in range(5):
|
||||
data = {
|
||||
@@ -168,45 +184,45 @@ class TestMongoDBRealConnection(unittest.TestCase):
|
||||
{"note_id": data["note_id"]},
|
||||
data
|
||||
)
|
||||
|
||||
|
||||
results = await store.find_many(
|
||||
"contents",
|
||||
{"user_id": test_user_id}
|
||||
)
|
||||
|
||||
|
||||
self.assertGreaterEqual(len(results), 5)
|
||||
|
||||
|
||||
limited_results = await store.find_many(
|
||||
"contents",
|
||||
{"user_id": test_user_id},
|
||||
limit=3
|
||||
)
|
||||
|
||||
|
||||
self.assertEqual(len(limited_results), 3)
|
||||
|
||||
|
||||
asyncio.run(test())
|
||||
|
||||
def test_real_create_index(self):
|
||||
async def test():
|
||||
store = MongoDBStoreBase(collection_prefix="test_xhs")
|
||||
|
||||
|
||||
await store.create_index(
|
||||
"contents",
|
||||
[("note_id", 1)],
|
||||
unique=True
|
||||
)
|
||||
|
||||
|
||||
collection = await store.get_collection("contents")
|
||||
indexes = await collection.index_information()
|
||||
|
||||
|
||||
self.assertIn("note_id_1", indexes)
|
||||
|
||||
|
||||
asyncio.run(test())
|
||||
|
||||
def test_xhs_store_implementation(self):
|
||||
async def test():
|
||||
store = XhsMongoStoreImplement()
|
||||
|
||||
|
||||
note_data = {
|
||||
"note_id": "xhs_test_001",
|
||||
"user_id": "user_001",
|
||||
@@ -219,7 +235,7 @@ class TestMongoDBRealConnection(unittest.TestCase):
|
||||
"comment_count": "20"
|
||||
}
|
||||
await store.store_content(note_data)
|
||||
|
||||
|
||||
comment_data = {
|
||||
"comment_id": "comment_001",
|
||||
"note_id": "xhs_test_001",
|
||||
@@ -229,7 +245,7 @@ class TestMongoDBRealConnection(unittest.TestCase):
|
||||
"like_count": "10"
|
||||
}
|
||||
await store.store_comment(comment_data)
|
||||
|
||||
|
||||
creator_data = {
|
||||
"user_id": "user_001",
|
||||
"nickname": "测试创作者",
|
||||
@@ -238,27 +254,27 @@ class TestMongoDBRealConnection(unittest.TestCase):
|
||||
"follows": "100"
|
||||
}
|
||||
await store.store_creator(creator_data)
|
||||
|
||||
|
||||
mongo_store = store.mongo_store
|
||||
|
||||
|
||||
note = await mongo_store.find_one("contents", {"note_id": "xhs_test_001"})
|
||||
self.assertIsNotNone(note)
|
||||
self.assertEqual(note["title"], "小红书测试笔记")
|
||||
|
||||
|
||||
comment = await mongo_store.find_one("comments", {"comment_id": "comment_001"})
|
||||
self.assertIsNotNone(comment)
|
||||
self.assertEqual(comment["content"], "这是一条测试评论")
|
||||
|
||||
|
||||
creator = await mongo_store.find_one("creators", {"user_id": "user_001"})
|
||||
self.assertIsNotNone(creator)
|
||||
self.assertEqual(creator["nickname"], "测试创作者")
|
||||
|
||||
|
||||
asyncio.run(test())
|
||||
|
||||
def test_douyin_store_implementation(self):
|
||||
async def test():
|
||||
store = DouyinMongoStoreImplement()
|
||||
|
||||
|
||||
video_data = {
|
||||
"aweme_id": "dy_test_001",
|
||||
"user_id": "user_001",
|
||||
@@ -269,7 +285,7 @@ class TestMongoDBRealConnection(unittest.TestCase):
|
||||
"comment_count": "100"
|
||||
}
|
||||
await store.store_content(video_data)
|
||||
|
||||
|
||||
comment_data = {
|
||||
"comment_id": "dy_comment_001",
|
||||
"aweme_id": "dy_test_001",
|
||||
@@ -278,32 +294,32 @@ class TestMongoDBRealConnection(unittest.TestCase):
|
||||
"content": "这是一条测试评论"
|
||||
}
|
||||
await store.store_comment(comment_data)
|
||||
|
||||
|
||||
creator_data = {
|
||||
"user_id": "user_001",
|
||||
"nickname": "测试创作者",
|
||||
"desc": "这是一个测试创作者"
|
||||
}
|
||||
await store.store_creator(creator_data)
|
||||
|
||||
|
||||
mongo_store = store.mongo_store
|
||||
|
||||
|
||||
video = await mongo_store.find_one("contents", {"aweme_id": "dy_test_001"})
|
||||
self.assertIsNotNone(video)
|
||||
self.assertEqual(video["title"], "抖音测试视频")
|
||||
|
||||
|
||||
comment = await mongo_store.find_one("comments", {"comment_id": "dy_comment_001"})
|
||||
self.assertIsNotNone(comment)
|
||||
|
||||
|
||||
creator = await mongo_store.find_one("creators", {"user_id": "user_001"})
|
||||
self.assertIsNotNone(creator)
|
||||
|
||||
|
||||
asyncio.run(test())
|
||||
|
||||
def test_concurrent_operations(self):
|
||||
async def test():
|
||||
store = MongoDBStoreBase(collection_prefix="test_xhs")
|
||||
|
||||
|
||||
tasks = []
|
||||
for i in range(10):
|
||||
data = {
|
||||
@@ -317,30 +333,30 @@ class TestMongoDBRealConnection(unittest.TestCase):
|
||||
data
|
||||
)
|
||||
tasks.append(task)
|
||||
|
||||
|
||||
results = await asyncio.gather(*tasks)
|
||||
|
||||
|
||||
self.assertTrue(all(results))
|
||||
|
||||
|
||||
for i in range(10):
|
||||
found = await store.find_one(
|
||||
"contents",
|
||||
{"note_id": f"concurrent_note_{i:03d}"}
|
||||
)
|
||||
self.assertIsNotNone(found)
|
||||
|
||||
|
||||
asyncio.run(test())
|
||||
|
||||
|
||||
def run_integration_tests():
|
||||
loader = unittest.TestLoader()
|
||||
suite = unittest.TestSuite()
|
||||
|
||||
|
||||
suite.addTests(loader.loadTestsFromTestCase(TestMongoDBRealConnection))
|
||||
|
||||
|
||||
runner = unittest.TextTestRunner(verbosity=2)
|
||||
result = runner.run(suite)
|
||||
|
||||
|
||||
return result
|
||||
|
||||
|
||||
@@ -353,9 +369,9 @@ if __name__ == "__main__":
|
||||
print(f" Port: {db_config.MONGODB_PORT}")
|
||||
print(f" Database: {db_config.MONGODB_DB_NAME}")
|
||||
print("="*70)
|
||||
|
||||
|
||||
result = run_integration_tests()
|
||||
|
||||
|
||||
print("\n" + "="*70)
|
||||
print("测试统计:")
|
||||
print(f"总测试数: {result.testsRun}")
|
||||
@@ -364,5 +380,5 @@ if __name__ == "__main__":
|
||||
print(f"错误: {len(result.errors)}")
|
||||
print(f"跳过: {len(result.skipped)}")
|
||||
print("="*70)
|
||||
|
||||
|
||||
sys.exit(0 if result.wasSuccessful() else 1)
|
||||
|
||||
@@ -1,12 +1,21 @@
|
||||
# 声明:本代码仅供学习和研究目的使用。使用者应遵守以下原则:
|
||||
# 1. 不得用于任何商业用途。
|
||||
# 2. 使用时应遵守目标平台的使用条款和robots.txt规则。
|
||||
# 3. 不得进行大规模爬取或对平台造成运营干扰。
|
||||
# 4. 应合理控制请求频率,避免给目标平台带来不必要的负担。
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright (c) 2025 relakkes@gmail.com
|
||||
#
|
||||
# This file is part of MediaCrawler project.
|
||||
# Repository: https://github.com/NanmiCoder/MediaCrawler/blob/main/test/test_proxy_ip_pool.py
|
||||
# GitHub: https://github.com/NanmiCoder
|
||||
# Licensed under NON-COMMERCIAL LEARNING LICENSE 1.1
|
||||
#
|
||||
|
||||
# 声明:本代码仅供学习和研究目的使用。使用者应遵守以下原则:
|
||||
# 1. 不得用于任何商业用途。
|
||||
# 2. 使用时应遵守目标平台的使用条款和robots.txt规则。
|
||||
# 3. 不得进行大规模爬取或对平台造成运营干扰。
|
||||
# 4. 应合理控制请求频率,避免给目标平台带来不必要的负担。
|
||||
# 5. 不得用于任何非法或不当的用途。
|
||||
#
|
||||
# 详细许可条款请参阅项目根目录下的LICENSE文件。
|
||||
# 使用本代码即表示您同意遵守上述原则和LICENSE中的所有条款。
|
||||
#
|
||||
# 详细许可条款请参阅项目根目录下的LICENSE文件。
|
||||
# 使用本代码即表示您同意遵守上述原则和LICENSE中的所有条款。
|
||||
|
||||
|
||||
# -*- coding: utf-8 -*-
|
||||
@@ -27,4 +36,3 @@ class TestIpPool(IsolatedAsyncioTestCase):
|
||||
ip_proxy_info: IpInfoModel = await pool.get_proxy()
|
||||
print(ip_proxy_info)
|
||||
self.assertIsNotNone(ip_proxy_info.ip, msg="验证 ip 是否获取成功")
|
||||
|
||||
|
||||
@@ -1,12 +1,21 @@
|
||||
# 声明:本代码仅供学习和研究目的使用。使用者应遵守以下原则:
|
||||
# 1. 不得用于任何商业用途。
|
||||
# 2. 使用时应遵守目标平台的使用条款和robots.txt规则。
|
||||
# 3. 不得进行大规模爬取或对平台造成运营干扰。
|
||||
# 4. 应合理控制请求频率,避免给目标平台带来不必要的负担。
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright (c) 2025 relakkes@gmail.com
|
||||
#
|
||||
# This file is part of MediaCrawler project.
|
||||
# Repository: https://github.com/NanmiCoder/MediaCrawler/blob/main/test/test_redis_cache.py
|
||||
# GitHub: https://github.com/NanmiCoder
|
||||
# Licensed under NON-COMMERCIAL LEARNING LICENSE 1.1
|
||||
#
|
||||
|
||||
# 声明:本代码仅供学习和研究目的使用。使用者应遵守以下原则:
|
||||
# 1. 不得用于任何商业用途。
|
||||
# 2. 使用时应遵守目标平台的使用条款和robots.txt规则。
|
||||
# 3. 不得进行大规模爬取或对平台造成运营干扰。
|
||||
# 4. 应合理控制请求频率,避免给目标平台带来不必要的负担。
|
||||
# 5. 不得用于任何非法或不当的用途。
|
||||
#
|
||||
# 详细许可条款请参阅项目根目录下的LICENSE文件。
|
||||
# 使用本代码即表示您同意遵守上述原则和LICENSE中的所有条款。
|
||||
#
|
||||
# 详细许可条款请参阅项目根目录下的LICENSE文件。
|
||||
# 使用本代码即表示您同意遵守上述原则和LICENSE中的所有条款。
|
||||
|
||||
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
@@ -1,12 +1,21 @@
|
||||
# 声明:本代码仅供学习和研究目的使用。使用者应遵守以下原则:
|
||||
# 1. 不得用于任何商业用途。
|
||||
# 2. 使用时应遵守目标平台的使用条款和robots.txt规则。
|
||||
# 3. 不得进行大规模爬取或对平台造成运营干扰。
|
||||
# 4. 应合理控制请求频率,避免给目标平台带来不必要的负担。
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright (c) 2025 relakkes@gmail.com
|
||||
#
|
||||
# This file is part of MediaCrawler project.
|
||||
# Repository: https://github.com/NanmiCoder/MediaCrawler/blob/main/test/test_utils.py
|
||||
# GitHub: https://github.com/NanmiCoder
|
||||
# Licensed under NON-COMMERCIAL LEARNING LICENSE 1.1
|
||||
#
|
||||
|
||||
# 声明:本代码仅供学习和研究目的使用。使用者应遵守以下原则:
|
||||
# 1. 不得用于任何商业用途。
|
||||
# 2. 使用时应遵守目标平台的使用条款和robots.txt规则。
|
||||
# 3. 不得进行大规模爬取或对平台造成运营干扰。
|
||||
# 4. 应合理控制请求频率,避免给目标平台带来不必要的负担。
|
||||
# 5. 不得用于任何非法或不当的用途。
|
||||
#
|
||||
# 详细许可条款请参阅项目根目录下的LICENSE文件。
|
||||
# 使用本代码即表示您同意遵守上述原则和LICENSE中的所有条款。
|
||||
#
|
||||
# 详细许可条款请参阅项目根目录下的LICENSE文件。
|
||||
# 使用本代码即表示您同意遵守上述原则和LICENSE中的所有条款。
|
||||
|
||||
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
Reference in New Issue
Block a user