forked from forks/microblog.pub
559 lines
16 KiB
Python
559 lines
16 KiB
Python
|
import base64
|
||
|
import os
|
||
|
import sys
|
||
|
import time
|
||
|
from datetime import datetime
|
||
|
from typing import Any
|
||
|
from typing import Type
|
||
|
|
||
|
import httpx
|
||
|
from dateutil.parser import isoparse
|
||
|
from fastapi import Depends
|
||
|
from fastapi import FastAPI
|
||
|
from fastapi import Request
|
||
|
from fastapi import Response
|
||
|
from fastapi.exceptions import HTTPException
|
||
|
from fastapi.responses import PlainTextResponse
|
||
|
from fastapi.responses import StreamingResponse
|
||
|
from fastapi.staticfiles import StaticFiles
|
||
|
from loguru import logger
|
||
|
from sqlalchemy.orm import Session
|
||
|
from sqlalchemy.orm import joinedload
|
||
|
from starlette.background import BackgroundTask
|
||
|
from starlette.responses import JSONResponse
|
||
|
|
||
|
from app import activitypub as ap
|
||
|
from app import admin
|
||
|
from app import config
|
||
|
from app import httpsig
|
||
|
from app import models
|
||
|
from app import templates
|
||
|
from app.actor import LOCAL_ACTOR
|
||
|
from app.actor import get_actors_metadata
|
||
|
from app.boxes import public_outbox_objects_count
|
||
|
from app.boxes import save_to_inbox
|
||
|
from app.config import BASE_URL
|
||
|
from app.config import DEBUG
|
||
|
from app.config import DOMAIN
|
||
|
from app.config import ID
|
||
|
from app.config import USER_AGENT
|
||
|
from app.config import USERNAME
|
||
|
from app.config import is_activitypub_requested
|
||
|
from app.database import get_db
|
||
|
from app.templates import is_current_user_admin
|
||
|
|
||
|
# TODO(ts):
|
||
|
#
|
||
|
# Next:
|
||
|
# - show likes/announces counter for outbox activities
|
||
|
# - update actor support
|
||
|
# - replies support
|
||
|
# - file upload + place/exif extraction (or not) support
|
||
|
# - custom emoji support
|
||
|
# - hash config/profile to detect when to send Update actor
|
||
|
#
|
||
|
# - [ ] block support
|
||
|
# - [ ] make the media proxy authenticated
|
||
|
# - [ ] prevent SSRF (urlutils from little-boxes)
|
||
|
# - [ ] Dockerization
|
||
|
# - [ ] Webmentions
|
||
|
# - [ ] custom emoji
|
||
|
# - [ ] poll/questions support
|
||
|
# - [ ] cleanup tasks
|
||
|
# - notifs:
|
||
|
# - MENTIONED
|
||
|
# - LIKED
|
||
|
# - ANNOUNCED
|
||
|
# - FOLLOWED
|
||
|
# - UNFOLLOWED
|
||
|
# - POLL_ENDED
|
||
|
|
||
|
app = FastAPI(docs_url=None, redoc_url=None)
|
||
|
app.mount("/static", StaticFiles(directory="app/static"), name="static")
|
||
|
app.include_router(admin.router, prefix="/admin")
|
||
|
app.include_router(admin.unauthenticated_router, prefix="/admin")
|
||
|
|
||
|
logger.configure(extra={"request_id": "no_req_id"})
|
||
|
logger.remove()
|
||
|
logger_format = (
|
||
|
"<green>{time:YYYY-MM-DD HH:mm:ss.SSS}</green> | "
|
||
|
"<level>{level: <8}</level> | "
|
||
|
"<cyan>{name}</cyan>:<cyan>{function}</cyan>:<cyan>{line}</cyan> | "
|
||
|
"{extra[request_id]} - <level>{message}</level>"
|
||
|
)
|
||
|
logger.add(sys.stdout, format=logger_format)
|
||
|
|
||
|
|
||
|
@app.middleware("http")
|
||
|
async def request_middleware(request, call_next):
|
||
|
start_time = time.perf_counter()
|
||
|
request_id = os.urandom(8).hex()
|
||
|
with logger.contextualize(request_id=request_id):
|
||
|
logger.info(
|
||
|
f"{request.client.host}:{request.client.port} - "
|
||
|
f"{request.method} {request.url}"
|
||
|
)
|
||
|
try:
|
||
|
response = await call_next(request)
|
||
|
response.headers["X-Request-ID"] = request_id
|
||
|
response.headers["Server"] = "microblogpub"
|
||
|
elapsed_time = time.perf_counter() - start_time
|
||
|
logger.info(f"status_code={response.status_code} {elapsed_time=:.2f}s")
|
||
|
return response
|
||
|
except Exception:
|
||
|
logger.exception("Request failed")
|
||
|
raise
|
||
|
|
||
|
|
||
|
@app.middleware("http")
|
||
|
async def add_security_headers(request: Request, call_next):
|
||
|
response = await call_next(request)
|
||
|
response.headers["referrer-policy"] = "no-referrer, strict-origin-when-cross-origin"
|
||
|
response.headers["x-content-type-options"] = "nosniff"
|
||
|
response.headers["x-xss-protection"] = "1; mode=block"
|
||
|
response.headers["x-frame-options"] = "SAMEORIGIN"
|
||
|
# TODO(ts): disallow inline CSS?
|
||
|
response.headers["content-security-policy"] = (
|
||
|
"default-src 'self'" + " style-src 'self' 'unsafe-inline';"
|
||
|
)
|
||
|
if not DEBUG:
|
||
|
response.headers[
|
||
|
"strict-transport-security"
|
||
|
] = "max-age=63072000; includeSubdomains"
|
||
|
return response
|
||
|
|
||
|
|
||
|
DEFAULT_CTX = COLLECTION_CTX = [
|
||
|
"https://www.w3.org/ns/activitystreams",
|
||
|
"https://w3id.org/security/v1",
|
||
|
{
|
||
|
# AS ext
|
||
|
"Hashtag": "as:Hashtag",
|
||
|
"sensitive": "as:sensitive",
|
||
|
"manuallyApprovesFollowers": "as:manuallyApprovesFollowers",
|
||
|
# toot
|
||
|
"toot": "http://joinmastodon.org/ns#",
|
||
|
# "featured": "toot:featured",
|
||
|
# schema
|
||
|
"schema": "http://schema.org#",
|
||
|
"PropertyValue": "schema:PropertyValue",
|
||
|
"value": "schema:value",
|
||
|
},
|
||
|
]
|
||
|
|
||
|
|
||
|
class ActivityPubResponse(JSONResponse):
|
||
|
media_type = "application/activity+json"
|
||
|
|
||
|
|
||
|
@app.get("/")
|
||
|
def index(
|
||
|
request: Request,
|
||
|
db: Session = Depends(get_db),
|
||
|
_: httpsig.HTTPSigInfo = Depends(httpsig.httpsig_checker),
|
||
|
) -> templates.TemplateResponse | ActivityPubResponse:
|
||
|
if is_activitypub_requested(request):
|
||
|
return ActivityPubResponse(LOCAL_ACTOR.ap_actor)
|
||
|
|
||
|
outbox_objects = (
|
||
|
db.query(models.OutboxObject)
|
||
|
.filter(
|
||
|
models.OutboxObject.visibility == ap.VisibilityEnum.PUBLIC,
|
||
|
models.OutboxObject.is_deleted.is_(False),
|
||
|
models.OutboxObject.is_hidden_from_homepage.is_(False),
|
||
|
)
|
||
|
.order_by(models.OutboxObject.ap_published_at.desc())
|
||
|
.limit(20)
|
||
|
.all()
|
||
|
)
|
||
|
|
||
|
return templates.render_template(
|
||
|
db,
|
||
|
request,
|
||
|
"index.html",
|
||
|
{"request": request, "objects": outbox_objects},
|
||
|
)
|
||
|
|
||
|
|
||
|
def _build_followx_collection(
|
||
|
db: Session,
|
||
|
model_cls: Type[models.Following | models.Follower],
|
||
|
path: str,
|
||
|
page: bool | None,
|
||
|
next_cursor: str | None,
|
||
|
) -> ap.RawObject:
|
||
|
total_items = db.query(model_cls).count()
|
||
|
|
||
|
if not page and not next_cursor:
|
||
|
return {
|
||
|
"@context": ap.AS_CTX,
|
||
|
"id": ID + path,
|
||
|
"first": ID + path + "?page=true",
|
||
|
"type": "OrderedCollection",
|
||
|
"totalItems": total_items,
|
||
|
}
|
||
|
|
||
|
q = db.query(model_cls).order_by(model_cls.created_at.desc()) # type: ignore
|
||
|
if next_cursor:
|
||
|
q = q.filter(model_cls.created_at < _decode_cursor(next_cursor)) # type: ignore
|
||
|
q = q.limit(20)
|
||
|
|
||
|
items = [followx for followx in q.all()]
|
||
|
next_cursor = None
|
||
|
if (
|
||
|
items
|
||
|
and db.query(model_cls)
|
||
|
.filter(model_cls.created_at < items[-1].created_at)
|
||
|
.count()
|
||
|
> 0
|
||
|
):
|
||
|
next_cursor = _encode_cursor(items[-1].created_at)
|
||
|
|
||
|
collection_page = {
|
||
|
"@context": ap.AS_CTX,
|
||
|
"id": (
|
||
|
ID + path + "?page=true"
|
||
|
if not next_cursor
|
||
|
else ID + path + f"?next_cursor={next_cursor}"
|
||
|
),
|
||
|
"partOf": ID + path,
|
||
|
"type": "OrderedCollectionPage",
|
||
|
"orderedItems": [item.ap_actor_id for item in items],
|
||
|
}
|
||
|
if next_cursor:
|
||
|
collection_page["next"] = ID + path + f"?next_cursor={next_cursor}"
|
||
|
|
||
|
return collection_page
|
||
|
|
||
|
|
||
|
def _encode_cursor(val: datetime) -> str:
|
||
|
return base64.urlsafe_b64encode(val.isoformat().encode()).decode()
|
||
|
|
||
|
|
||
|
def _decode_cursor(cursor: str) -> datetime:
|
||
|
return isoparse(base64.urlsafe_b64decode(cursor).decode())
|
||
|
|
||
|
|
||
|
@app.get("/followers")
|
||
|
def followers(
|
||
|
request: Request,
|
||
|
page: bool | None = None,
|
||
|
next_cursor: str | None = None,
|
||
|
prev_cursor: str | None = None,
|
||
|
db: Session = Depends(get_db),
|
||
|
_: httpsig.HTTPSigInfo = Depends(httpsig.httpsig_checker),
|
||
|
) -> ActivityPubResponse | templates.TemplateResponse:
|
||
|
if is_activitypub_requested(request):
|
||
|
return ActivityPubResponse(
|
||
|
_build_followx_collection(
|
||
|
db=db,
|
||
|
model_cls=models.Follower,
|
||
|
path="/followers",
|
||
|
page=page,
|
||
|
next_cursor=next_cursor,
|
||
|
)
|
||
|
)
|
||
|
|
||
|
followers = (
|
||
|
db.query(models.Follower)
|
||
|
.options(joinedload(models.Follower.actor))
|
||
|
.order_by(models.Follower.created_at.desc())
|
||
|
.limit(20)
|
||
|
.all()
|
||
|
)
|
||
|
|
||
|
# TODO: support next_cursor/prev_cursor
|
||
|
actors_metadata = {}
|
||
|
if is_current_user_admin(request):
|
||
|
actors_metadata = get_actors_metadata(
|
||
|
db,
|
||
|
[f.actor for f in followers],
|
||
|
)
|
||
|
|
||
|
return templates.render_template(
|
||
|
db,
|
||
|
request,
|
||
|
"followers.html",
|
||
|
{
|
||
|
"followers": followers,
|
||
|
"actors_metadata": actors_metadata,
|
||
|
},
|
||
|
)
|
||
|
|
||
|
|
||
|
@app.get("/following")
|
||
|
def following(
|
||
|
request: Request,
|
||
|
page: bool | None = None,
|
||
|
next_cursor: str | None = None,
|
||
|
prev_cursor: str | None = None,
|
||
|
db: Session = Depends(get_db),
|
||
|
_: httpsig.HTTPSigInfo = Depends(httpsig.httpsig_checker),
|
||
|
) -> ActivityPubResponse | templates.TemplateResponse:
|
||
|
if is_activitypub_requested(request):
|
||
|
return ActivityPubResponse(
|
||
|
_build_followx_collection(
|
||
|
db=db,
|
||
|
model_cls=models.Following,
|
||
|
path="/following",
|
||
|
page=page,
|
||
|
next_cursor=next_cursor,
|
||
|
)
|
||
|
)
|
||
|
|
||
|
q = (
|
||
|
db.query(models.Following)
|
||
|
.options(joinedload(models.Following.actor))
|
||
|
.order_by(models.Following.created_at.desc())
|
||
|
.limit(20)
|
||
|
)
|
||
|
following = q.all()
|
||
|
|
||
|
# TODO: support next_cursor/prev_cursor
|
||
|
actors_metadata = {}
|
||
|
if is_current_user_admin(request):
|
||
|
actors_metadata = get_actors_metadata(
|
||
|
db,
|
||
|
[f.actor for f in following],
|
||
|
)
|
||
|
|
||
|
return templates.render_template(
|
||
|
db,
|
||
|
request,
|
||
|
"following.html",
|
||
|
{
|
||
|
"following": following,
|
||
|
"actors_metadata": actors_metadata,
|
||
|
},
|
||
|
)
|
||
|
|
||
|
|
||
|
@app.get("/outbox")
|
||
|
def outbox(
|
||
|
db: Session = Depends(get_db),
|
||
|
_: httpsig.HTTPSigInfo = Depends(httpsig.httpsig_checker),
|
||
|
) -> ActivityPubResponse:
|
||
|
outbox_objects = (
|
||
|
db.query(models.OutboxObject)
|
||
|
.filter(
|
||
|
models.OutboxObject.visibility == ap.VisibilityEnum.PUBLIC,
|
||
|
models.OutboxObject.is_deleted.is_(False),
|
||
|
)
|
||
|
.order_by(models.OutboxObject.ap_published_at.desc())
|
||
|
.limit(20)
|
||
|
.all()
|
||
|
)
|
||
|
return ActivityPubResponse(
|
||
|
{
|
||
|
"@context": DEFAULT_CTX,
|
||
|
"id": f"{ID}/outbox",
|
||
|
"type": "OrderedCollection",
|
||
|
"totalItems": len(outbox_objects),
|
||
|
"orderedItems": [
|
||
|
ap.remove_context(ap.wrap_object_if_needed(a.ap_object))
|
||
|
for a in outbox_objects
|
||
|
],
|
||
|
}
|
||
|
)
|
||
|
|
||
|
|
||
|
@app.get("/o/{public_id}")
|
||
|
def outbox_by_public_id(
|
||
|
public_id: str,
|
||
|
request: Request,
|
||
|
db: Session = Depends(get_db),
|
||
|
_: httpsig.HTTPSigInfo = Depends(httpsig.httpsig_checker),
|
||
|
) -> ActivityPubResponse | templates.TemplateResponse:
|
||
|
# TODO: ACL?
|
||
|
maybe_object = (
|
||
|
db.query(models.OutboxObject)
|
||
|
.filter(
|
||
|
models.OutboxObject.public_id == public_id,
|
||
|
# models.OutboxObject.is_deleted.is_(False),
|
||
|
)
|
||
|
.one_or_none()
|
||
|
)
|
||
|
if not maybe_object:
|
||
|
raise HTTPException(status_code=404)
|
||
|
#
|
||
|
if is_activitypub_requested(request):
|
||
|
return ActivityPubResponse(maybe_object.ap_object)
|
||
|
|
||
|
return templates.render_template(
|
||
|
db,
|
||
|
request,
|
||
|
"object.html",
|
||
|
{
|
||
|
"outbox_object": maybe_object,
|
||
|
},
|
||
|
)
|
||
|
|
||
|
|
||
|
@app.get("/o/{public_id}/activity")
|
||
|
def outbox_activity_by_public_id(
|
||
|
public_id: str,
|
||
|
db: Session = Depends(get_db),
|
||
|
_: httpsig.HTTPSigInfo = Depends(httpsig.httpsig_checker),
|
||
|
) -> ActivityPubResponse:
|
||
|
# TODO: ACL?
|
||
|
maybe_object = (
|
||
|
db.query(models.OutboxObject)
|
||
|
.filter(models.OutboxObject.public_id == public_id)
|
||
|
.one_or_none()
|
||
|
)
|
||
|
if not maybe_object:
|
||
|
raise HTTPException(status_code=404)
|
||
|
|
||
|
return ActivityPubResponse(ap.wrap_object(maybe_object.ap_object))
|
||
|
|
||
|
|
||
|
@app.get("/t/{tag}")
|
||
|
def tag_by_name(
|
||
|
tag: str,
|
||
|
request: Request,
|
||
|
db: Session = Depends(get_db),
|
||
|
_: httpsig.HTTPSigInfo = Depends(httpsig.httpsig_checker),
|
||
|
) -> ActivityPubResponse | templates.TemplateResponse:
|
||
|
# TODO(ts): implement HTML version
|
||
|
# if is_activitypub_requested(request):
|
||
|
return ActivityPubResponse(
|
||
|
{
|
||
|
"@context": ap.AS_CTX,
|
||
|
"id": BASE_URL + f"/t/{tag}",
|
||
|
"type": "OrderedCollection",
|
||
|
"totalItems": 0,
|
||
|
"orderedItems": [],
|
||
|
}
|
||
|
)
|
||
|
|
||
|
|
||
|
@app.post("/inbox")
|
||
|
async def inbox(
|
||
|
request: Request,
|
||
|
db: Session = Depends(get_db),
|
||
|
httpsig_info: httpsig.HTTPSigInfo = Depends(httpsig.enforce_httpsig),
|
||
|
) -> Response:
|
||
|
logger.info(f"headers={request.headers}")
|
||
|
payload = await request.json()
|
||
|
logger.info(f"{payload=}")
|
||
|
save_to_inbox(db, payload)
|
||
|
return Response(status_code=204)
|
||
|
|
||
|
|
||
|
@app.get("/.well-known/webfinger")
|
||
|
def wellknown_webfinger(resource: str) -> JSONResponse:
|
||
|
"""Exposes/servers WebFinger data."""
|
||
|
if resource not in [f"acct:{USERNAME}@{DOMAIN}", ID]:
|
||
|
raise HTTPException(status_code=404)
|
||
|
|
||
|
out = {
|
||
|
"subject": f"acct:{USERNAME}@{DOMAIN}",
|
||
|
"aliases": [ID],
|
||
|
"links": [
|
||
|
{
|
||
|
"rel": "http://webfinger.net/rel/profile-page",
|
||
|
"type": "text/html",
|
||
|
"href": ID,
|
||
|
},
|
||
|
{"rel": "self", "type": "application/activity+json", "href": ID},
|
||
|
{
|
||
|
"rel": "http://ostatus.org/schema/1.0/subscribe",
|
||
|
"template": DOMAIN + "/authorize_interaction?uri={uri}",
|
||
|
},
|
||
|
],
|
||
|
}
|
||
|
|
||
|
return JSONResponse(out, media_type="application/jrd+json; charset=utf-8")
|
||
|
|
||
|
|
||
|
@app.get("/.well-known/nodeinfo")
|
||
|
async def well_known_nodeinfo() -> dict[str, Any]:
|
||
|
return {
|
||
|
"links": [
|
||
|
{
|
||
|
"rel": "http://nodeinfo.diaspora.software/ns/schema/2.1",
|
||
|
"href": f"{BASE_URL}/nodeinfo",
|
||
|
}
|
||
|
]
|
||
|
}
|
||
|
|
||
|
|
||
|
@app.get("/nodeinfo")
|
||
|
def nodeinfo(
|
||
|
db: Session = Depends(get_db),
|
||
|
):
|
||
|
local_posts = public_outbox_objects_count(db)
|
||
|
return JSONResponse(
|
||
|
{
|
||
|
"version": "2.1",
|
||
|
"software": {
|
||
|
"name": "microblogpub",
|
||
|
"version": config.VERSION,
|
||
|
"repository": "https://github.com/tsileo/microblog.pub",
|
||
|
},
|
||
|
"protocols": ["activitypub"],
|
||
|
"services": {"inbound": [], "outbound": []},
|
||
|
"openRegistrations": False,
|
||
|
"usage": {"users": {"total": 1}, "localPosts": local_posts},
|
||
|
"metadata": {
|
||
|
"nodeName": LOCAL_ACTOR.handle,
|
||
|
},
|
||
|
},
|
||
|
media_type=(
|
||
|
"application/json; "
|
||
|
"profile=http://nodeinfo.diaspora.software/ns/schema/2.1#"
|
||
|
),
|
||
|
)
|
||
|
|
||
|
|
||
|
proxy_client = httpx.AsyncClient()
|
||
|
|
||
|
|
||
|
@app.get("/proxy/media/{encoded_url}")
|
||
|
async def serve_proxy_media(request: Request, encoded_url: str) -> StreamingResponse:
|
||
|
# Decode the base64-encoded URL
|
||
|
url = base64.urlsafe_b64decode(encoded_url).decode()
|
||
|
# Request the URL (and filter request headers)
|
||
|
proxy_req = proxy_client.build_request(
|
||
|
request.method,
|
||
|
url,
|
||
|
headers=[
|
||
|
(k, v)
|
||
|
for (k, v) in request.headers.raw
|
||
|
if k.lower()
|
||
|
not in [b"host", b"cookie", b"x-forwarded-for", b"x-real-ip", b"user-agent"]
|
||
|
]
|
||
|
+ [(b"user-agent", USER_AGENT.encode())],
|
||
|
)
|
||
|
proxy_resp = await proxy_client.send(proxy_req, stream=True)
|
||
|
# Filter the headers
|
||
|
proxy_resp_headers = [
|
||
|
(k, v)
|
||
|
for (k, v) in proxy_resp.headers.items()
|
||
|
if k.lower()
|
||
|
in [
|
||
|
"content-length",
|
||
|
"content-type",
|
||
|
"content-range",
|
||
|
"accept-ranges" "etag",
|
||
|
"cache-control",
|
||
|
"expires",
|
||
|
"date",
|
||
|
"last-modified",
|
||
|
]
|
||
|
]
|
||
|
return StreamingResponse(
|
||
|
proxy_resp.aiter_raw(),
|
||
|
status_code=proxy_resp.status_code,
|
||
|
headers=dict(proxy_resp_headers),
|
||
|
background=BackgroundTask(proxy_resp.aclose),
|
||
|
)
|
||
|
|
||
|
|
||
|
@app.get("/robots.txt", response_class=PlainTextResponse)
|
||
|
async def robots_file():
|
||
|
return """User-agent: *
|
||
|
Disallow: /followers
|
||
|
Disallow: /following
|
||
|
Disallow: /admin"""
|