Files
scatcher/app.py

136 lines
4.8 KiB
Python

from fastapi import FastAPI, Request, Response, status
from datetime import datetime
import json
import os
app = FastAPI()
HISTORY_LIMIT = 10
HISTORY_STORAGE = "storage"
def store_last_request(
request_data,
namespace="requests"
):
# Check if the directory exists, if not, create it
if not os.path.exists(HISTORY_STORAGE):
os.makedirs(HISTORY_STORAGE)
filename = f"{HISTORY_STORAGE}/{namespace}.json"
try:
with open(filename, "r") as f:
data = json.load(f)
except (FileNotFoundError, json.JSONDecodeError):
data = []
data.append(request_data)
if len(data) > HISTORY_LIMIT:
data.pop(0)
with open(filename, "w") as f:
json.dump(data, f)
@app.get("/", status_code=status.HTTP_200_OK)
@app.post("/", status_code=status.HTTP_200_OK)
@app.put("/", status_code=status.HTTP_200_OK)
@app.delete("/", status_code=status.HTTP_200_OK)
@app.patch("/", status_code=status.HTTP_200_OK)
@app.options("/", status_code=status.HTTP_200_OK)
@app.head("/", status_code=status.HTTP_200_OK)
@app.get("/{namespace}", status_code=status.HTTP_200_OK)
@app.post("/{namespace}", status_code=status.HTTP_200_OK)
@app.put("/{namespace}", status_code=status.HTTP_200_OK)
@app.delete("/{namespace}", status_code=status.HTTP_200_OK)
@app.patch("/{namespace}", status_code=status.HTTP_200_OK)
@app.options("/{namespace}", status_code=status.HTTP_200_OK)
@app.head("/{namespace}", status_code=status.HTTP_200_OK)
async def catch(request: Request, response: Response, namespace: str = "requests"):
try:
json = await request.json()
except:
json = {"invalid": "json"}
last_request = {
"data": json,
"method": request.method,
"url": str(request.url),
"headers": dict(request.headers),
"time": datetime.now().isoformat(),
}
store_last_request(request_data=last_request, namespace=namespace)
response.status_code = status.HTTP_200_OK
return {"status": "ok", "message": "Request catched."}
@app.get("/api/__help", status_code=status.HTTP_200_OK)
def help():
return {
"message": "This is a simple webhook service. It stores the last 10 requests and returns them on demand.",
"endpoints": {
"/": "Accepts a webhook request and stores it.",
"/{namespace}": "Accepts a webhook request and stores it in the given namespace.",
"/docs": "Swagger UI for the API.",
"/redoc": "ReDoc UI for the API.",
"/api/__last_request": "GET: Returns the last request received.",
"/api/__last_request/{namespace}": "GET: Returns the last request received for the given namespace.",
"/api/__history": "GET: Returns the last 10 requests received.",
"/api/__history/{namespace}": "GET: Returns the last 10 requests received for the given namespace.",
"/api/__history/{id}": "GET: Returns the request with the given ID.",
"/api/__history/{namespace}/{id}": "GET: Returns the request with the given ID for the given namespace.",
"/api/__clear": "GET: Clears the request history.",
"/api/__clear/{namespace}": "GET: Clears the request history for the given namespace.",
},
}
@app.get("/api/__last_request", status_code=status.HTTP_200_OK)
@app.get("/api/__last_request/{namespace}", status_code=status.HTTP_200_OK)
async def last_requests(namespace: str = "requests"):
filename = f"{HISTORY_STORAGE}/{namespace}.json"
try:
with open(filename, "r") as f:
data = json.load(f)
except (FileNotFoundError, json.JSONDecodeError):
data = []
if len(data) == 0:
return {"status": "error", "message": "No requests found."}
return data[-1]
@app.get("/api/__history/{id}", status_code=status.HTTP_200_OK)
@app.get("/api/__history", status_code=status.HTTP_200_OK)
@app.get("/api/__history/{namespace}/{id}", status_code=status.HTTP_200_OK)
@app.get("/api/__history/{namespace}", status_code=status.HTTP_200_OK)
async def history(id: int = None, namespace: str = "requests"):
filename = f"{HISTORY_STORAGE}/{namespace}.json"
try:
with open(filename, "r") as f:
data = json.load(f)
except (FileNotFoundError, json.JSONDecodeError):
data = []
if id is not None:
if len(data) == 0:
return {"status": "error", "message": "No requests found."}
return data[-id]
return data
@app.get("/api/__clear", status_code=status.HTTP_200_OK)
@app.get("/api/__clear/{namespace}", status_code=status.HTTP_200_OK)
async def clear_history(namespace: str = "requests"):
filename = f"{HISTORY_STORAGE}/{namespace}.json"
with open(filename, "w") as f:
json.dump([], f)
return {"status": "ok", "message": "History cleared."}
if __name__ == "__main__":
import uvicorn
uvicorn.run(app, host="0.0.0.0", port=8000)