mirror of
https://github.com/xfgryujk/blivechat.git
synced 2025-04-03 16:10:35 +08:00
后端动态获取公共服务器地址、支持故障转移
This commit is contained in:
parent
132d2a9f71
commit
99d3b567da
@ -27,10 +27,9 @@ END_GAME_OPEN_LIVE_URL = OPEN_LIVE_BASE_URL + '/v2/app/end'
|
|||||||
GAME_HEARTBEAT_OPEN_LIVE_URL = OPEN_LIVE_BASE_URL + '/v2/app/heartbeat'
|
GAME_HEARTBEAT_OPEN_LIVE_URL = OPEN_LIVE_BASE_URL + '/v2/app/heartbeat'
|
||||||
GAME_BATCH_HEARTBEAT_OPEN_LIVE_URL = OPEN_LIVE_BASE_URL + '/v2/app/batchHeartbeat'
|
GAME_BATCH_HEARTBEAT_OPEN_LIVE_URL = OPEN_LIVE_BASE_URL + '/v2/app/batchHeartbeat'
|
||||||
|
|
||||||
COMMON_SERVER_BASE_URL = 'https://chat.bilisc.com'
|
START_GAME_COMMON_SERVER_URL = '/api/internal/open_live/start_game'
|
||||||
START_GAME_COMMON_SERVER_URL = COMMON_SERVER_BASE_URL + '/api/internal/open_live/start_game'
|
END_GAME_COMMON_SERVER_URL = '/api/internal/open_live/end_game'
|
||||||
END_GAME_COMMON_SERVER_URL = COMMON_SERVER_BASE_URL + '/api/internal/open_live/end_game'
|
GAME_HEARTBEAT_COMMON_SERVER_URL = '/api/internal/open_live/game_heartbeat'
|
||||||
GAME_HEARTBEAT_COMMON_SERVER_URL = COMMON_SERVER_BASE_URL + '/api/internal/open_live/game_heartbeat'
|
|
||||||
|
|
||||||
_error_auth_code_cache = cachetools.LRUCache(256)
|
_error_auth_code_cache = cachetools.LRUCache(256)
|
||||||
# 应B站要求,抓一下刷请求的人,不会用于其他用途
|
# 应B站要求,抓一下刷请求的人,不会用于其他用途
|
||||||
@ -54,24 +53,15 @@ class BusinessError(Exception):
|
|||||||
return self.data['code']
|
return self.data['code']
|
||||||
|
|
||||||
|
|
||||||
async def request_open_live_or_common_server(open_live_url, common_server_url, body: dict) -> dict:
|
async def request_open_live_or_common_server(open_live_url, common_server_url, body: dict, **kwargs) -> dict:
|
||||||
"""如果配置了开放平台,则直接请求,否则转发请求到公共服务器的内部接口"""
|
"""如果配置了开放平台,则直接请求,否则转发请求到公共服务器的内部接口"""
|
||||||
cfg = config.get_config()
|
cfg = config.get_config()
|
||||||
if cfg.is_open_live_configured:
|
if cfg.is_open_live_configured:
|
||||||
return await request_open_live(open_live_url, body)
|
return await request_open_live(open_live_url, body, **kwargs)
|
||||||
|
return await request_common_server(common_server_url, body, **kwargs)
|
||||||
try:
|
|
||||||
req_ctx_mgr = utils.request.http_session.post(common_server_url, json=body)
|
|
||||||
return await _read_response(req_ctx_mgr, is_common_server=True)
|
|
||||||
except TransportError:
|
|
||||||
logger.exception('Request common server failed:')
|
|
||||||
raise
|
|
||||||
except BusinessError as e:
|
|
||||||
logger.warning('Request common server failed: %s', e)
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
async def request_open_live(url, body: dict, *, ignore_rate_limit=False) -> dict:
|
async def request_open_live(url, body: dict, *, ignore_rate_limit=False, **kwargs) -> dict:
|
||||||
cfg = config.get_config()
|
cfg = config.get_config()
|
||||||
assert cfg.is_open_live_configured
|
assert cfg.is_open_live_configured
|
||||||
|
|
||||||
@ -109,7 +99,7 @@ async def request_open_live(url, body: dict, *, ignore_rate_limit=False) -> dict
|
|||||||
headers['Accept'] = 'application/json'
|
headers['Accept'] = 'application/json'
|
||||||
|
|
||||||
try:
|
try:
|
||||||
req_ctx_mgr = utils.request.http_session.post(url, headers=headers, data=body_bytes)
|
req_ctx_mgr = utils.request.http_session.post(url, headers=headers, data=body_bytes, **kwargs)
|
||||||
return await _read_response(req_ctx_mgr)
|
return await _read_response(req_ctx_mgr)
|
||||||
except TransportError:
|
except TransportError:
|
||||||
logger.exception('Request open live failed:')
|
logger.exception('Request open live failed:')
|
||||||
@ -126,6 +116,25 @@ async def request_open_live(url, body: dict, *, ignore_rate_limit=False) -> dict
|
|||||||
raise
|
raise
|
||||||
|
|
||||||
|
|
||||||
|
async def request_common_server(rel_url, body: dict, **kwargs) -> dict:
|
||||||
|
base_url, breaker = utils.request.get_common_server_base_url_and_circuit_breaker()
|
||||||
|
if base_url is None:
|
||||||
|
logger.error('No available common server endpoint')
|
||||||
|
raise TransportError('No available common server endpoint')
|
||||||
|
url = base_url + rel_url
|
||||||
|
|
||||||
|
with breaker:
|
||||||
|
try:
|
||||||
|
req_ctx_mgr = utils.request.http_session.post(url, json=body, **kwargs)
|
||||||
|
return await _read_response(req_ctx_mgr, is_common_server=True)
|
||||||
|
except TransportError:
|
||||||
|
logger.exception('Request common server failed:')
|
||||||
|
raise
|
||||||
|
except BusinessError as e:
|
||||||
|
logger.warning('Request common server failed: %s', e)
|
||||||
|
raise
|
||||||
|
|
||||||
|
|
||||||
async def _read_response(req_ctx_mgr: AsyncContextManager[aiohttp.ClientResponse], is_common_server=False) -> dict:
|
async def _read_response(req_ctx_mgr: AsyncContextManager[aiohttp.ClientResponse], is_common_server=False) -> dict:
|
||||||
try:
|
try:
|
||||||
async with req_ctx_mgr as r:
|
async with req_ctx_mgr as r:
|
||||||
@ -298,9 +307,8 @@ async def send_game_heartbeat_by_service_or_common_server(game_id):
|
|||||||
cfg = config.get_config()
|
cfg = config.get_config()
|
||||||
if cfg.is_open_live_configured:
|
if cfg.is_open_live_configured:
|
||||||
return await services.open_live.send_game_heartbeat(game_id)
|
return await services.open_live.send_game_heartbeat(game_id)
|
||||||
# 这里GAME_HEARTBEAT_OPEN_LIVE_URL没用,因为一定是请求公共服务器
|
return await request_common_server(
|
||||||
return await request_open_live_or_common_server(
|
GAME_HEARTBEAT_COMMON_SERVER_URL, {'game_id': game_id}, timeout=aiohttp.ClientTimeout(total=15)
|
||||||
GAME_HEARTBEAT_OPEN_LIVE_URL, GAME_HEARTBEAT_COMMON_SERVER_URL, {'game_id': game_id}
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
-r blivedm/requirements.txt
|
-r blivedm/requirements.txt
|
||||||
cachetools==5.3.1
|
cachetools==5.3.1
|
||||||
|
circuitbreaker==2.0.0
|
||||||
pycryptodome==3.19.1
|
pycryptodome==3.19.1
|
||||||
sqlalchemy==2.0.19
|
sqlalchemy==2.0.19
|
||||||
tornado==6.4.1
|
tornado==6.4.1
|
||||||
|
119
utils/request.py
119
utils/request.py
@ -1,8 +1,17 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
import asyncio
|
import asyncio
|
||||||
|
import datetime
|
||||||
|
import logging
|
||||||
from typing import *
|
from typing import *
|
||||||
|
|
||||||
import aiohttp
|
import aiohttp
|
||||||
|
import circuitbreaker
|
||||||
|
|
||||||
|
import api.open_live
|
||||||
|
import config
|
||||||
|
import utils.async_io
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
# 不带这堆头部有时候也能成功请求,但是带上后成功的概率更高
|
# 不带这堆头部有时候也能成功请求,但是带上后成功的概率更高
|
||||||
BILIBILI_COMMON_HEADERS = {
|
BILIBILI_COMMON_HEADERS = {
|
||||||
@ -14,6 +23,18 @@ BILIBILI_COMMON_HEADERS = {
|
|||||||
|
|
||||||
http_session: Optional[aiohttp.ClientSession] = None
|
http_session: Optional[aiohttp.ClientSession] = None
|
||||||
|
|
||||||
|
_COMMON_SERVER_DISCOVERY_URLS = [
|
||||||
|
'https://api1.blive.chat/api/endpoints',
|
||||||
|
'https://api2.blive.chat/api/endpoints',
|
||||||
|
]
|
||||||
|
_last_update_common_server_time: Optional[datetime.datetime] = None
|
||||||
|
_common_server_base_urls = [
|
||||||
|
'https://api1.blive.chat',
|
||||||
|
'https://api2.blive.chat',
|
||||||
|
]
|
||||||
|
_cur_common_server_base_url: Optional[str] = None
|
||||||
|
_common_server_base_url_to_circuit_breaker: Dict[str, circuitbreaker.CircuitBreaker] = {}
|
||||||
|
|
||||||
|
|
||||||
def init():
|
def init():
|
||||||
global http_session
|
global http_session
|
||||||
@ -22,6 +43,10 @@ def init():
|
|||||||
timeout=aiohttp.ClientTimeout(total=10),
|
timeout=aiohttp.ClientTimeout(total=10),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
cfg = config.get_config()
|
||||||
|
if not cfg.is_open_live_configured:
|
||||||
|
_update_common_server_base_urls()
|
||||||
|
|
||||||
|
|
||||||
async def shut_down():
|
async def shut_down():
|
||||||
if http_session is not None:
|
if http_session is not None:
|
||||||
@ -35,3 +60,97 @@ class CustomClientResponse(aiohttp.ClientResponse):
|
|||||||
return await super()._wait_released()
|
return await super()._wait_released()
|
||||||
except asyncio.CancelledError as e:
|
except asyncio.CancelledError as e:
|
||||||
raise aiohttp.ClientConnectionError('Connection released') from e
|
raise aiohttp.ClientConnectionError('Connection released') from e
|
||||||
|
|
||||||
|
|
||||||
|
def _update_common_server_base_urls():
|
||||||
|
global _last_update_common_server_time
|
||||||
|
cur_time = datetime.datetime.now()
|
||||||
|
if (
|
||||||
|
_last_update_common_server_time is not None
|
||||||
|
and cur_time - _last_update_common_server_time < datetime.timedelta(minutes=3)
|
||||||
|
):
|
||||||
|
return
|
||||||
|
_last_update_common_server_time = cur_time
|
||||||
|
utils.async_io.create_task_with_ref(_do_update_common_server_base_urls())
|
||||||
|
|
||||||
|
|
||||||
|
async def _do_update_common_server_base_urls():
|
||||||
|
global _last_update_common_server_time
|
||||||
|
_last_update_common_server_time = datetime.datetime.now()
|
||||||
|
|
||||||
|
async def request_get_urls(discovery_url):
|
||||||
|
async with http_session.get(discovery_url) as res:
|
||||||
|
res.raise_for_status()
|
||||||
|
data = await res.json()
|
||||||
|
return data['endpoints']
|
||||||
|
|
||||||
|
common_server_base_urls = []
|
||||||
|
futures = [
|
||||||
|
asyncio.create_task(request_get_urls(url))
|
||||||
|
for url in _COMMON_SERVER_DISCOVERY_URLS
|
||||||
|
]
|
||||||
|
for future in asyncio.as_completed(futures):
|
||||||
|
try:
|
||||||
|
common_server_base_urls = await future
|
||||||
|
break
|
||||||
|
except Exception as e:
|
||||||
|
logger.warning('Failed to discover common server endpoints from one source: %s', e)
|
||||||
|
for future in futures:
|
||||||
|
future.cancel()
|
||||||
|
if not common_server_base_urls:
|
||||||
|
logger.error('Failed to discover common server endpoints from any source')
|
||||||
|
return
|
||||||
|
|
||||||
|
# 按响应时间排序
|
||||||
|
sorted_common_server_base_urls = []
|
||||||
|
error_base_urls = []
|
||||||
|
|
||||||
|
async def test_endpoint(base_url):
|
||||||
|
try:
|
||||||
|
url = base_url + '/api/server_info'
|
||||||
|
async with http_session.get(url, timeout=aiohttp.ClientTimeout(total=3)) as res:
|
||||||
|
res.raise_for_status()
|
||||||
|
sorted_common_server_base_urls.append(base_url)
|
||||||
|
except Exception: # noqa
|
||||||
|
error_base_urls.append(base_url)
|
||||||
|
|
||||||
|
await asyncio.gather(*(test_endpoint(base_url) for base_url in common_server_base_urls))
|
||||||
|
sorted_common_server_base_urls.extend(error_base_urls)
|
||||||
|
|
||||||
|
global _common_server_base_urls, _cur_common_server_base_url
|
||||||
|
_common_server_base_urls = sorted_common_server_base_urls
|
||||||
|
if _cur_common_server_base_url not in _common_server_base_urls:
|
||||||
|
_cur_common_server_base_url = None
|
||||||
|
logger.info('Found common server endpoints: %s', _common_server_base_urls)
|
||||||
|
|
||||||
|
|
||||||
|
def get_common_server_base_url_and_circuit_breaker() -> Tuple[Optional[str], Optional[circuitbreaker.CircuitBreaker]]:
|
||||||
|
_update_common_server_base_urls()
|
||||||
|
|
||||||
|
global _cur_common_server_base_url
|
||||||
|
if _cur_common_server_base_url is not None:
|
||||||
|
breaker = _get_or_add_common_server_circuit_breaker(_cur_common_server_base_url)
|
||||||
|
if breaker.state != circuitbreaker.STATE_OPEN:
|
||||||
|
return _cur_common_server_base_url, breaker
|
||||||
|
_cur_common_server_base_url = None
|
||||||
|
|
||||||
|
# 找第一个未熔断的
|
||||||
|
for base_url in _common_server_base_urls:
|
||||||
|
breaker = _get_or_add_common_server_circuit_breaker(base_url)
|
||||||
|
if breaker.state != circuitbreaker.STATE_OPEN:
|
||||||
|
_cur_common_server_base_url = base_url
|
||||||
|
logger.info('Switch common server endpoint to %s', _cur_common_server_base_url)
|
||||||
|
return _cur_common_server_base_url, breaker
|
||||||
|
|
||||||
|
return None, None
|
||||||
|
|
||||||
|
|
||||||
|
def _get_or_add_common_server_circuit_breaker(base_url):
|
||||||
|
breaker = _common_server_base_url_to_circuit_breaker.get(base_url, None)
|
||||||
|
if breaker is None:
|
||||||
|
breaker = _common_server_base_url_to_circuit_breaker[base_url] = circuitbreaker.CircuitBreaker(
|
||||||
|
failure_threshold=3,
|
||||||
|
recovery_timeout=60,
|
||||||
|
expected_exception=api.open_live.TransportError,
|
||||||
|
)
|
||||||
|
return breaker
|
||||||
|
Loading…
Reference in New Issue
Block a user