fa45d8aa5f
- health_checklist.json: 192.168.1.122→node122
- ocr_client.py: docstring IP→node122
- docs/market-data-requirements.md: IP→node122
- 所有API调用通过ProxyHandler({})绕过系统代理
Privoxy对node122:18003返回500,直连正常
253 lines
8.0 KiB
Python
253 lines
8.0 KiB
Python
#!/usr/bin/env python
|
|
# -*- coding:utf-8 -*-
|
|
"""
|
|
Date: 2025/3/9 22:00
|
|
Desc: 东方财富网-行情中心-期货市场-国际期货
|
|
https://quote.eastmoney.com/center/gridlist.html#futures_global
|
|
"""
|
|
|
|
import math
|
|
from typing import Optional
|
|
|
|
import pandas as pd
|
|
import requests
|
|
|
|
from akshare.utils.tqdm import get_tqdm
|
|
|
|
|
|
def __futures_global_hist_market_code(symbol: str = "HG00Y") -> Optional[int]:
|
|
"""
|
|
东方财富网-行情中心-期货市场-国际期货-品种市场对照表
|
|
https://quote.eastmoney.com/center/gridlist.html#futures_global
|
|
:param symbol: HG00Y, 品种代码;可以通过 ak.futures_global_spot_em() 来获取所有可获取历史行情数据的品种代码
|
|
:type symbol: str
|
|
:return: 品种所属于的市场
|
|
:rtype: str
|
|
"""
|
|
# 提取品种代码(去掉年份和月份部分)
|
|
base_symbol = ""
|
|
i = 0
|
|
while i < len(symbol) and not symbol[i].isdigit():
|
|
base_symbol += symbol[i]
|
|
i += 1
|
|
# 如果代码中没有数字(异常情况),则返回整个代码作为基础品种代码
|
|
if not base_symbol and i == len(symbol):
|
|
base_symbol = symbol
|
|
# 金属和贵金属品种 - 101
|
|
if base_symbol in ["HG", "GC", "SI", "QI", "QO", "MGC", "LTH"]:
|
|
return 101
|
|
# 能源品种 - 102
|
|
if base_symbol in ["CL", "NG", "RB", "HO", "PA", "PL", "QM"]:
|
|
return 102
|
|
# 农产品和金融品种 - 103
|
|
if base_symbol in [
|
|
"ZW",
|
|
"ZM",
|
|
"ZS",
|
|
"ZC",
|
|
"XC",
|
|
"XK",
|
|
"XW",
|
|
"YM",
|
|
"TY",
|
|
"US",
|
|
"EH",
|
|
"ZL",
|
|
"ZR",
|
|
"ZO",
|
|
"FV",
|
|
"TU",
|
|
"UL",
|
|
"NQ",
|
|
"ES",
|
|
]:
|
|
return 103
|
|
# 中国市场特有品种 - 104
|
|
if base_symbol in ["TF", "RT", "CN"]:
|
|
return 104
|
|
# 软商品期货 - 108
|
|
if base_symbol in ["SB", "CT", "SF"]:
|
|
return 108
|
|
# 特殊L开头品种 - 109
|
|
if base_symbol in ["LCPT", "LZNT", "LALT", "LTNT", "LLDT", "LNKT"]:
|
|
return 109
|
|
# MPM开头品种 - 110
|
|
if base_symbol == "MPM":
|
|
return 110
|
|
# 日本市场品种 - 111
|
|
if base_symbol.startswith("J"):
|
|
return 111
|
|
# 单字母代码品种 - 112
|
|
if base_symbol in ["M", "B", "G"]:
|
|
return 112
|
|
# 如果没有匹配到任何规则,返回一个默认值或者错误
|
|
return None
|
|
|
|
|
|
def futures_global_spot_em() -> pd.DataFrame:
|
|
"""
|
|
东方财富网-行情中心-期货市场-国际期货
|
|
https://quote.eastmoney.com/center/gridlist.html#futures_global
|
|
:return: 行情数据
|
|
:rtype: pandas.DataFrame
|
|
"""
|
|
url = "https://futsseapi.eastmoney.com/list/COMEX,NYMEX,COBOT,SGX,NYBOT,LME,MDEX,TOCOM,IPE"
|
|
params = {
|
|
"orderBy": "dm",
|
|
"sort": "desc",
|
|
"pageSize": "20",
|
|
"pageIndex": "0",
|
|
"token": "58b2fa8f54638b60b87d69b31969089c",
|
|
"field": "dm,sc,name,p,zsjd,zde,zdf,f152,o,h,l,zjsj,vol,wp,np,ccl",
|
|
"blockName": "callback",
|
|
}
|
|
r = requests.get(url, params=params)
|
|
data_json = r.json()
|
|
total_num = data_json["total"]
|
|
total_page = math.ceil(total_num / 20) - 1
|
|
tqdm = get_tqdm()
|
|
big_df = pd.DataFrame()
|
|
for page in tqdm(range(total_page), leave=False):
|
|
params.update({"pageIndex": page})
|
|
r = requests.get(url, params=params)
|
|
data_json = r.json()
|
|
temp_df = pd.DataFrame(data_json["list"])
|
|
big_df = pd.concat(objs=[big_df, temp_df], ignore_index=True)
|
|
big_df.reset_index(inplace=True)
|
|
big_df["index"] = big_df["index"] + 1
|
|
big_df.rename(
|
|
columns={
|
|
"index": "序号",
|
|
"np": "卖盘",
|
|
"h": "最高",
|
|
"dm": "代码",
|
|
"zsjd": "-",
|
|
"l": "最低",
|
|
"ccl": "持仓量",
|
|
"o": "今开",
|
|
"p": "最新价",
|
|
"sc": "-",
|
|
"vol": "成交量",
|
|
"name": "名称",
|
|
"wp": "买盘",
|
|
"zde": "涨跌额",
|
|
"zdf": "涨跌幅",
|
|
"zjsj": "昨结",
|
|
},
|
|
inplace=True,
|
|
)
|
|
big_df = big_df[
|
|
[
|
|
"序号",
|
|
"代码",
|
|
"名称",
|
|
"最新价",
|
|
"涨跌额",
|
|
"涨跌幅",
|
|
"今开",
|
|
"最高",
|
|
"最低",
|
|
"昨结",
|
|
"成交量",
|
|
"买盘",
|
|
"卖盘",
|
|
"持仓量",
|
|
]
|
|
]
|
|
big_df["最新价"] = pd.to_numeric(big_df["最新价"], errors="coerce")
|
|
big_df["涨跌额"] = pd.to_numeric(big_df["涨跌额"], errors="coerce")
|
|
big_df["涨跌幅"] = pd.to_numeric(big_df["涨跌幅"], errors="coerce")
|
|
big_df["今开"] = pd.to_numeric(big_df["今开"], errors="coerce")
|
|
big_df["最高"] = pd.to_numeric(big_df["最高"], errors="coerce")
|
|
big_df["最低"] = pd.to_numeric(big_df["最低"], errors="coerce")
|
|
big_df["昨结"] = pd.to_numeric(big_df["昨结"], errors="coerce")
|
|
big_df["成交量"] = pd.to_numeric(big_df["成交量"], errors="coerce")
|
|
big_df["买盘"] = pd.to_numeric(big_df["买盘"], errors="coerce")
|
|
big_df["卖盘"] = pd.to_numeric(big_df["卖盘"], errors="coerce")
|
|
big_df["持仓量"] = pd.to_numeric(big_df["持仓量"], errors="coerce")
|
|
return big_df
|
|
|
|
|
|
def futures_global_hist_em(symbol: str = "HG00Y") -> pd.DataFrame:
|
|
"""
|
|
东方财富网-行情中心-期货市场-国际期货-历史行情数据
|
|
https://quote.eastmoney.com/globalfuture/HG25J.html
|
|
:param symbol: 品种代码;可以通过 ak.futures_global_spot_em() 来获取所有可获取历史行情数据的品种代码
|
|
:type symbol: str
|
|
:return: 历史行情数据
|
|
:rtype: pandas.DataFrame
|
|
"""
|
|
url = "https://push2his.eastmoney.com/api/qt/stock/kline/get"
|
|
market_code = __futures_global_hist_market_code(symbol)
|
|
params = {
|
|
"secid": f"{market_code}.{symbol}",
|
|
"klt": "101",
|
|
"fqt": "1",
|
|
"lmt": "6600",
|
|
"end": "20500000",
|
|
"iscca": "1",
|
|
"fields1": "f1,f2,f3,f4,f5,f6,f7,f8",
|
|
"fields2": "f51,f52,f53,f54,f55,f56,f57,f58,f59,f60,f61,f62,f63,f64",
|
|
"ut": "f057cbcbce2a86e2866ab8877db1d059",
|
|
"forcect": "1",
|
|
}
|
|
r = requests.get(url, params=params)
|
|
data_json = r.json()
|
|
temp_df = pd.DataFrame([item.split(",") for item in data_json["data"]["klines"]])
|
|
temp_df["code"] = data_json["data"]["code"]
|
|
temp_df["name"] = data_json["data"]["name"]
|
|
temp_df.columns = [
|
|
"日期",
|
|
"开盘",
|
|
"最新价",
|
|
"最高",
|
|
"最低",
|
|
"总量",
|
|
"-",
|
|
"-",
|
|
"涨幅",
|
|
"-",
|
|
"-",
|
|
"-",
|
|
"持仓",
|
|
"日增",
|
|
"代码",
|
|
"名称",
|
|
]
|
|
temp_df = temp_df[
|
|
[
|
|
"日期",
|
|
"代码",
|
|
"名称",
|
|
"开盘",
|
|
"最新价",
|
|
"最高",
|
|
"最低",
|
|
"总量",
|
|
"涨幅",
|
|
"持仓",
|
|
"日增",
|
|
]
|
|
]
|
|
temp_df["日期"] = pd.to_datetime(temp_df["日期"], errors="coerce").dt.date
|
|
temp_df["开盘"] = pd.to_numeric(temp_df["开盘"], errors="coerce")
|
|
temp_df["最新价"] = pd.to_numeric(temp_df["最新价"], errors="coerce")
|
|
temp_df["最高"] = pd.to_numeric(temp_df["最高"], errors="coerce")
|
|
temp_df["最低"] = pd.to_numeric(temp_df["最低"], errors="coerce")
|
|
temp_df["总量"] = pd.to_numeric(temp_df["总量"], errors="coerce")
|
|
temp_df["涨幅"] = pd.to_numeric(temp_df["涨幅"], errors="coerce")
|
|
temp_df["日增"] = pd.to_numeric(temp_df["日增"], errors="coerce")
|
|
# 日增修复为有符号32位整数值
|
|
unsigned_max, signed_max = (2**32) - 1, (2**31) - 1
|
|
mask = temp_df["日增"] > signed_max
|
|
temp_df.loc[mask, "日增"] = temp_df.loc[mask, "日增"] - (unsigned_max + 1)
|
|
return temp_df
|
|
|
|
|
|
if __name__ == "__main__":
|
|
futures_global_spot_em_df = futures_global_spot_em()
|
|
print(futures_global_spot_em_df)
|
|
|
|
futures_global_hist_em_df = futures_global_hist_em(symbol="HG00Y")
|
|
print(futures_global_hist_em_df)
|