main
Jack 6 months ago
parent 9bed7bc1fb
commit ad98c97949
  1. 0
      wqb-filter-alpha/account.txt
  2. 245
      wqb-filter-alpha/filter-alpha.py
  3. 99
      wqb-filter/filter.py

@ -0,0 +1,245 @@
import httpx
from httpx import BasicAuth
import os
def login(credentials_file='account.txt'):
"""登录WorldQuant Brain API"""
# 读取本地账号密码
if not os.path.exists(credentials_file):
print("未找到 account.txt 文件")
with open(credentials_file, 'w') as f: f.write("")
print("account.txt 文件已创建,请填写账号密码, 格式: ['username', 'password]")
exit(1)
with open(credentials_file) as f:
credentials = eval(f.read())
username, password = credentials[0], credentials[1]
print(f"正在登录账户: {username}")
# 创建客户端并认证
client = httpx.Client(auth=BasicAuth(username, password))
# 发送登录请求
response = client.post('https://api.worldquantbrain.com/authentication')
print(f"登录状态: {response.status_code}")
if response.status_code == 201:
print("登录成功!")
return client
else:
print(f"登录失败: {response.json()}")
client.close()
return None
def get_alphas_data(client, page_size=10):
"""获取所有alpha数据(自动处理分页)"""
all_alphas = []
offset = 0
total_count = 0
while True:
print(f"正在获取第 {offset//page_size + 1} 页数据 (offset={offset})...")
# 构建请求参数
params = {
'limit': page_size,
'offset': offset,
'status': 'UNSUBMITTED%1FIS_FAIL',
'is.sharpe%3E1.25': '',
'order': '-dateSubmitted',
'hidden': 'false'
}
url = 'https://api.worldquantbrain.com/users/self/alphas'
try:
response = client.get(url, params=params)
print(f"获取数据状态码: {response.status_code}")
if response.status_code != 200:
print(f"获取数据失败: {response.text}")
break
data = response.json()
# 第一次请求时获取总数量
if offset == 0:
total_count = data.get('count', 0)
print(f"总数据量: {total_count}")
if total_count == 0:
print("没有符合条件的Alpha因子")
break
# 添加当前页的数据
if 'alphas' in data:
current_page_alphas = data['alphas']
all_alphas.extend(current_page_alphas)
print(f"当前页获取到 {len(current_page_alphas)} 条alpha数据")
# 检查是否还有更多数据
current_count = offset + len(data.get('alphas', []))
print(f"进度: {current_count}/{total_count}")
# 如果已经获取了所有数据,则退出循环
if current_count >= total_count:
print("所有数据已获取完毕")
break
# 更新offset继续获取下一页
offset += page_size
except Exception as e:
print(f"请求发生错误: {e}")
break
return {
'count': len(all_alphas),
'alphas': all_alphas
}
def extract_alpha_info(alpha_data):
"""
从alpha数据中提取有用信息
"""
extracted_data = []
# 检查数据结构
if not alpha_data or 'results' not in alpha_data:
print("无效的数据格式")
return extracted_data
for alpha in alpha_data['results']:
# 提取基本信息
info = {
# 基础信息
'id': alpha.get('id', 'N/A'),
'name': alpha.get('name', 'N/A'),
'author': alpha.get('author', 'N/A'),
'status': alpha.get('status', 'N/A'),
'grade': alpha.get('grade', 'N/A'),
'stage': alpha.get('stage', 'N/A'),
'date_created': alpha.get('dateCreated', 'N/A'),
'date_submitted': alpha.get('dateSubmitted', 'N/A'),
# 因子代码和设置
'code': alpha.get('regular', {}).get('code', 'N/A'),
'operator_count': alpha.get('regular', {}).get('operatorCount', 0),
# 回测设置
'settings': {
'region': alpha.get('settings', {}).get('region', 'N/A'),
'universe': alpha.get('settings', {}).get('universe', 'N/A'),
'neutralization': alpha.get('settings', {}).get('neutralization', 'N/A'),
'truncation': alpha.get('settings', {}).get('truncation', 'N/A'),
'start_date': alpha.get('settings', {}).get('startDate', 'N/A'),
'end_date': alpha.get('settings', {}).get('endDate', 'N/A')
},
# 回测表现指标
'performance': {
'sharpe': alpha.get('is', {}).get('sharpe', 0),
'fitness': alpha.get('is', {}).get('fitness', 0),
'returns': alpha.get('is', {}).get('returns', 0),
'turnover': alpha.get('is', {}).get('turnover', 0),
'drawdown': alpha.get('is', {}).get('drawdown', 0),
'pnl': alpha.get('is', {}).get('pnl', 0),
'book_size': alpha.get('is', {}).get('bookSize', 0),
'long_count': alpha.get('is', {}).get('longCount', 0),
'short_count': alpha.get('is', {}).get('shortCount', 0),
'margin': alpha.get('is', {}).get('margin', 0)
},
# 检查结果
'checks': {}
}
# 提取检查结果
checks = alpha.get('is', {}).get('checks', [])
for check in checks:
check_name = check.get('name', 'N/A')
info['checks'][check_name] = {
'result': check.get('result', 'N/A'),
'limit': check.get('limit', 'N/A'),
'value': check.get('value', 'N/A')
}
# 分类信息
classifications = alpha.get('classifications', [])
if classifications:
info['classifications'] = [cls.get('name', 'N/A') for cls in classifications]
else:
info['classifications'] = []
extracted_data.append(info)
return extracted_data
def print_alpha_summary(extracted_data):
"""
打印提取的alpha信息摘要
"""
print(f"\n总共提取了 {len(extracted_data)} 个Alpha因子")
print("=" * 100)
for i, alpha in enumerate(extracted_data, 1):
print(f"\n{i}. Alpha ID: {alpha['id']}")
print(f" 代码: {alpha['code'][:80]}{'...' if len(alpha['code']) > 80 else ''}")
print(f" 状态: {alpha['status']} | 等级: {alpha['grade']} | 阶段: {alpha['stage']}")
print(f" 创建时间: {alpha['date_created']}")
print(f" 设置: {alpha['settings']['region']} | {alpha['settings']['universe']} | "
f"中性化: {alpha['settings']['neutralization']}")
print(f" 表现: Sharpe={alpha['performance']['sharpe']:.2f} | "
f"Fitness={alpha['performance']['fitness']:.2f} | "
f"收益={alpha['performance']['returns']:.2%} | "
f"换手率={alpha['performance']['turnover']:.2f}")
# 显示关键检查结果
key_checks = ['LOW_SHARPE', 'LOW_FITNESS', 'LOW_TURNOVER', 'HIGH_TURNOVER']
check_results = []
for check_name in key_checks:
if check_name in alpha['checks']:
check = alpha['checks'][check_name]
result_symbol = "" if check['result'] == 'PASS' else ""
check_results.append(f"{check_name}: {result_symbol}")
if check_results:
print(f" 检查: {', '.join(check_results)}")
# 使用示例
def process_alpha_data(raw_data):
"""
处理原始alpha数据的完整流程
"""
# 提取信息
extracted_data = extract_alpha_info(raw_data)
# 打印摘要
print_alpha_summary(extracted_data)
return extracted_data
def main():
# 登录
client = login()
if not client:
return
try:
# 一次性获取所有alpha数据
all_data = get_alphas_data(client, page_size=10)
print(f"\n最终结果: 总共获取到 {all_data['count']} 条alpha数据")
if all_data and 'alphas' in all_data:
# 处理提取信息
processed_data = process_alpha_data({'results': all_data['alphas']})
finally:
# 确保客户端被关闭
client.close()
print("\n连接已关闭")
# 使用示例
if __name__ == "__main__":
main()

