Compare commits
5 Commits
d4159dcd9e
...
master
| Author | SHA1 | Date | |
|---|---|---|---|
| 6c8e42f1ef | |||
| 6090d8f596 | |||
| 205d62a634 | |||
| ecc96a3e28 | |||
| a0f2316d53 |
10
.env.example
10
.env.example
@@ -9,10 +9,11 @@ TIMEZONE=America/Edmonton
|
|||||||
PUBLIC_UPLOAD_PAGE_ENABLED=true
|
PUBLIC_UPLOAD_PAGE_ENABLED=true
|
||||||
|
|
||||||
# Local dedupe cache (SQLite)
|
# Local dedupe cache (SQLite)
|
||||||
STATE_DB=./data/state.db
|
#STATE_DB=./data/state.db
|
||||||
|
|
||||||
# Base URL for generating absolute invite links (recommended for production)
|
# Base URL for generating absolute invite links
|
||||||
# e.g., PUBLIC_BASE_URL=https://photos.example.com
|
# Recommended for production, also sets CORS headers
|
||||||
|
# e.g., PUBLIC_BASE_URL=https://upload.example.com
|
||||||
#PUBLIC_BASE_URL=
|
#PUBLIC_BASE_URL=
|
||||||
|
|
||||||
LOG_LEVEL=INFO
|
LOG_LEVEL=INFO
|
||||||
@@ -27,5 +28,8 @@ CHUNK_SIZE_MB=50
|
|||||||
# create a bot using @BotFather then copy the API key here
|
# create a bot using @BotFather then copy the API key here
|
||||||
# get your account's ID by messaging https://t.me/userinfobot
|
# get your account's ID by messaging https://t.me/userinfobot
|
||||||
# Leave these blank to disable
|
# Leave these blank to disable
|
||||||
|
# Example:
|
||||||
|
# TELEGRAM_BOT_API_KEY=1234567890:ABCDefghIjKlmnOPQRsT-UVWXyzABCdefGH
|
||||||
|
# TELEGRAM_BOT_OWNER_ID=12345678
|
||||||
TELEGRAM_BOT_API_KEY=
|
TELEGRAM_BOT_API_KEY=
|
||||||
TELEGRAM_BOT_OWNER_ID=
|
TELEGRAM_BOT_OWNER_ID=
|
||||||
|
|||||||
@@ -8,8 +8,7 @@ ENV PYTHONDONTWRITEBYTECODE=1 \
|
|||||||
|
|
||||||
# Install Python deps
|
# Install Python deps
|
||||||
COPY requirements.txt .
|
COPY requirements.txt .
|
||||||
RUN pip install --no-cache-dir -r requirements.txt \
|
RUN pip install --no-cache-dir -r requirements.txt
|
||||||
&& pip install --no-cache-dir python-multipart
|
|
||||||
|
|
||||||
# Copy app code
|
# Copy app code
|
||||||
COPY . /file_drop
|
COPY . /file_drop
|
||||||
|
|||||||
168
README.md
168
README.md
@@ -1,18 +1,19 @@
|
|||||||
# File Drop Uploader
|
# File Drop Uploader
|
||||||
|
|
||||||
A tiny web app for collecting files and media and saving them to the local filesystem.
|
A self-hosted web app for uploading files and media and saving them to the filesystem on your server.
|
||||||
Admin users log in to create public invite links; invite links are always public-by-URL. A public uploader page is optional and enabled by default.
|
Useful for letting people upload vacation photos, etc. just by sending them a link.
|
||||||
|
|
||||||
Forked from "Immich Drop Uploader": https://github.com/Nasogaa/immich-drop
|
Admin user can create invite links with optional limits and password protection. A public uploader page is optional and enabled by default.
|
||||||
|
|
||||||

