8f7f082c3d
- 重构项目目录结构,将功能模块移至 modules/ 目录 - 创建平台同步基础架构,包括发布器基类和 GitHub 发布器 - 新增 UI 状态管理模块 (modules/ui/state.py) 统一管理会话状态 - 更新依赖配置,添加平台同步所需依赖 (httpx, pyperclip) - 整理文档结构,将所有文档分类移至 docs/ 目录 - 添加 .cursorrules 文件定义项目开发规范 - 清理根目录重复文件,保持项目结构整洁
836 lines
27 KiB
Markdown
836 lines
27 KiB
Markdown
# 平台文章同步功能实现指南
|
||
|
||
> 实现20个平台的文章同步功能:8个API平台 + 12个一键复制平台
|
||
|
||
## 📋 目录
|
||
|
||
1. [技术架构设计](#技术架构设计)
|
||
2. [数据库设计](#数据库设计)
|
||
3. [模块划分](#模块划分)
|
||
4. [核心代码实现](#核心代码实现)
|
||
5. [实施步骤](#实施步骤)
|
||
6. [测试方案](#测试方案)
|
||
|
||
---
|
||
|
||
## 🏗️ 技术架构设计
|
||
|
||
### 整体架构
|
||
|
||
```
|
||
┌─────────────────────────────────────────────────────────┐
|
||
│ Streamlit UI Layer │
|
||
│ (平台配置、发布管理、状态展示、一键复制) │
|
||
└─────────────────────────────────────────────────────────┘
|
||
│
|
||
┌─────────────────────────────────────────────────────────┐
|
||
│ Platform Sync Manager │
|
||
│ (统一发布接口、任务队列、状态管理) │
|
||
└─────────────────────────────────────────────────────────┘
|
||
│
|
||
┌───────────────────┴───────────────────┐
|
||
│ │
|
||
┌───────────────────┐ ┌──────────────────────┐
|
||
│ API Publishers │ │ Copy-to-Clipboard │
|
||
│ (8个平台) │ │ (12个平台) │
|
||
│ │ │ │
|
||
│ - GitHub │ │ - 头条号 │
|
||
│ - 微信公众号 │ │ - 小红书 │
|
||
│ - B站 │ │ - 抖音 │
|
||
│ - 知乎 │ │ - 简书 │
|
||
│ - CSDN │ │ - QQ空间 │
|
||
│ - 百家号 │ │ - 新浪博客/新闻 │
|
||
│ - 企鹅号 │ │ - 搜狐号 │
|
||
│ - 网易号 │ │ - 一点号 │
|
||
│ │ │ - 东方财富 │
|
||
│ │ │ - 邦阅网 │
|
||
│ │ │ - 原创力文档 │
|
||
└───────────────────┘ └──────────────────────┘
|
||
│ │
|
||
└───────────────────┬───────────────────┘
|
||
│
|
||
┌─────────────────────────────────────────────────────────┐
|
||
│ Data Storage Layer (SQLite) │
|
||
│ (平台账号、发布记录、文章状态) │
|
||
└─────────────────────────────────────────────────────────┘
|
||
```
|
||
|
||
### 技术栈
|
||
|
||
- **后端框架**:Streamlit(已有)
|
||
- **数据库**:SQLite(已有)
|
||
- **HTTP客户端**:`httpx` 或 `requests`
|
||
- **OAuth2.0**:`requests-oauthlib`
|
||
- **任务队列**:`asyncio`(异步发布)
|
||
- **内容转换**:`markdown`、`html2text`、`Pillow`(图片处理)
|
||
|
||
---
|
||
|
||
## 💾 数据库设计
|
||
|
||
### 新增表结构
|
||
|
||
#### 1. platform_accounts(平台账号表)
|
||
|
||
```sql
|
||
CREATE TABLE IF NOT EXISTS platform_accounts (
|
||
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||
platform TEXT NOT NULL, -- 平台名称
|
||
account_type TEXT NOT NULL, -- 'api' 或 'manual'
|
||
account_name TEXT, -- 账号名称/标识
|
||
access_token TEXT, -- OAuth token(加密存储)
|
||
refresh_token TEXT, -- 刷新token(加密存储)
|
||
token_expires_at TIMESTAMP, -- token过期时间
|
||
api_key TEXT, -- API Key(如GitHub)
|
||
api_secret TEXT, -- API Secret
|
||
config_json TEXT, -- 平台特定配置(JSON)
|
||
is_active INTEGER DEFAULT 1, -- 是否激活
|
||
brand TEXT, -- 关联品牌
|
||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||
UNIQUE(platform, brand, account_name)
|
||
);
|
||
```
|
||
|
||
#### 2. publish_records(发布记录表)
|
||
|
||
```sql
|
||
CREATE TABLE IF NOT EXISTS publish_records (
|
||
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||
article_id INTEGER, -- 关联articles表
|
||
platform TEXT NOT NULL,
|
||
publish_method TEXT NOT NULL, -- 'api' 或 'copy'
|
||
publish_status TEXT NOT NULL, -- 'pending', 'success', 'failed', 'copied'
|
||
publish_url TEXT, -- 发布后的URL(API发布)
|
||
publish_id TEXT, -- 平台返回的发布ID
|
||
error_message TEXT, -- 错误信息
|
||
retry_count INTEGER DEFAULT 0, -- 重试次数
|
||
published_at TIMESTAMP, -- 发布时间
|
||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||
FOREIGN KEY (article_id) REFERENCES articles(id)
|
||
);
|
||
```
|
||
|
||
#### 3. platform_configs(平台配置表)
|
||
|
||
```sql
|
||
CREATE TABLE IF NOT EXISTS platform_configs (
|
||
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||
platform TEXT NOT NULL UNIQUE,
|
||
has_api INTEGER DEFAULT 0, -- 是否有API
|
||
api_docs_url TEXT, -- API文档链接
|
||
content_format TEXT, -- 内容格式要求
|
||
max_length INTEGER, -- 最大字数
|
||
min_length INTEGER, -- 最小字数
|
||
supports_images INTEGER DEFAULT 0, -- 是否支持图片
|
||
supports_tags INTEGER DEFAULT 0, -- 是否支持标签
|
||
publish_guide TEXT, -- 发布指南(一键复制平台)
|
||
format_template TEXT, -- 格式模板(一键复制平台)
|
||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||
);
|
||
```
|
||
|
||
#### 4. publish_queue(发布队列表)
|
||
|
||
```sql
|
||
CREATE TABLE IF NOT EXISTS publish_queue (
|
||
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||
article_id INTEGER NOT NULL,
|
||
platform TEXT NOT NULL,
|
||
priority INTEGER DEFAULT 0, -- 优先级
|
||
scheduled_at TIMESTAMP, -- 计划发布时间
|
||
status TEXT DEFAULT 'pending', -- 'pending', 'processing', 'completed', 'failed'
|
||
retry_count INTEGER DEFAULT 0,
|
||
max_retries INTEGER DEFAULT 3,
|
||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||
FOREIGN KEY (article_id) REFERENCES articles(id)
|
||
);
|
||
```
|
||
|
||
### 扩展articles表
|
||
|
||
```sql
|
||
ALTER TABLE articles ADD COLUMN publish_status TEXT DEFAULT 'draft';
|
||
ALTER TABLE articles ADD COLUMN publish_urls TEXT; -- JSON格式存储各平台发布URL
|
||
```
|
||
|
||
---
|
||
|
||
## 📦 模块划分
|
||
|
||
### 目录结构
|
||
|
||
```
|
||
geo_tool/
|
||
├── platform_sync/ # 新增:平台同步模块
|
||
│ ├── __init__.py
|
||
│ ├── base_publisher.py # 发布器基类
|
||
│ ├── api_publishers/ # API发布器
|
||
│ │ ├── __init__.py
|
||
│ │ ├── github_publisher.py
|
||
│ │ ├── wechat_publisher.py
|
||
│ │ ├── bilibili_publisher.py
|
||
│ │ ├── zhihu_publisher.py
|
||
│ │ ├── csdn_publisher.py
|
||
│ │ ├── baijiahao_publisher.py
|
||
│ │ ├── qq_publisher.py
|
||
│ │ └── netease_publisher.py
|
||
│ ├── copy_publishers/ # 一键复制发布器
|
||
│ │ ├── __init__.py
|
||
│ │ ├── copy_manager.py
|
||
│ │ └── format_templates.py
|
||
│ ├── content_converter.py # 内容格式转换
|
||
│ ├── sync_manager.py # 同步管理器
|
||
│ └── account_manager.py # 账号管理
|
||
├── platform_templates/ # 新增:平台模板
|
||
│ ├── __init__.py
|
||
│ ├── new_platforms/ # 新增8个平台的Prompt模板
|
||
│ │ ├── sina_blog.py
|
||
│ │ ├── sina_news.py
|
||
│ │ ├── sohu.py
|
||
│ │ ├── qzone.py
|
||
│ │ ├── bangyue.py
|
||
│ │ ├── yidian.py
|
||
│ │ ├── eastmoney.py
|
||
│ │ └── yuanchuangli.py
|
||
│ └── existing_platforms/ # 原有平台模板(已有)
|
||
├── modules/data_storage.py # 扩展:添加发布相关方法
|
||
└── geo_tool.py # 扩展:添加发布UI
|
||
```
|
||
|
||
---
|
||
|
||
## 💻 核心代码实现
|
||
|
||
### 1. 发布器基类 (modules/base_publisher.py)
|
||
|
||
```python
|
||
"""
|
||
平台发布器基类
|
||
"""
|
||
from abc import ABC, abstractmethod
|
||
from typing import Dict, Optional, Any
|
||
from datetime import datetime
|
||
|
||
|
||
class BasePublisher(ABC):
|
||
"""发布器基类"""
|
||
|
||
def __init__(self, platform: str, account_config: Dict[str, Any]):
|
||
self.platform = platform
|
||
self.account_config = account_config
|
||
self.access_token = account_config.get('access_token')
|
||
self.refresh_token = account_config.get('refresh_token')
|
||
|
||
@abstractmethod
|
||
def publish(self, content: str, title: str, **kwargs) -> Dict[str, Any]:
|
||
"""
|
||
发布内容
|
||
|
||
Args:
|
||
content: 文章内容
|
||
title: 文章标题
|
||
**kwargs: 其他参数(标签、图片等)
|
||
|
||
Returns:
|
||
{
|
||
'success': bool,
|
||
'publish_url': str,
|
||
'publish_id': str,
|
||
'error': str
|
||
}
|
||
"""
|
||
pass
|
||
|
||
@abstractmethod
|
||
def upload_image(self, image_path: str) -> Optional[str]:
|
||
"""上传图片,返回图片URL"""
|
||
pass
|
||
|
||
def refresh_token_if_needed(self) -> bool:
|
||
"""刷新token(如果需要)"""
|
||
# 子类实现
|
||
return True
|
||
|
||
def validate_account(self) -> bool:
|
||
"""验证账号是否有效"""
|
||
# 子类实现
|
||
return True
|
||
```
|
||
|
||
### 2. GitHub发布器示例 (api_publishers/github_publisher.py)
|
||
|
||
```python
|
||
"""
|
||
GitHub发布器
|
||
"""
|
||
import base64
|
||
import requests
|
||
from typing import Dict, Any, Optional
|
||
from .base_publisher import BasePublisher
|
||
|
||
|
||
class GitHubPublisher(BasePublisher):
|
||
"""GitHub发布器"""
|
||
|
||
def __init__(self, account_config: Dict[str, Any]):
|
||
super().__init__("GitHub", account_config)
|
||
self.api_key = account_config.get('api_key')
|
||
self.repo_owner = account_config.get('repo_owner')
|
||
self.repo_name = account_config.get('repo_name')
|
||
self.base_url = "https://api.github.com"
|
||
self.headers = {
|
||
"Authorization": f"token {self.api_key}",
|
||
"Accept": "application/vnd.github.v3+json"
|
||
}
|
||
|
||
def publish(self, content: str, title: str, **kwargs) -> Dict[str, Any]:
|
||
"""发布到GitHub"""
|
||
try:
|
||
# 生成文件路径
|
||
file_path = kwargs.get('file_path', f"content/{title.replace(' ', '_')}.md")
|
||
|
||
# 编码内容
|
||
content_bytes = content.encode('utf-8')
|
||
content_base64 = base64.b64encode(content_bytes).decode('utf-8')
|
||
|
||
# 创建或更新文件
|
||
url = f"{self.base_url}/repos/{self.repo_owner}/{self.repo_name}/contents/{file_path}"
|
||
|
||
# 检查文件是否存在
|
||
response = requests.get(url, headers=self.headers)
|
||
sha = None
|
||
if response.status_code == 200:
|
||
sha = response.json().get('sha')
|
||
|
||
data = {
|
||
"message": f"Publish: {title}",
|
||
"content": content_base64,
|
||
"branch": kwargs.get('branch', 'main')
|
||
}
|
||
if sha:
|
||
data["sha"] = sha
|
||
|
||
response = requests.put(url, json=data, headers=self.headers)
|
||
|
||
if response.status_code in [200, 201]:
|
||
result = response.json()
|
||
return {
|
||
'success': True,
|
||
'publish_url': result.get('content', {}).get('html_url', ''),
|
||
'publish_id': result.get('content', {}).get('sha', ''),
|
||
'error': None
|
||
}
|
||
else:
|
||
return {
|
||
'success': False,
|
||
'publish_url': '',
|
||
'publish_id': '',
|
||
'error': f"GitHub API错误: {response.text}"
|
||
}
|
||
except Exception as e:
|
||
return {
|
||
'success': False,
|
||
'publish_url': '',
|
||
'publish_id': '',
|
||
'error': str(e)
|
||
}
|
||
|
||
def upload_image(self, image_path: str) -> Optional[str]:
|
||
"""GitHub不支持直接上传图片,需要先上传到仓库"""
|
||
# 实现图片上传逻辑
|
||
return None
|
||
|
||
def validate_account(self) -> bool:
|
||
"""验证GitHub账号"""
|
||
try:
|
||
response = requests.get(f"{self.base_url}/user", headers=self.headers)
|
||
return response.status_code == 200
|
||
except:
|
||
return False
|
||
```
|
||
|
||
### 3. 一键复制管理器 (copy_publishers/copy_manager.py)
|
||
|
||
```python
|
||
"""
|
||
一键复制管理器
|
||
"""
|
||
import pyperclip
|
||
from typing import Dict, Any
|
||
from .format_templates import FormatTemplates
|
||
|
||
|
||
class CopyManager:
|
||
"""一键复制管理器"""
|
||
|
||
def __init__(self):
|
||
self.templates = FormatTemplates()
|
||
|
||
def format_for_platform(self, platform: str, content: str, title: str, **kwargs) -> str:
|
||
"""
|
||
为平台格式化内容
|
||
|
||
Args:
|
||
platform: 平台名称
|
||
content: 原始内容
|
||
title: 标题
|
||
**kwargs: 其他参数(标签、摘要等)
|
||
|
||
Returns:
|
||
格式化后的内容
|
||
"""
|
||
template = self.templates.get_template(platform)
|
||
if not template:
|
||
# 默认格式
|
||
return f"{title}\n\n{content}"
|
||
|
||
return template.format(
|
||
title=title,
|
||
content=content,
|
||
**kwargs
|
||
)
|
||
|
||
def copy_to_clipboard(self, text: str) -> bool:
|
||
"""复制到剪贴板"""
|
||
try:
|
||
pyperclip.copy(text)
|
||
return True
|
||
except Exception as e:
|
||
print(f"复制失败: {e}")
|
||
return False
|
||
|
||
def generate_publish_guide(self, platform: str) -> str:
|
||
"""生成发布指南"""
|
||
guides = {
|
||
"头条号": """
|
||
发布步骤:
|
||
1. 登录头条号后台
|
||
2. 点击"发布" -> "文章"
|
||
3. 粘贴标题和内容
|
||
4. 添加封面图和标签
|
||
5. 点击发布
|
||
""",
|
||
"小红书": """
|
||
发布步骤:
|
||
1. 打开小红书APP
|
||
2. 点击"+"号发布
|
||
3. 选择"图文"
|
||
4. 粘贴标题和内容
|
||
5. 添加图片和标签
|
||
6. 发布
|
||
""",
|
||
# ... 其他平台指南
|
||
}
|
||
return guides.get(platform, "请参考平台官方发布指南")
|
||
```
|
||
|
||
### 4. 同步管理器 (modules/sync_manager.py)
|
||
|
||
```python
|
||
"""
|
||
平台同步管理器
|
||
"""
|
||
import asyncio
|
||
from typing import List, Dict, Any, Optional
|
||
from datetime import datetime
|
||
from data_storage import DataStorage
|
||
from platform_sync.api_publishers import get_api_publisher
|
||
from platform_sync.copy_publishers import CopyManager
|
||
|
||
|
||
class SyncManager:
|
||
"""平台同步管理器"""
|
||
|
||
def __init__(self, storage: DataStorage):
|
||
self.storage = storage
|
||
self.copy_manager = CopyManager()
|
||
|
||
async def publish_article(
|
||
self,
|
||
article_id: int,
|
||
platform: str,
|
||
account_config: Optional[Dict[str, Any]] = None
|
||
) -> Dict[str, Any]:
|
||
"""
|
||
发布文章到指定平台
|
||
|
||
Args:
|
||
article_id: 文章ID
|
||
platform: 平台名称
|
||
account_config: 账号配置
|
||
|
||
Returns:
|
||
发布结果
|
||
"""
|
||
# 获取文章
|
||
article = self.storage.get_article_by_id(article_id)
|
||
if not article:
|
||
return {'success': False, 'error': '文章不存在'}
|
||
|
||
# 获取平台配置
|
||
platform_config = self.storage.get_platform_config(platform)
|
||
if not platform_config:
|
||
return {'success': False, 'error': '平台配置不存在'}
|
||
|
||
# 判断发布方式
|
||
if platform_config.get('has_api'):
|
||
# API发布
|
||
if not account_config:
|
||
account_config = self.storage.get_platform_account(platform)
|
||
|
||
if not account_config:
|
||
return {'success': False, 'error': '账号未配置'}
|
||
|
||
publisher = get_api_publisher(platform, account_config)
|
||
result = await self._publish_via_api(publisher, article, platform_config)
|
||
else:
|
||
# 一键复制
|
||
result = await self._publish_via_copy(article, platform, platform_config)
|
||
|
||
# 保存发布记录
|
||
self.storage.save_publish_record(
|
||
article_id=article_id,
|
||
platform=platform,
|
||
publish_method='api' if platform_config.get('has_api') else 'copy',
|
||
publish_status='success' if result['success'] else 'failed',
|
||
publish_url=result.get('publish_url', ''),
|
||
publish_id=result.get('publish_id', ''),
|
||
error_message=result.get('error')
|
||
)
|
||
|
||
return result
|
||
|
||
async def _publish_via_api(
|
||
self,
|
||
publisher,
|
||
article: Dict[str, Any],
|
||
platform_config: Dict[str, Any]
|
||
) -> Dict[str, Any]:
|
||
"""通过API发布"""
|
||
try:
|
||
# 内容格式转换
|
||
content = self._convert_content(article['content'], platform_config)
|
||
|
||
# 发布
|
||
result = publisher.publish(
|
||
content=content,
|
||
title=article.get('title', article['keyword']),
|
||
keyword=article['keyword'],
|
||
brand=article.get('brand', '')
|
||
)
|
||
|
||
return result
|
||
except Exception as e:
|
||
return {'success': False, 'error': str(e)}
|
||
|
||
async def _publish_via_copy(
|
||
self,
|
||
article: Dict[str, Any],
|
||
platform: str,
|
||
platform_config: Dict[str, Any]
|
||
) -> Dict[str, Any]:
|
||
"""通过一键复制发布"""
|
||
try:
|
||
# 格式化内容
|
||
formatted_content = self.copy_manager.format_for_platform(
|
||
platform=platform,
|
||
content=article['content'],
|
||
title=article.get('title', article['keyword']),
|
||
keyword=article['keyword'],
|
||
brand=article.get('brand', '')
|
||
)
|
||
|
||
# 复制到剪贴板
|
||
success = self.copy_manager.copy_to_clipboard(formatted_content)
|
||
|
||
if success:
|
||
return {
|
||
'success': True,
|
||
'publish_url': '',
|
||
'publish_id': '',
|
||
'copied_content': formatted_content,
|
||
'guide': self.copy_manager.generate_publish_guide(platform)
|
||
}
|
||
else:
|
||
return {'success': False, 'error': '复制到剪贴板失败'}
|
||
except Exception as e:
|
||
return {'success': False, 'error': str(e)}
|
||
|
||
def _convert_content(self, content: str, platform_config: Dict[str, Any]) -> str:
|
||
"""内容格式转换"""
|
||
# 根据平台要求转换格式
|
||
# Markdown -> HTML / 纯文本等
|
||
return content
|
||
|
||
async def batch_publish(
|
||
self,
|
||
article_ids: List[int],
|
||
platforms: List[str],
|
||
delay_seconds: int = 5
|
||
) -> List[Dict[str, Any]]:
|
||
"""批量发布"""
|
||
results = []
|
||
for article_id in article_ids:
|
||
for platform in platforms:
|
||
result = await self.publish_article(article_id, platform)
|
||
results.append({
|
||
'article_id': article_id,
|
||
'platform': platform,
|
||
'result': result
|
||
})
|
||
# 延迟,避免频率限制
|
||
await asyncio.sleep(delay_seconds)
|
||
return results
|
||
```
|
||
|
||
### 5. 扩展DataStorage (modules/data_storage.py扩展)
|
||
|
||
```python
|
||
# 在DataStorage类中添加以下方法
|
||
|
||
def save_platform_account(self, platform: str, account_config: Dict[str, Any], brand: str):
|
||
"""保存平台账号"""
|
||
# 实现保存逻辑
|
||
|
||
def get_platform_account(self, platform: str, brand: str) -> Optional[Dict[str, Any]]:
|
||
"""获取平台账号"""
|
||
# 实现获取逻辑
|
||
|
||
def save_publish_record(self, article_id: int, platform: str, publish_method: str,
|
||
publish_status: str, publish_url: str = '', publish_id: str = '',
|
||
error_message: str = ''):
|
||
"""保存发布记录"""
|
||
# 实现保存逻辑
|
||
|
||
def get_publish_records(self, article_id: Optional[int] = None,
|
||
platform: Optional[str] = None) -> List[Dict]:
|
||
"""获取发布记录"""
|
||
# 实现获取逻辑
|
||
|
||
def save_platform_config(self, platform: str, config: Dict[str, Any]):
|
||
"""保存平台配置"""
|
||
# 实现保存逻辑
|
||
|
||
def get_platform_config(self, platform: str) -> Optional[Dict[str, Any]]:
|
||
"""获取平台配置"""
|
||
# 实现获取逻辑
|
||
```
|
||
|
||
### 6. UI集成 (geo_tool.py扩展)
|
||
|
||
```python
|
||
# 在geo_tool.py中添加新的Tab
|
||
|
||
def show_platform_sync_tab():
|
||
"""平台同步Tab"""
|
||
st.header("📤 平台文章同步")
|
||
|
||
# 1. 平台账号配置
|
||
with st.expander("🔐 平台账号配置", expanded=False):
|
||
platform = st.selectbox("选择平台", ALL_PLATFORMS)
|
||
account_type = "API" if platform in API_PLATFORMS else "手动"
|
||
st.info(f"发布方式: {account_type}")
|
||
|
||
if account_type == "API":
|
||
# API账号配置
|
||
api_key = st.text_input("API Key", type="password")
|
||
api_secret = st.text_input("API Secret", type="password")
|
||
# ... 其他配置
|
||
|
||
if st.button("保存账号配置"):
|
||
# 保存配置
|
||
pass
|
||
else:
|
||
st.info("该平台使用一键复制功能,无需配置账号")
|
||
|
||
# 2. 发布管理
|
||
st.subheader("📝 发布管理")
|
||
|
||
# 选择文章
|
||
articles = storage.get_articles(brand=brand)
|
||
selected_articles = st.multiselect("选择要发布的文章", articles, format_func=lambda x: x['keyword'])
|
||
|
||
# 选择平台
|
||
selected_platforms = st.multiselect("选择发布平台", ALL_PLATFORMS)
|
||
|
||
# 发布选项
|
||
col1, col2 = st.columns(2)
|
||
with col1:
|
||
publish_mode = st.radio("发布模式", ["立即发布", "定时发布"])
|
||
with col2:
|
||
delay_seconds = st.number_input("发布间隔(秒)", min_value=0, value=5)
|
||
|
||
# 发布按钮
|
||
if st.button("🚀 开始发布", type="primary"):
|
||
sync_manager = SyncManager(storage)
|
||
|
||
with st.spinner("正在发布..."):
|
||
results = asyncio.run(sync_manager.batch_publish(
|
||
article_ids=[a['id'] for a in selected_articles],
|
||
platforms=selected_platforms,
|
||
delay_seconds=delay_seconds
|
||
))
|
||
|
||
# 显示结果
|
||
for result in results:
|
||
if result['result']['success']:
|
||
st.success(f"✅ {result['platform']}: 发布成功")
|
||
else:
|
||
st.error(f"❌ {result['platform']}: {result['result']['error']}")
|
||
|
||
# 3. 发布记录
|
||
st.subheader("📊 发布记录")
|
||
records = storage.get_publish_records()
|
||
if records:
|
||
df = pd.DataFrame(records)
|
||
st.dataframe(df)
|
||
else:
|
||
st.info("暂无发布记录")
|
||
```
|
||
|
||
---
|
||
|
||
## 🚀 实施步骤
|
||
|
||
### 阶段一:基础架构(第1-2周)
|
||
|
||
1. **数据库扩展**
|
||
```bash
|
||
# 运行数据库迁移脚本
|
||
python scripts/migrate_database.py
|
||
```
|
||
|
||
2. **创建模块结构**
|
||
```bash
|
||
mkdir -p platform_sync/api_publishers
|
||
mkdir -p platform_sync/copy_publishers
|
||
mkdir -p platform_templates/new_platforms
|
||
```
|
||
|
||
3. **实现基础类**
|
||
- `BasePublisher` 基类
|
||
- `SyncManager` 管理器
|
||
- `CopyManager` 复制管理器
|
||
|
||
### 阶段二:API发布器(第3-6周)
|
||
|
||
1. **GitHub发布器**(1-2天)
|
||
2. **微信公众号发布器**(3-4天)
|
||
3. **B站发布器**(3-4天)
|
||
4. **知乎发布器**(3-4天)
|
||
5. **CSDN发布器**(3-4天)
|
||
6. **百家号发布器**(4-5天)
|
||
7. **企鹅号发布器**(4-5天)
|
||
8. **网易号发布器**(4-5天)
|
||
|
||
### 阶段三:新增平台内容生成(第4-5周)
|
||
|
||
为8个新增平台创建Prompt模板:
|
||
|
||
```python
|
||
# platform_templates/new_platforms/sina_blog.py
|
||
SINA_BLOG_TEMPLATE = """
|
||
你是GEO专家 + 新浪博客作者。
|
||
【关键词】{keyword}
|
||
【品牌】{brand}
|
||
【优势】{advantages}
|
||
【要求】
|
||
1) 3个吸引人的标题
|
||
2) 开头:故事化或热点引入
|
||
3) 正文:深度分析、案例丰富、观点鲜明
|
||
4) 自然提及品牌2-4次
|
||
5) 适合新浪博客:内容深度、可读性强
|
||
6) 字数:1500-3000字
|
||
7) 结尾:总结+延伸思考
|
||
【格式】标题-正文-总结
|
||
【开始】
|
||
"""
|
||
```
|
||
|
||
### 阶段四:一键复制功能(第7-8周)
|
||
|
||
1. **格式模板开发**(12个平台)
|
||
2. **剪贴板集成**
|
||
3. **发布指南生成**
|
||
|
||
### 阶段五:UI集成(第9周)
|
||
|
||
1. **平台账号配置界面**
|
||
2. **发布管理界面**
|
||
3. **发布记录展示**
|
||
|
||
### 阶段六:测试和优化(第10周)
|
||
|
||
1. **单元测试**
|
||
2. **集成测试**
|
||
3. **性能优化**
|
||
|
||
---
|
||
|
||
## 🧪 测试方案
|
||
|
||
### 单元测试
|
||
|
||
```python
|
||
# tests/test_github_publisher.py
|
||
import pytest
|
||
from platform_sync.api_publishers.github_publisher import GitHubPublisher
|
||
|
||
def test_github_publish():
|
||
config = {
|
||
'api_key': 'test_key',
|
||
'repo_owner': 'test_owner',
|
||
'repo_name': 'test_repo'
|
||
}
|
||
publisher = GitHubPublisher(config)
|
||
result = publisher.publish("Test content", "Test Title")
|
||
assert result['success'] == True
|
||
```
|
||
|
||
### 集成测试
|
||
|
||
```python
|
||
# tests/test_sync_manager.py
|
||
async def test_batch_publish():
|
||
manager = SyncManager(storage)
|
||
results = await manager.batch_publish([1, 2], ['GitHub', '知乎'])
|
||
assert len(results) == 4
|
||
```
|
||
|
||
---
|
||
|
||
## 📝 依赖安装
|
||
|
||
```bash
|
||
pip install httpx requests-oauthlib pyperclip markdown html2text Pillow
|
||
```
|
||
|
||
更新 `requirements.txt`:
|
||
|
||
```
|
||
httpx>=0.24.0
|
||
requests-oauthlib>=1.3.1
|
||
pyperclip>=1.8.2
|
||
markdown>=3.4.0
|
||
html2text>=2020.1.16
|
||
Pillow>=10.0.0
|
||
```
|
||
|
||
---
|
||
|
||
## ⚠️ 注意事项
|
||
|
||
1. **Token安全**:所有token需要加密存储
|
||
2. **频率限制**:遵守各平台的API调用频率限制
|
||
3. **错误处理**:完善的错误处理和重试机制
|
||
4. **日志记录**:详细的发布日志
|
||
5. **用户体验**:清晰的发布状态反馈
|
||
|
||
---
|
||
|
||
**实施时间**:10周(2.5个月)
|
||
**开发人员**:1-2人
|
||
**优先级**:高
|