@ -1,99 +0,0 @@
import httpx
from httpx import BasicAuth
import os
def login(credentials_file='account.txt'):
"""登录WorldQuant Brain API"""
# 读取本地账号密码
if not os.path.exists(credentials_file):
print("未找到 account.txt 文件")
with open(credentials_file, 'w') as f: f.write("")
print("account.txt 文件已创建,请填写账号密码, 格式: ['username', 'password]")
exit(1)
with open(credentials_file) as f:
credentials = eval(f.read())
username, password = credentials[0], credentials[1]
print(f"正在登录账户: {username}")
# 创建客户端并认证
client = httpx.Client(auth=BasicAuth(username, password))
# 发送登录请求
response = client.post('https://api.worldquantbrain.com/authentication')
print(f"登录状态: {response.status_code}")
if response.status_code == 201:
print("登录成功!")
print(response.json())
return client
else:
print(f"登录失败: {response.json()}")
client.close()
return None
def get_alphas_data(client, limit=10, offset=0):
"""获取alpha数据"""
# 使用params参数而不是直接拼接URL
params = {
'limit': limit,
'offset': offset,
'status': 'UNSUBMITTED%1FIS_FAIL', # 注意:这里可能需要根据实际API要求调整编码
'is.sharpe%3E1.25': '', # 注意:这里可能需要调整参数格式
'order': '-dateSubmitted',
'hidden': 'false'
}
url = 'https://api.worldquantbrain.com/users/self/alphas'
try:
response = client.get(url, params=params)
print(f"获取数据状态码: {response.status_code}")
if response.status_code == 200:
return response.json()
else:
print(f"获取数据失败: {response.text}")
return None
except Exception as e:
print(f"请求发生错误: {e}")
return None
def main():
# 登录
client = login()
if not client:
return
try:
# 第一次请求获取count
first_data = get_alphas_data(client, limit=10, offset=0)
if first_data and 'count' in first_data:
count = first_data['count']
print(f"获取到的count值: {count}")
if int(count) == 0:
print("没有符合条件的Alpha因子")
return
# 第二次请求,使用count作为offset
second_data = get_alphas_data(client, limit=count, offset=count)
if second_data:
print("第二次请求返回的数据:")
print(second_data)
else:
print("第二次请求失败")
else:
print("第一次请求失败或未找到count字段")
finally:
# 确保客户端被关闭
client.close()
print("\n连接已关闭")
# 使用示例
if __name__ == "__main__":
main()
Loading…
Cancel
Save