| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238 |
- import requests
- import decimal
- import time
- import threading
- import json
- from flask import Flask, render_template, jsonify
- from collections import deque
- import logging
- import plotly
- import plotly.graph_objects as go
- from plotly.utils import PlotlyJSONEncoder
- # --- 配置部分 ---
- IN_TOKEN_ADDRESS_SOLANA = 'So11111111111111111111111111111111111111112'
- OUT_TOKEN_ADDRESS_SOLANA = 'ED5nyyWEzpPPiWimP8vYm7sD7TD3LAt3Q3gRTWHzPJBY'
- AMOUNT_TO_QUERY_OPENOCEAN_IN_SOL = decimal.Decimal('10')
- GATEIO_SPOT_PAIR_RFC_USDT = 'MOODENG_USDT'
- BINANCE_SOL_USDT_PAIR = 'SOLUSDT'
- proxies = None
- decimal.getcontext().prec = 36
- # --- 价格获取函数 (与上一版相同,此处省略以保持简洁) ---
- def get_openocean_price_solana(in_token_addr, out_token_addr, human_amount_in_decimal_for_request):
- chain = 'solana';
- url = f'https://open-api.openocean.finance/v4/{chain}/quote'
- params = {'inTokenAddress': in_token_addr, 'outTokenAddress': out_token_addr,
- 'amount': str(human_amount_in_decimal_for_request), 'gasPrice': '0.000005'}
- try:
- r = requests.get(url, params=params, proxies=proxies, timeout=10);
- r.raise_for_status();
- data = r.json()
- if data.get('code') == 200 and data.get('data'):
- d = data['data'];
- req = ['inToken', 'outToken', 'inAmount', 'outAmount']
- if not all(k in d for k in req) or d['inToken'].get('decimals') is None or d['outToken'].get(
- 'decimals') is None: return {"error": "OO API缺少数据"}
- in_dec, out_dec = int(d['inToken']['decimals']), int(d['outToken']['decimals']);
- atomic_in, atomic_out = decimal.Decimal(d['inAmount']), decimal.Decimal(d['outAmount'])
- h_in_sol = atomic_in / (10 ** in_dec);
- h_out_rfc = atomic_out / (10 ** out_dec)
- if h_out_rfc == 0: return {"error": "OO输出RFC为0"}
- return {"price_sol_per_rfc": h_in_sol / h_out_rfc}
- else:
- return {
- "error": f"OO API错误 - Code:{data.get('code', 'N/A')}, Msg:{data.get('msg', data.get('message', 'N/A')) if isinstance(data, dict) else '格式错误'}"}
- except Exception as e:
- return {"error": f"OO(Solana)请求错误: {e}"}
- def get_gateio_spot_price_usdt(pair_symbol):
- url = f'https://api.gateio.ws/api/v4/spot/tickers';
- params = {'currency_pair': pair_symbol}
- try:
- r = requests.get(url, params=params, proxies=proxies, timeout=10);
- r.raise_for_status();
- data = r.json()
- if isinstance(data, list) and data:
- td = data[0]
- if td.get('currency_pair') == pair_symbol and td.get('lowest_ask'):
- return {"price_rfc_usdt": decimal.Decimal(td['lowest_ask'])}
- else:
- return {"error": f"Gate现货({pair_symbol})数据不匹配或无价格"}
- elif isinstance(data, dict) and data.get('label'):
- return {"error": f"Gate现货API错误: {data['label']}-{data.get('message', '')}"}
- else:
- return {"error": f"Gate现货API({pair_symbol})数据格式错误或未找到"}
- except Exception as e:
- return {"error": f"Gate现货({pair_symbol})请求错误: {e}"}
- def get_binance_spot_price_usdt(symbol):
- url = "https://api.binance.com/api/v3/ticker/price";
- params = {'symbol': symbol}
- try:
- response = requests.get(url, params=params, proxies=proxies, timeout=10);
- response.raise_for_status();
- data = response.json()
- if 'price' in data and data.get('symbol') == symbol:
- return {"price_sol_usdt": decimal.Decimal(data['price'])}
- else:
- msg = data.get('msg', '未知错误'); return {"error": f"Binance API错误 ({symbol}): {msg}"}
- except requests.exceptions.RequestException as e:
- return {"error": f"Binance API ({symbol}) 请求失败: {e}"}
- except Exception as e:
- return {"error": f"Binance API ({symbol}) 意外错误: {e}"}
- app = Flask(__name__)
- log = logging.getLogger('werkzeug');
- log.setLevel(logging.ERROR)
- MAX_HISTORY_POINTS_PLOTLY = 86400
- REFRESH_INTERVAL_SECONDS = 1
- historical_data_points = deque(maxlen=MAX_HISTORY_POINTS_PLOTLY)
- latest_values_for_table = {
- "oo_rfc_sol_price": "N/A", "gate_spot_rfc_sol_price": "N/A", "sol_usdt_price_binance": "N/A",
- "diff_oo_vs_spot_rfc_sol_percentage": "N/A",
- "oo_error": None, "gate_spot_error": None, "binance_sol_error": None,
- "last_updated": "N/A", "gate_spot_pair_name_usdt": GATEIO_SPOT_PAIR_RFC_USDT,
- "target_asset_symbol": GATEIO_SPOT_PAIR_RFC_USDT.split('_')[0]
- }
- data_lock = threading.Lock()
- def calculate_percentage_diff(price_a, price_b):
- if price_a is not None and price_b is not None and isinstance(price_a, decimal.Decimal) and isinstance(price_b,
- decimal.Decimal) and price_b != 0:
- return ((price_a - price_b) / price_b) * 100
- return None
- def convert_usdt_price_to_sol_price(price_asset_usdt, price_sol_usdt):
- if price_asset_usdt is not None and price_sol_usdt is not None and price_sol_usdt > 0:
- return price_asset_usdt / price_sol_usdt
- return None
- def update_data_for_plotly_and_table():
- global historical_data_points, latest_values_for_table
- print("数据更新线程启动 (仅现货)...")
- while True:
- fetch_time_full = time.strftime("%Y-%m-%d %H:%M:%S");
- fetch_time_chart = time.strftime("%H:%M:%S")
- oo_data = get_openocean_price_solana(IN_TOKEN_ADDRESS_SOLANA, OUT_TOKEN_ADDRESS_SOLANA,
- AMOUNT_TO_QUERY_OPENOCEAN_IN_SOL)
- spot_usdt_data = get_gateio_spot_price_usdt(GATEIO_SPOT_PAIR_RFC_USDT)
- binance_sol_usdt_data = get_binance_spot_price_usdt(BINANCE_SOL_USDT_PAIR)
- oo_rfc_sol_price = oo_data.get("price_sol_per_rfc");
- spot_rfc_usdt_price = spot_usdt_data.get("price_rfc_usdt")
- sol_usdt_price = binance_sol_usdt_data.get("price_sol_usdt")
- oo_err = oo_data.get("error");
- spot_err = spot_usdt_data.get("error");
- binance_err = binance_sol_usdt_data.get("error")
- spot_rfc_sol_price = convert_usdt_price_to_sol_price(spot_rfc_usdt_price, sol_usdt_price)
- diff_oo_spot_pct = calculate_percentage_diff(oo_rfc_sol_price, spot_rfc_sol_price)
- current_point = {
- "time": fetch_time_chart,
- "oo_rfc_sol": float(oo_rfc_sol_price) if oo_rfc_sol_price else None,
- "spot_rfc_sol": float(spot_rfc_sol_price) if spot_rfc_sol_price else None,
- "diff_oo_spot_rfc_sol": float(diff_oo_spot_pct) if diff_oo_spot_pct is not None else None,
- }
- with data_lock:
- historical_data_points.append(current_point)
- latest_values_for_table["oo_rfc_sol_price"] = str(oo_rfc_sol_price) if oo_rfc_sol_price else "N/A"
- latest_values_for_table["gate_spot_rfc_sol_price"] = str(
- spot_rfc_sol_price) if spot_rfc_sol_price else "N/A"
- latest_values_for_table["sol_usdt_price_binance"] = str(sol_usdt_price) if sol_usdt_price else "N/A"
- latest_values_for_table[
- "diff_oo_vs_spot_rfc_sol_percentage"] = f"{diff_oo_spot_pct:+.4f}%" if diff_oo_spot_pct is not None else "N/A"
- latest_values_for_table["oo_error"] = oo_err;
- latest_values_for_table["gate_spot_error"] = spot_err
- latest_values_for_table["binance_sol_error"] = binance_err;
- latest_values_for_table["last_updated"] = fetch_time_full
- print(
- f"{fetch_time_chart} Fetch | OO_RFC/SOL:{'OK' if oo_rfc_sol_price else 'F'} | Spot_RFC/SOL:{'OK' if spot_rfc_sol_price else 'F'} | SOL/USDT:{'OK' if sol_usdt_price else 'F'}")
- time.sleep(REFRESH_INTERVAL_SECONDS)
- @app.route('/')
- def index_plotly():
- target_asset = latest_values_for_table["target_asset_symbol"]
- return render_template('index_plotly_sol_spot_only.html', target_asset=target_asset,
- spot_pair_usdt=GATEIO_SPOT_PAIR_RFC_USDT, sol_usdt_pair_binance=BINANCE_SOL_USDT_PAIR,
- refresh_interval_ms=REFRESH_INTERVAL_SECONDS * 1000)
- @app.route('/table-data')
- def get_table_data():
- with data_lock: return jsonify(latest_values_for_table)
- @app.route('/plotly-chart-data')
- def get_plotly_chart_data():
- with data_lock:
- points = list(historical_data_points)
- if not points:
- fig = go.Figure();
- fig.update_layout(title_text="暂无数据")
- empty_json = json.loads(json.dumps(fig, cls=PlotlyJSONEncoder))
- return jsonify({"price_chart": empty_json, "diff_chart": empty_json})
- times = [p['time'] for p in points]
- target_asset_symbol = latest_values_for_table["target_asset_symbol"]
- common_xaxis_config = dict(title='时间')
- if len(times) > 1:
- common_xaxis_config['range'] = [times[0], times[-1]]
- else:
- common_xaxis_config['autorange'] = True
- common_legend_config = dict(orientation="h", yanchor="bottom", y=1.02, xanchor="right", x=1)
- fig_prices = go.Figure()
- fig_prices.add_trace(
- go.Scatter(x=times, y=[p['oo_rfc_sol'] for p in points], mode='lines', name='OpenOcean (RFC/SOL)',
- line=dict(color='rgb(75, 192, 192)'),
- hovertemplate=f'<b>OpenOcean (RFC/SOL)</b><br>Time: %{{x}}<br>Price: %{{y:.8f}} SOL<extra></extra>'))
- fig_prices.add_trace(go.Scatter(x=times, y=[p['spot_rfc_sol'] for p in points], mode='lines',
- name=f'Gate Spot ({GATEIO_SPOT_PAIR_RFC_USDT} → RFC/SOL)',
- line=dict(color='rgb(255, 99, 132)'),
- hovertemplate=f'<b>Gate Spot (RFC/SOL)</b><br>Time: %{{x}}<br>Price: %{{y:.8f}} SOL<extra></extra>'))
- fig_prices.update_layout(title_text=f'{target_asset_symbol}/SOL 价格历史', xaxis=common_xaxis_config.copy(),
- yaxis_title=f'价格 (1 {target_asset_symbol} = X SOL)', legend_title_text='平台',
- legend=common_legend_config.copy(), hovermode='x unified',
- margin=dict(l=70, r=30, t=80, b=50))
- fig_diffs = go.Figure()
- fig_diffs.add_trace(go.Scatter(x=times, y=[p['diff_oo_spot_rfc_sol'] for p in points], mode='lines',
- name=f'OO vs Spot (RFC/SOL)', line=dict(color='rgb(255, 159, 64)'),
- hovertemplate=f'<b>OO vs Spot (RFC/SOL)</b><br>Time: %{{x}}<br>Diff: %{{y:+.4f}}%<extra></extra>'))
- fig_diffs.update_layout(title_text=f'{target_asset_symbol}/SOL 价差百分比历史 (OO vs Spot)',
- xaxis=common_xaxis_config.copy(),
- yaxis_title='价差 (%)',
- legend=common_legend_config.copy(),
- # Applying to diff chart as well, or set showlegend=False
- # showlegend=False, # Uncomment this if you don't want legend on diff chart
- yaxis_zeroline=True, hovermode='x unified', margin=dict(l=70, r=30, t=80, b=50))
- combined_figure_data = {"price_chart": json.loads(json.dumps(fig_prices, cls=PlotlyJSONEncoder)),
- "diff_chart": json.loads(json.dumps(fig_diffs, cls=PlotlyJSONEncoder))}
- return jsonify(combined_figure_data)
- if __name__ == "__main__":
- print("应用启动...")
- asset_symbol = GATEIO_SPOT_PAIR_RFC_USDT.split('_')[0]
- print(f"目标资产: {asset_symbol}")
- print(
- f"OpenOcean: {asset_symbol}/SOL (通过 {IN_TOKEN_ADDRESS_SOLANA[-6:]}...SOL / {OUT_TOKEN_ADDRESS_SOLANA[-6:]}...{asset_symbol})")
- print(f"Gate.io 现货: {GATEIO_SPOT_PAIR_RFC_USDT} (将转换为 {asset_symbol}/SOL)")
- print(f"币安转换汇率: {BINANCE_SOL_USDT_PAIR}")
- data_thread = threading.Thread(target=update_data_for_plotly_and_table, daemon=True);
- data_thread.start()
- print(f"Flask 服务将在 http://0.0.0.0:5000 上运行 (刷新间隔: {REFRESH_INTERVAL_SECONDS}s)")
- app.run(debug=False, host='0.0.0.0', port=5000, use_reloader=False)
|