|
[View Screenshots](screenshots.md)
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
- **Local Saving:** All uploaded files are saved to the local filesystem.
|
- **Local Saving:** All uploaded files are saved to the server's local filesystem.
|
||||||
- **Invite Links:** Create public-by-URL links for uploads; one-time or multi-use.
|
- **Drag and Drop:** Upload multiple files and folders by dragging them onto the page.
|
||||||
|
- **Invite Links:** Create sharable links for uploads; one-time or multi-use.
|
||||||
- **Manage Links:** Search/sort, enable/disable, delete, edit name/expiry.
|
- **Manage Links:** Search/sort, enable/disable, delete, edit name/expiry.
|
||||||
- **Passwords (optional):** Protect invites with a password gate.
|
- **Passwords (optional):** Protect invite links with a password.
|
||||||
- **Albums:** Upload into a specific folder (auto-create supported). Preserves client-side folder structure on upload.
|
- **Albums:** Upload into a specific folder (auto-create supported). Preserves client-side folder structure on upload.
|
||||||
- **Duplicate Prevention:** Local SHA‑1 cache prevents re-uploading the same file.
|
- **Duplicate Prevention:** Local SHA‑1 cache prevents re-uploading the same file.
|
||||||
- **Telegram Notifications (optional):** Get notified via Telegram when upload batches are complete.
|
- **Telegram Notifications (optional):** Get notified via Telegram when upload batches are complete.
|
||||||
@@ -26,7 +27,9 @@ Clone the repo.
|
|||||||
|
|
||||||
Copy `.env.example` to `.env` and edit.
|
Copy `.env.example` to `.env` and edit.
|
||||||
|
|
||||||
### docker-compose.yml
|
### Docker Compose
|
||||||
|
|
||||||
|
Create `docker-compose.yml` and edit:
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
services:
|
services:
|
||||||
@@ -49,17 +52,87 @@ services:
|
|||||||
start_period: 10s
|
start_period: 10s
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Start the service:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ sudo docker compose up --build -d
|
$ sudo docker compose up --build -d
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Set up nginx / a reverse proxy and point it to the web app.
|
||||||
|
|
||||||
|
Make sure it allows WebSocket connections through, for example:
|
||||||
|
|
||||||
|
```
|
||||||
|
server {
|
||||||
|
root /var/www/html;
|
||||||
|
index index.html index.htm;
|
||||||
|
server_name upload.example.com;
|
||||||
|
|
||||||
|
listen 80;
|
||||||
|
|
||||||
|
client_max_body_size 100M;
|
||||||
|
|
||||||
|
location / {
|
||||||
|
proxy_pass http://127.0.0.1:8080/;
|
||||||
|
proxy_set_header Host $http_host;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
proxy_set_header X-Forwarded-Proto $scheme;
|
||||||
|
|
||||||
|
# websockets
|
||||||
|
proxy_http_version 1.1;
|
||||||
|
proxy_set_header Upgrade $http_upgrade;
|
||||||
|
proxy_set_header Connection "upgrade";
|
||||||
|
proxy_redirect off;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Then restart nginx and set up HTTPS:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ sudo service nginx restart
|
||||||
|
$ sudo certbot --nginx
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### Config Changes
|
||||||
|
|
||||||
|
If you change the `.env` file config, simply run:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ sudo docker compose down
|
||||||
|
$ sudo docker compose up --build -d
|
||||||
|
```
|
||||||
|
|
||||||
|
### Updating
|
||||||
|
|
||||||
|
To update the code:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ sudo docker compose down
|
||||||
|
$ git pull --rebase
|
||||||
|
$ sudo docker compose up --build -d
|
||||||
|
```
|
||||||
|
|
||||||
|
### Telegram Bot
|
||||||
|
|
||||||
|
An optional Telegram bot can send you notifications when uploads complete. This is useful to see if random people are filling your disk up.
|
||||||
|
|
||||||
|
To create a bot, message @BotFather on Telegram. Come up with a name and username. Botfather will then send you an API key you can paste into the `.env` config directly.
|
||||||
|
|
||||||
|
Next you'll need to find your own Telegram user ID. You can message @userinfobot and it will reply with your ID. Beware of impersonator bots (they have the name "userinfobot" but a different username).
|
||||||
|
|
||||||
|
Then message the bot you just created "/start" so that it's able to interact with you.
|
||||||
|
|
||||||
|
|
||||||
### Chunked Uploads
|
### Chunked Uploads
|
||||||
|
|
||||||
- Chunked uploads are enabled by default. Uses setting `CHUNKED_UPLOADS_ENABLED=true`.
|
- Chunked uploads are enabled by default. Uses setting `CHUNKED_UPLOADS_ENABLED=true`.
|
||||||
- Configure chunk size with `CHUNK_SIZE_MB` (default: `50`). The client only uses chunked mode for files larger than this.
|
- Configure chunk size with `CHUNK_SIZE_MB` (default: `50`). The client only uses chunked mode for files larger than this.
|
||||||
- Intended to bypass upstream proxy limits (e.g., 100MB) while preserving duplicate checks, EXIF timestamps, album add, and per‑item progress via WebSocket.
|
- Intended to bypass upstream proxy limits (e.g., 100MB) while preserving duplicate checks, EXIF timestamps, album add, and per‑item progress via WebSocket.
|
||||||
|
|
||||||
## Developtment
|
## Development
|
||||||
|
|
||||||
### Architecture
|
### Architecture
|
||||||
|
|
||||||
@@ -70,11 +143,22 @@ $ sudo docker compose up --build -d
|
|||||||
- WebSocket `/ws` pushes per‑item progress to the current browser session only.
|
- WebSocket `/ws` pushes per‑item progress to the current browser session only.
|
||||||
- **Persistence:** A local SQLite database (`state.db`) prevents re‑uploads across sessions. Uploaded files are stored in `/data/uploads`.
|
- **Persistence:** A local SQLite database (`state.db`) prevents re‑uploads across sessions. Uploaded files are stored in `/data/uploads`.
|
||||||
|
|
||||||
### Requirements
|
### Setup
|
||||||
|
|
||||||
- **Python** 3.11
|
Requires Python 3.11+.
|
||||||
|
|
||||||
### Local dev quickstart
|
Create a venv, activate it, and install:
|
||||||
|
|
||||||
|
```text
|
||||||
|
$ virtualenv -p python3 env
|
||||||
|
$ source env/bin/activate
|
||||||
|
(env) $ pip install -r requirements.txt
|
||||||
|
```
|
||||||
|
|
||||||
|
```text
|
||||||
|
(env) $ cp .env.example .env
|
||||||
|
(env) $ vim .env
|
||||||
|
```
|
||||||
|
|
||||||
Run with live reload:
|
Run with live reload:
|
||||||
|
|
||||||
@@ -82,54 +166,30 @@ Run with live reload:
|
|||||||
python main.py
|
python main.py
|
||||||
```
|
```
|
||||||
|
|
||||||
The backend contains docstrings so you can generate docs later if desired.
|
|
||||||
|
|
||||||
### Dev Configuration (.env)
|
|
||||||
|
|
||||||
```ini
|
|
||||||
# Server (dev only)
|
|
||||||
HOST=0.0.0.0
|
|
||||||
PORT=8080
|
|
||||||
|
|
||||||
# Public uploader page (optional) — disabled by default
|
|
||||||
PUBLIC_UPLOAD_PAGE_ENABLED=TRUE
|
|
||||||
|
|
||||||
# Local dedupe cache (SQLite)
|
|
||||||
STATE_DB=./data/state.db
|
|
||||||
|
|
||||||
# Telegram Bot for notifications (optional)
|
|
||||||
#TELEGRAM_BOT_API_KEY=
|
|
||||||
#TELEGRAM_BOT_OWNER_ID=
|
|
||||||
|
|
||||||
# Base URL for generating absolute invite links (recommended for production)
|
|
||||||
# e.g., PUBLIC_BASE_URL=https://photos.example.com
|
|
||||||
#PUBLIC_BASE_URL=
|
|
||||||
|
|
||||||
# Session and security
|
|
||||||
SESSION_SECRET=SET-A-STRONG-RANDOM-VALUE
|
|
||||||
LOG_LEVEL=DEBUG
|
|
||||||
|
|
||||||
# Chunked uploads (optional)
|
|
||||||
CHUNKED_UPLOADS_ENABLED=true
|
|
||||||
CHUNK_SIZE_MB=95
|
|
||||||
|
|
||||||
```
|
|
||||||
|
|
||||||
You can keep a checked‑in `/.env.example` with the keys above for onboarding.
|
|
||||||
|
|
||||||
### How it works
|
### How it works
|
||||||
|
|
||||||
1. **Queue** – Files selected in the browser are queued; each gets a client-side ID.
|
1. **Queue** - Files selected in the browser are queued; each gets a client-side ID.
|
||||||
2. **De-dupe (local)** – Server computes **SHA‑1** and checks `state.db`. If seen, marks as **duplicate**.
|
2. **De-dupe (local)** - Server computes **SHA‑1** and checks `state.db`. If seen, marks as **duplicate**.
|
||||||
3. **Save** – The file is saved to the local filesystem under `./data/uploads`.
|
3. **Save** - The file is saved to the local filesystem under `./data/uploads`.
|
||||||
4. **Album** – If an album is specified via an invite link, or a folder name is provided on the public page, the file is saved into a corresponding subdirectory. Client-side folder structure is also preserved.
|
4. **Album** - If an album is specified via an invite link, or a folder name is provided on the public page, the file is saved into a corresponding subdirectory. Client-side folder structure is also preserved.
|
||||||
5. **Progress** – Backend streams progress via WebSocket to the same session.
|
5. **Progress** - Backend streams progress via WebSocket to the same session.
|
||||||
6. **Privacy** – The UI shows only the current session's items. It does not provide a way to browse saved files.
|
6. **Privacy** - The UI shows only the current session's items. It does not provide a way to browse saved files.
|
||||||
|
|
||||||
### Security notes
|
### Security notes
|
||||||
|
|
||||||
- The menu and invite creation are behind login. Logout clears the session.
|
- The menu and invite creation are behind login. Logout clears the session.
|
||||||
- Invite links are public by URL; share only with intended recipients.
|
- Invite links are public by URL; share only with intended recipients.
|
||||||
- The default uploader page at `/` is disabled unless `PUBLIC_UPLOAD_PAGE_ENABLED=true`.
|
- The public uploader page at `/` is enabled unless disabled with `PUBLIC_UPLOAD_PAGE_ENABLED=false`.
|
||||||
- No browsing of uploaded media; only ephemeral session state is shown.
|
- No browsing of uploaded media; only ephemeral session state is shown.
|
||||||
- Run behind HTTPS with a reverse proxy and restrict CORS to your domain(s).
|
- Run behind HTTPS with a reverse proxy and restrict CORS to your domain(s).
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
This program is free and open-source software licensed under the MIT License. Please see the `LICENSE` file for details.
|
||||||
|
|
||||||
|
That means you have the right to study, change, and distribute the software and source code to anyone and for any purpose. You deserve these rights.
|
||||||
|
|
||||||
|
## Acknowledgements
|
||||||
|
|
||||||
|
This project was forked from "Immich Drop Uploader" by Simon Adams: https://github.com/Nasogaa/immich-drop
|
||||||
|
|
||||||
|
|||||||
12
app/app.py
12
app/app.py
@@ -41,16 +41,20 @@ from app.config import Settings, load_settings
|
|||||||
|
|
||||||
# ---- App & static ----
|
# ---- App & static ----
|
||||||
app = FastAPI(title="Immich Drop Uploader (Python)")
|
app = FastAPI(title="Immich Drop Uploader (Python)")
|
||||||
|
# Global settings (read-only at runtime)
|
||||||
|
SETTINGS: Settings = load_settings()
|
||||||
|
|
||||||
|
# CORS
|
||||||
|
origins = ["*"]
|
||||||
|
if SETTINGS.public_base_url:
|
||||||
|
origins = [SETTINGS.public_base_url.strip().rstrip('/')]
|
||||||
app.add_middleware(
|
app.add_middleware(
|
||||||
CORSMiddleware,
|
CORSMiddleware,
|
||||||
allow_origins=["*"],
|
allow_origins=origins,
|
||||||
allow_credentials=True,
|
allow_credentials=True,
|
||||||
allow_methods=["*"],
|
allow_methods=["*"],
|
||||||
allow_headers=["*"],
|
allow_headers=["*"],
|
||||||
)
|
)
|
||||||
|
|
||||||
# Global settings (read-only at runtime)
|
|
||||||
SETTINGS: Settings = load_settings()
|
|
||||||
_public_uploads_enabled_runtime = SETTINGS.public_upload_page_enabled
|
_public_uploads_enabled_runtime = SETTINGS.public_upload_page_enabled
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -17,13 +17,13 @@ class Settings:
|
|||||||
"""App settings loaded from environment variables (.env)."""
|
"""App settings loaded from environment variables (.env)."""
|
||||||
admin_password: str
|
admin_password: str
|
||||||
max_concurrent: int
|
max_concurrent: int
|
||||||
public_upload_page_enabled: bool = False
|
public_upload_page_enabled: bool = True
|
||||||
public_base_url: str = ""
|
public_base_url: str = ""
|
||||||
state_db: str = ""
|
state_db: str = ""
|
||||||
session_secret: str = ""
|
session_secret: str = ""
|
||||||
log_level: str = "INFO"
|
log_level: str = "INFO"
|
||||||
chunked_uploads_enabled: bool = False
|
chunked_uploads_enabled: bool = True
|
||||||
chunk_size_mb: int = 95
|
chunk_size_mb: int = 50
|
||||||
timezone: str = "UTC"
|
timezone: str = "UTC"
|
||||||
telegram_bot_api_key: str = ""
|
telegram_bot_api_key: str = ""
|
||||||
telegram_bot_owner_id: str = ""
|
telegram_bot_owner_id: str = ""
|
||||||
@@ -48,7 +48,8 @@ def load_settings() -> Settings:
|
|||||||
load_dotenv()
|
load_dotenv()
|
||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
admin_password = os.getenv("ADMIN_PASSWORD", "admin") # Default for convenience, should be changed
|
|
||||||
|
admin_password = os.getenv("ADMIN_PASSWORD", "test123") # Default for convenience, should be changed
|
||||||
if not admin_password.startswith("pbkdf2_sha256-"):
|
if not admin_password.startswith("pbkdf2_sha256-"):
|
||||||
print("="*60)
|
print("="*60)
|
||||||
print("WARNING: ADMIN_PASSWORD is in plaintext.")
|
print("WARNING: ADMIN_PASSWORD is in plaintext.")
|
||||||
@@ -57,27 +58,34 @@ def load_settings() -> Settings:
|
|||||||
if hashed_pw:
|
if hashed_pw:
|
||||||
print(f"ADMIN_PASSWORD={hashed_pw}")
|
print(f"ADMIN_PASSWORD={hashed_pw}")
|
||||||
print("="*60)
|
print("="*60)
|
||||||
# Safe defaults: disable public uploader and invites unless explicitly enabled
|
|
||||||
def as_bool(v: str, default: bool = False) -> bool:
|
def as_bool(v: str, default: bool = False) -> bool:
|
||||||
if v is None:
|
if v is None:
|
||||||
return default
|
return default
|
||||||
return str(v).strip().lower() in {"1","true","yes","on"}
|
return str(v).strip().lower() in {"1","true","yes","on"}
|
||||||
public_upload = as_bool(os.getenv("PUBLIC_UPLOAD_PAGE_ENABLED", "false"), False)
|
|
||||||
|
public_upload = as_bool(os.getenv("PUBLIC_UPLOAD_PAGE_ENABLED", "false"), True)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
maxc = int(os.getenv("MAX_CONCURRENT", "3"))
|
maxc = int(os.getenv("MAX_CONCURRENT", "3"))
|
||||||
except ValueError:
|
except ValueError:
|
||||||
maxc = 3
|
maxc = 3
|
||||||
state_db = os.getenv("STATE_DB", "/data/state.db")
|
|
||||||
|
state_db = os.getenv("STATE_DB", "./data/state.db")
|
||||||
session_secret = os.getenv("SESSION_SECRET") or secrets.token_hex(32)
|
session_secret = os.getenv("SESSION_SECRET") or secrets.token_hex(32)
|
||||||
log_level = os.getenv("LOG_LEVEL", "INFO").upper()
|
log_level = os.getenv("LOG_LEVEL", "INFO").upper()
|
||||||
chunked_uploads_enabled = as_bool(os.getenv("CHUNKED_UPLOADS_ENABLED", "false"), False)
|
|
||||||
|
chunked_uploads_enabled = as_bool(os.getenv("CHUNKED_UPLOADS_ENABLED", "false"), True)
|
||||||
try:
|
try:
|
||||||
chunk_size_mb = int(os.getenv("CHUNK_SIZE_MB", "95"))
|
chunk_size_mb = int(os.getenv("CHUNK_SIZE_MB", "50"))
|
||||||
except ValueError:
|
except ValueError:
|
||||||
chunk_size_mb = 95
|
chunk_size_mb = 50
|
||||||
|
|
||||||
timezone = os.getenv("TIMEZONE", "UTC")
|
timezone = os.getenv("TIMEZONE", "UTC")
|
||||||
|
|
||||||
telegram_bot_api_key = os.getenv("TELEGRAM_BOT_API_KEY", "")
|
telegram_bot_api_key = os.getenv("TELEGRAM_BOT_API_KEY", "")
|
||||||
telegram_bot_owner_id = os.getenv("TELEGRAM_BOT_OWNER_ID", "")
|
telegram_bot_owner_id = os.getenv("TELEGRAM_BOT_OWNER_ID", "")
|
||||||
|
|
||||||
return Settings(
|
return Settings(
|
||||||
admin_password=admin_password,
|
admin_password=admin_password,
|
||||||
max_concurrent=maxc,
|
max_concurrent=maxc,
|
||||||
|
|||||||
BIN
media/admin-page.png
Normal file
BIN
media/admin-page.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 116 KiB |
BIN
media/after-uploading.png
Normal file
BIN
media/after-uploading.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 69 KiB |
BIN
media/invite-page.png
Normal file
BIN
media/invite-page.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 61 KiB |
BIN
media/public-uploader.png
Normal file
BIN
media/public-uploader.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 29 KiB |
BIN
media/telegram-bot.png
Normal file
BIN
media/telegram-bot.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 63 KiB |
BIN
screenshot.png
BIN
screenshot.png
Binary file not shown.
|
Before Width: | Height: | Size: 160 KiB |
21
screenshots.md
Normal file
21
screenshots.md
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
# Screenshots
|
||||||
|
|
||||||
|
## Public upload page
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## After uploading files
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## Admin page
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## Invite link (with password)
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## Telegram bot
|
||||||
|
|
||||||
|

|
||||||
Reference in New Issue
Block a user