oStella is a Java service that fetches osu! data and exposes simple HTTP endpoints for status,
multiplayer info, and rendered osu! images.
It is the backend for Seira bot, and also provides a standalone API for other clients to consume.
- PNG score panels for best (
/bo) and recent (/rs) osu! scores - PNG beatmap card endpoint (
/m) - PNG beatmapset card endpoint (
/ms) - PNG player comparison leaderboard endpoint (
/pk) - PNG user PP leaderboard endpoint (
/lb) - Replay video generation for solo and multiplayer replay showcases (
/replay) - Multiplayer room summary endpoint (
/mp) - Current daily challenge endpoint (
/daily) - Health endpoint (
/status) - Automatic OAuth token renewal for osu! API
Here are some demo:
- JDK 25
- Maven 3.9+
- osu! OAuth app credentials (
client_id,client_secret)
- Copy default config file in the project root.
- Install Playwright dependencies if not already present.
- Run.
- Call an endpoint.
The default config file is generated as config.yml when you first start the service.
You can also copy the example config from ostella-example-config.yml
mvn exec:java -e -Dexec.mainClass=com.microsoft.playwright.CLI -Dexec.args="install-deps"mvn -U clean compile exec:javacurl "http://localhost:8721/bo?u=12345678&n=20" --output best_of_20.pngBase URL: http://localhost:<OSTELLA_PORT>
Most JSON endpoints return: {"success": boolean, "message": string, "data": any}.
Image endpoints return PNG bytes. Replay download returns video/mp4.
| Method | Path | Purpose | Query / Path Params | Response |
|---|---|---|---|---|
| GET | /status |
Service health and osu! API health | none | JSON |
| GET | /daily |
Current daily challenge room summary | none | JSON |
| GET | /mp |
Top multiplayer rooms (up to 20) | none | JSON |
| GET | /sms |
Search beatmapsets | q (search keyword) |
JSON |
| GET | /lb |
User PP leaderboard image | u (comma-separated user IDs) |
PNG |
| GET | /bo |
Best-of-N scores image | u (user ID), n (count) |
PNG |
| GET | /rs |
Recent scores image | u (user ID), n (count) |
PNG |
| Method | Path | Purpose | Query / Path Params | Response |
|---|---|---|---|---|
| GET | /m |
Beatmap card image | m (+ optional mod) or ms + i (+ optional mod) or of + u + i |
PNG |
| GET | /ms |
Beatmapset card image | ms or m or of + u + i |
PNG |
| GET | /s |
Score card image | s or m + u or ms + i + u or of + u + i |
PNG |
| GET | /pk |
Compare players on one beatmap | m + u (comma-separated user IDs) or of + i + us + u |
PNG |
Notes:
ofreferences source list type (rsorbo).iis a 1-based index within the referenced list or sorted beatmapset difficulties.
| Method | Path | Purpose | Query / Path Params | Response |
|---|---|---|---|---|
| GET | /replay/status |
Replay renderer overview | none | JSON |
| GET | /replay/render |
Queue single replay render | s or m + u or ms + i + u or of + u + i |
202 JSON |
| GET | /replay/showcase |
Queue multi-score showcase render | s (comma-separated score IDs) or u + m or of + i + us + u |
202 JSON |
| GET | /replay/status/{jobId} |
Get render job state | {jobId} |
JSON |
| GET | /replay/video/{jobId} |
Download rendered video | {jobId} |
video/mp4 |
| DELETE | /replay/video/{jobId} |
Remove rendered video and job metadata | {jobId} |
text |
| Method | Path | Purpose | Query / Path Params | Response |
|---|---|---|---|---|
| GET | /debug/bypass |
Debug passthrough call to osu! API helper | raw query string | JSON |
oStella caches downloaded assets on disk to reduce repeated upstream requests.
- Beatmap files used for difficulty/PP calculation are cached in
data/cache/beatmap/. - Remote images used by templates (avatars, covers, flags) are cached in
data/cache/image/. - Image cache keys are generated from the image URL, so the same URL reuses the same cached file.
- Cache is file-based and persists across restarts.
- Public HTTP endpoints currently use on-demand caching (download when missing).
To clear cache, stop the service and remove files under data/cache/; they will be re-downloaded on future requests.
mvn clean package generates:
target/oStella-{version}.jartarget/oStella-{version}-jar-with-dependencies.jar
Image rendering depends on Playwright Chromium. If your environment is missing browser binaries, when first started, Playwright will attempt to download them.
oStella is designed to be highly concurrent, but its resource usage scales directly with how you configure its rendering features. The core web server is incredibly lightweight, but image (Playwright/Chromium) and video (Danser) rendering require careful hardware consideration.
- CPU: 2+ Cores (4+ Cores heavily recommended if video rendering is enabled)
- RAM: 2 GB minimum (4 GB recommended for stable multi-worker rendering)
- Storage: 5+ GB free space (for caching osu! beatmaps, replays, and rendered videos)
Memory consumption is strictly controlled by your worker pool configurations. By default, oStella prevents Out-Of-Memory crashes by queuing requests rather than spawning infinite browser instances.
- Core Java Server: ~250MB - 500MB (depending on JVM garbage collection and cache size).
- Image Rendering (Playwright/Chromium): ~100MB - 150MB per active worker.
If you configure
ostella.renderWorkers: 4, expect Chromium to reserve up to ~600MB of RAM under peak load. - Video Rendering (Danser): ~200MB - 300MB per active Danser instance during an active render.
- API Routing & Network: Near 0% CPU impact. Asynchronous request handling allows the server to idle efficiently.
- Image Rendering: Moderate, bursty CPU usage. Chromium utilizes separate OS processes for rendering, meaning concurrent image requests will actively utilize multiple CPU cores for brief moments.
- Video Rendering (Replays): Extreme CPU usage. Software encoding (e.g.,
libx264) will easily pin your CPU to 100%.
If you are running oStella on a low-resource environment (e.g., 2GB RAM, 2 CPU cores), it is crucial to:
- Limit your Playwright worker pool to
2or3to prevent memory exhaustion. - Avoid enabling video rendering or limit it to a single worker with hardware encoding.
- Monitor your server's resource usage closely, especially under load, to ensure it remains responsive
Log files are written to logs/:
latest.log(application logs)javalin-server.log(Javalin/Jetty logs)danser.log(Danser-CLI logs)- rolled
*.log.gzarchives
MIT. See LICENSE.