Web Design Update

This commit is contained in:
2026-05-10 18:22:19 +00:00
parent 131dec501a
commit 8d9d29cff1
4 changed files with 121 additions and 12 deletions
+82
View File
@@ -0,0 +1,82 @@
# GMMC Media Server
A high-performance, public media download server built with Rust and Axum. It is designed to serve extremely large files (100GB+) seamlessly and features a retro, Minecraft-themed web dashboard for easy access.
## Features
- **Blazing Fast & Lightweight**: Powered by Rust, Axum, and Tokio.
- **Large File Optimization**: Streams files using configurable chunk sizes (default 8MB) to handle massive files with low memory footprint.
- **Resume Support**: Full support for HTTP `Range` requests, allowing paused or interrupted downloads to be resumed.
- **Retro Dashboard**: A dynamic, Minecraft-style UI built with vanilla HTML/CSS and the VT323 font. Includes 1-click copy-to-clipboard commands for terminal downloading.
- **No Authentication Required**: Built for public, hassle-free access.
- **Security & Access Control**: Built-in security headers and optional IP whitelisting.
## Prerequisites
- [Rust & Cargo](https://rustup.rs/) (edition 2021)
- A `.env` file for server configuration
- A `media_config.json` file for file registry
## Configuration
### 1. Environment Variables (`.env`)
Create a `.env` file in the root directory:
```env
PORT=3000
HOST=0.0.0.0
# Request timeout in seconds (default 3600 for 1 hour)
REQUEST_TIMEOUT_SECS=3600
# Streaming chunk size in KB (default 8192 for 8MB)
CHUNK_SIZE_KB=8192
# Optional: comma-separated list of allowed IPs
# ALLOWED_IPS=127.0.0.1,192.168.1.100
# Optional: Domain for generating absolute download URLs
# DOMAIN=https://downloads.example.com
```
### 2. Media Registry (`media_config.json`)
Define the files you want to host. Create a `media_config.json` in the root directory:
```json
{
"files": {
"ubuntu-iso": {
"name": "Ubuntu 24.04 LTS",
"path": "/path/to/ubuntu-24.04-desktop-amd64.iso",
"description": "The latest LTS release of Ubuntu.",
"content_type": "application/x-iso9660-image"
},
"large-dataset": {
"name": "AI Training Dataset",
"path": "./data/dataset.tar.gz",
"description": "Huge 150GB dataset for machine learning models."
}
}
}
```
## Running the Server
1. Clone the repository and navigate to the project root.
2. Build and run the server:
```bash
cargo run --release
```
3. Open your browser and navigate to `http://localhost:3000` (or your configured `HOST` and `PORT`).
## Usage
- **Web Dashboard**: Visit the root URL to view available files, their sizes, and descriptions. Click "Download" or copy the provided cURL commands.
- **Direct Download (Browser)**: `http://localhost:3000/download/{file_id}`
- **cURL Download**:
```bash
curl -O http://localhost:3000/download/{file_id}
```
*(For resumable downloads via cURL, add `-C -`)*
## Tech Stack
- **[Rust](https://www.rust-lang.org/)**
- **[Axum](https://github.com/tokio-rs/axum)** (Web Framework)
- **[Tokio](https://tokio.rs/)** (Asynchronous Runtime)
- **[Tower](https://github.com/tower-rs/tower)** (Middleware & Networking)
- **[Serde](https://serde.rs/)** (Serialization/Deserialization)
+11
View File
@@ -76,6 +76,7 @@ async fn main() {
let app = Router::new()
.route("/", get(list_files))
.route("/image.png", get(serve_image))
.route("/download/:file_id", get(download_file))
.route("/health", get(health_check))
.layer(middleware::from_fn_with_state(
@@ -207,6 +208,16 @@ async fn health_check() -> impl IntoResponse {
(StatusCode::OK, "OK")
}
async fn serve_image() -> impl IntoResponse {
match tokio::fs::read("templates/image.png").await {
Ok(bytes) => (
[(header::CONTENT_TYPE, "image/png")],
bytes,
).into_response(),
Err(_) => StatusCode::NOT_FOUND.into_response(),
}
}
async fn list_files(State(state): State<AppState>) -> Html<String> {
// Read template from file
let template_path = "templates/index.html";
Binary file not shown.

After

Width:  |  Height:  |  Size: 2.8 MiB

+28 -12
View File
@@ -32,11 +32,10 @@
body {
font-family: 'VT323', monospace;
background-color: var(--mc-bg);
background-image:
linear-gradient(45deg, #1a1a1a 25%, transparent 25%, transparent 75%, #1a1a1a 75%, #1a1a1a),
linear-gradient(45deg, #1a1a1a 25%, transparent 25%, transparent 75%, #1a1a1a 75%, #1a1a1a);
background-size: 40px 40px;
background-position: 0 0, 20px 20px;
background-image: url('/image.png');
background-size: cover;
background-position: center;
background-attachment: fixed;
color: var(--mc-text);
margin: 0;
padding: 24px;
@@ -48,6 +47,11 @@
.container {
max-width: 900px;
margin: 0 auto;
background: rgba(17, 17, 17, 0.85);
padding: 2rem;
border: 4px solid #000;
box-shadow: inset 4px 4px 0px rgba(255, 255, 255, 0.15), inset -4px -4px 0px rgba(0, 0, 0, 0.6);
backdrop-filter: blur(2px);
}
h1 {
@@ -63,6 +67,16 @@
letter-spacing: 2px;
}
a {
color: var(--mc-gold);
text-decoration: none;
text-shadow: var(--mc-text-shadow);
}
a:hover {
color: var(--mc-green);
}
/* Stats Grid - Inventory Style */
.stats {
display: grid;
@@ -72,10 +86,10 @@
}
.stat-card {
background: #212121;
background: rgba(33, 33, 33, 0.9);
padding: 1rem;
border: 2px solid #555;
box-shadow: inset 2px 2px 0px #000, inset -2px -2px 0px #333;
border: 2px solid #000;
box-shadow: inset 2px 2px 0px #444, inset -2px -2px 0px #111;
display: flex;
flex-direction: column;
align-items: center;
@@ -115,8 +129,9 @@
}
.file-item {
background: rgba(0, 0, 0, 0.5);
border: 2px solid #555;
background: rgba(20, 20, 20, 0.9);
border: 2px solid #000;
box-shadow: inset 2px 2px 0px #3a3a3a, inset -2px -2px 0px #0a0a0a;
padding: 1rem;
display: grid;
grid-template-columns: 1fr auto;
@@ -126,8 +141,9 @@
}
.file-item:hover {
background: rgba(0, 0, 0, 0.7);
border-color: #777;
background: rgba(30, 30, 30, 0.95);
border-color: #555;
box-shadow: inset 2px 2px 0px #4a4a4a, inset -2px -2px 0px #1a1a1a;
}
/* 1. Name */