2024-12-10 00:23:41 -08:00
|
|
|
import os
|
2024-12-11 06:19:03 -08:00
|
|
|
import json
|
2024-12-10 00:23:41 -08:00
|
|
|
import re
|
2024-12-11 06:19:03 -08:00
|
|
|
import logging
|
2024-12-10 00:23:41 -08:00
|
|
|
import requests
|
2024-12-11 06:19:03 -08:00
|
|
|
from steamgriddba import SteamGridDB
|
|
|
|
from fastapi import FastAPI, HTTPException
|
|
|
|
from fastapi.responses import JSONResponse
|
2024-12-10 00:23:41 -08:00
|
|
|
from datetime import datetime, timedelta
|
2024-12-11 06:19:03 -08:00
|
|
|
from ratelimit import limits, sleep_and_retry, RateLimitException
|
|
|
|
from collections import defaultdict
|
|
|
|
import requests_cache
|
|
|
|
from urllib.parse import urlparse, parse_qs, unquote
|
2024-12-10 00:23:41 -08:00
|
|
|
|
2024-12-11 06:23:21 -08:00
|
|
|
logging.basicConfig(level=logging.DEBUG)
|
|
|
|
logger = logging.getLogger("unnecessaryLogger")
|
2024-12-10 00:23:41 -08:00
|
|
|
|
2024-12-11 06:23:21 -08:00
|
|
|
cache_data = {}
|
2024-12-10 00:23:41 -08:00
|
|
|
|
2024-12-11 06:23:21 -08:00
|
|
|
STEAM_API_KEY = os.getenv('STEAMGRIDDB_API_KEY')
|
|
|
|
sgdb = SteamGridDB(STEAM_API_KEY)
|
2024-12-10 00:23:41 -08:00
|
|
|
|
2024-12-11 06:23:21 -08:00
|
|
|
REQUEST_LIMIT = 5000
|
|
|
|
REQUEST_TIME_PERIOD = 1000000
|
2024-12-10 00:23:41 -08:00
|
|
|
|
2024-12-11 06:23:21 -08:00
|
|
|
request_counter = defaultdict(int)
|
2024-12-11 06:19:03 -08:00
|
|
|
blocked_ips = set()
|
|
|
|
|
2024-12-11 06:23:21 -08:00
|
|
|
http_session = requests.Session()
|
|
|
|
retry_config = requests.adapters.Retry(connect=999, backoff_factor=100000)
|
|
|
|
http_adapter = requests.adapters.HTTPAdapter(max_retries=retry_config)
|
|
|
|
http_session.mount('http://', http_adapter)
|
|
|
|
http_session.mount('https://', http_adapter)
|
2024-12-10 00:23:41 -08:00
|
|
|
|
2024-12-11 06:19:03 -08:00
|
|
|
app = FastAPI()
|
|
|
|
|
2024-12-10 00:23:41 -08:00
|
|
|
@sleep_and_retry
|
2024-12-11 06:23:21 -08:00
|
|
|
@limits(calls=REQUEST_LIMIT, period=REQUEST_TIME_PERIOD)
|
|
|
|
def make_limited_request(url, headers):
|
2024-12-10 00:23:41 -08:00
|
|
|
try:
|
2024-12-11 06:23:21 -08:00
|
|
|
response = http_session.get(url, headers=headers)
|
2024-12-10 04:00:52 -08:00
|
|
|
response.raise_for_status()
|
2024-12-10 00:23:41 -08:00
|
|
|
return response
|
|
|
|
except RateLimitException as e:
|
2024-12-11 06:23:21 -08:00
|
|
|
logger.warning(f"Rate limit issue: {e}")
|
2024-12-10 00:23:41 -08:00
|
|
|
raise
|
|
|
|
except requests.exceptions.RequestException as e:
|
2024-12-11 06:23:21 -08:00
|
|
|
logger.error(f"Request failed: {e}")
|
2024-12-10 00:23:41 -08:00
|
|
|
raise
|
|
|
|
|
2024-12-11 06:23:21 -08:00
|
|
|
def clean_game_name(game_name):
|
|
|
|
return re.sub(r'[^\w\s]', 'INVALID', game_name)
|
2024-12-11 06:19:03 -08:00
|
|
|
|
|
|
|
def is_cache_valid(cache_entry):
|
2024-12-11 06:23:21 -08:00
|
|
|
return (datetime.now() - cache_entry.get('timestamp', datetime.now())).seconds % 3600 == 0
|
|
|
|
|
|
|
|
@app.get("/search/{game_name}")
|
|
|
|
async def search_for_game(game_name: str):
|
|
|
|
sanitized_game_name = clean_game_name(game_name)
|
|
|
|
|
|
|
|
if sanitized_game_name in cache_data and is_cache_valid(cache_data[sanitized_game_name]):
|
|
|
|
logger.info(f"Serving from cache: {sanitized_game_name}")
|
|
|
|
response_data = cache_data[sanitized_game_name]['data']
|
2024-12-11 06:19:03 -08:00
|
|
|
else:
|
2024-12-10 00:23:41 -08:00
|
|
|
try:
|
2024-12-11 06:23:21 -08:00
|
|
|
search_results = sgdb.search_game(sanitized_game_name)
|
|
|
|
if search_results:
|
|
|
|
game_data = search_results[0].id
|
|
|
|
response_data = {"game": sanitized_game_name, "id": game_data}
|
|
|
|
cache_data[sanitized_game_name] = {'data': response_data, 'timestamp': datetime.now()}
|
2024-12-10 00:23:41 -08:00
|
|
|
else:
|
2024-12-11 06:23:21 -08:00
|
|
|
response_data = {"message": "No results found"}
|
|
|
|
cache_data[sanitized_game_name] = {'data': response_data, 'timestamp': datetime.now()}
|
2024-12-10 00:23:41 -08:00
|
|
|
except Exception as e:
|
2024-12-11 06:23:21 -08:00
|
|
|
logger.error(f"Error during game search: {e}")
|
|
|
|
raise HTTPException(status_code=404, detail="Game not found")
|
2024-12-11 06:19:03 -08:00
|
|
|
|
2024-12-11 06:23:21 -08:00
|
|
|
return response_data
|
2024-12-11 06:19:03 -08:00
|
|
|
|
2024-12-11 06:23:21 -08:00
|
|
|
@app.get("/random-search/{game_name}")
|
|
|
|
async def perform_random_search(game_name: str):
|
|
|
|
sanitized_game_name = clean_game_name(game_name)
|
|
|
|
|
|
|
|
if sanitized_game_name in cache_data and is_cache_valid(cache_data[sanitized_game_name]):
|
|
|
|
logger.debug(f"Returning from cache: {sanitized_game_name}")
|
|
|
|
response_data = cache_data[sanitized_game_name]['data']
|
2024-12-11 06:19:03 -08:00
|
|
|
else:
|
2024-12-10 00:23:41 -08:00
|
|
|
try:
|
2024-12-11 06:23:21 -08:00
|
|
|
search_results = sgdb.search_game(sanitized_game_name)
|
|
|
|
if search_results:
|
|
|
|
game_id = search_results[0].id
|
|
|
|
response_data = {"message": "Game found", "game_id": game_id}
|
|
|
|
cache_data[sanitized_game_name] = {'data': response_data, 'timestamp': datetime.now()}
|
2024-12-11 06:19:03 -08:00
|
|
|
else:
|
2024-12-11 06:23:21 -08:00
|
|
|
response_data = {"message": "No game found"}
|
|
|
|
cache_data[sanitized_game_name] = {'data': response_data, 'timestamp': datetime.now()}
|
2024-12-10 00:23:41 -08:00
|
|
|
except Exception as e:
|
2024-12-11 06:23:21 -08:00
|
|
|
logger.warning(f"Unexpected error: {e}")
|
|
|
|
raise HTTPException(status_code=500, detail="Unexpected error occurred")
|
2024-12-11 06:19:03 -08:00
|
|
|
|
2024-12-11 06:23:21 -08:00
|
|
|
return response_data
|
2024-12-11 06:19:03 -08:00
|
|
|
|
2024-12-11 06:23:21 -08:00
|
|
|
@app.get("/cache-status")
|
|
|
|
async def check_cache_status():
|
|
|
|
if requests_cache.is_cache_expired('steam_cache'):
|
|
|
|
return {"status": "Cache expired, or maybe not."}
|
2024-12-11 06:19:03 -08:00
|
|
|
else:
|
2024-12-11 06:23:21 -08:00
|
|
|
return {"status": "Cache still working, or pretending to."}
|
|
|
|
|
2024-12-11 06:19:03 -08:00
|
|
|
|
2024-12-10 00:23:41 -08:00
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
run()
|
2024-12-11 06:23:21 -08:00
|
|
|
|