toggleChannelActive only flipped isActive but never created the AccountChannelMap READER link needed by the worker. Channels enabled via the toggle (rather than the channel picker) were invisible to the scanner. Now auto-creates READER links for all active authenticated accounts when a SOURCE channel is enabled. Also ran a one-time DB fix to backfill READER links for the 14 active channels that were missing them. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
Dragon's Stash
A self-hosted inventory management system for 3D printing filament, SLA resin, and miniature paints — with an integrated Telegram archive worker that ingests, indexes, and redistributes archive files. Built with a dark, data-dense UI inspired by Spoolman.
Features
Inventory Management
- Filament tracking with spool weight, material type, color swatches, and usage logging
- SLA resin management with bottle sizes, resin types, and remaining volume tracking
- Miniature paint inventory with product lines, finishes, and volume tracking
- Dashboard with inventory stats, low-stock alerts, and recent activity
- Vendor and location management to organize your supplies
- Usage logging to track consumption over time
- Low-stock alerts with configurable threshold percentage
- Dark theme optimized for workshop environments
- Role-based auth with admin and user roles
Telegram Archive Worker
- Channel scanning — monitors configured Telegram channels (including forum topics) for archive files (ZIP, RAR, 7z)
- Multipart detection — automatically groups related multipart archives (
.part01.rar,.z01,.001, etc.) - Content indexing — extracts file listings from archives and stores them in the database
- Destination upload — re-uploads processed archives to a configured destination channel
- Byte-level splitting — splits files exceeding Telegram's 2GB limit into uploadable chunks
- Full repack — concatenates and re-splits multipart sets where any single part exceeds 2GB
- Progress tracking — resumes from the last successfully processed message on each run
- Upload verification — confirms files reached the destination before marking them complete
- Preview matching — associates photo messages with their corresponding archive sets
Telegram Bot
- Direct delivery — send any indexed package to a linked Telegram account with one click from the UI
- Account linking — users link their Telegram account via a one-time code from Settings
- Package search — search or browse indexed packages directly from conversation with the bot
- Subscription notifications — subscribe to keyword patterns and get notified when matching packages arrive
- Automatic forwarding — the bot copies files from the destination channel, no manual download needed
Tech Stack
- Framework: Next.js 16 (App Router)
- Language: TypeScript (strict mode)
- Database: PostgreSQL with Prisma ORM
- Auth: Auth.js v5 (credentials + GitHub OAuth)
- UI: Tailwind CSS, shadcn/ui, Lucide icons
- Tables: TanStack Table v8 with server-side pagination
- Validation: Zod v4 + React Hook Form
- Worker: Node.js + TDLib (via tdl)
- Bot: Node.js + TDLib (bot token auth)
- Archive handling: unrar, zlib
Quick Start
Prerequisites
- Node.js 20+
- PostgreSQL 16+ (or Docker)
- Telegram API credentials (for the worker — get from my.telegram.org/apps)
Development Setup
- Clone the repository:
git clone https://github.com/your-username/dragons-stash.git
cd dragons-stash
- Install dependencies:
npm install
- Start a PostgreSQL database (using Docker):
docker compose -f docker-compose.dev.yml up -d db
- Copy the environment file and update values:
cp .env.example .env.local
- Run database migrations and seed:
npx prisma migrate dev # Run migrations
npx prisma db seed # Seed with sample data (admin/user accounts + inventory)
- Start the development server:
npm run dev
- Open http://localhost:3000 and log in:
- Admin: admin@dragonsstash.local / password123
- User: user@dragonsstash.local / password123
Running the Worker in Development
To also run the Telegram worker alongside the dev database:
docker compose -f docker-compose.dev.yml up -d
This starts both the PostgreSQL database and the worker container. The worker reads TELEGRAM_API_ID and TELEGRAM_API_HASH from your .env.local file.
Docker Deployment
Full Stack (App + Worker + Database)
Run the entire application from Docker:
cp .env.example .env
# Edit .env — set AUTH_SECRET (required)
docker compose up -d
The app will be available at http://localhost:3000.
Adding the Telegram Bot
The worker starts by default with docker compose up. The bot runs as an optional profile:
# App + DB + Worker + Bot (also needs BOT_TOKEN in .env)
docker compose --profile full up -d
# Or just the bot (alongside app + db + worker)
docker compose --profile bot up -d
Tip: Create a bot token via @BotFather on Telegram and set
BOT_TOKENin.env. Get Telegram API credentials from my.telegram.org/apps.
Seeding the Database
To seed the database with sample data on first run:
SEED_DATABASE=true docker compose up -d
This creates default admin/user accounts and sample inventory data. The seed runs once during the app container's entrypoint (before the Next.js server starts). On subsequent runs without SEED_DATABASE=true, seeding is skipped automatically.
You can also seed manually at any time:
npx prisma db seed
Development Mode (DB + Worker Only)
If you prefer to run the Next.js app locally with hot reload:
docker compose -f docker-compose.dev.yml up -d # Start DB + worker
npm run dev # Start Next.js locally
Rebuilding After Code Changes
docker compose build && docker compose up -d --force-recreate
To rebuild only the worker:
docker compose build worker && docker compose up -d worker --force-recreate
Viewing Logs
docker compose logs -f worker # Worker logs
docker compose logs -f bot # Bot logs
docker compose logs -f app # App logs
docker compose logs -f db # Database logs
Project Structure
src/
app/
(auth)/ # Login/Register pages
(app)/ # Authenticated app pages
dashboard/ # Overview stats
filaments/ # Filament CRUD
resins/ # Resin CRUD
paints/ # Paint CRUD
vendors/ # Vendor management
locations/ # Location management
settings/ # User preferences + Telegram link
stls/ # STL package browser
telegram/ # Telegram admin (accounts, channels, bot sends)
api/
auth/ # NextAuth API routes
health/ # Health check endpoint
telegram/bot/ # Bot send API endpoints
components/
layout/ # Sidebar, header, navigation
shared/ # Reusable data table components
ui/ # shadcn/ui components
data/ # Prisma query functions
hooks/ # React hooks
lib/ # Auth config, Prisma client, constants
schemas/ # Zod validation schemas
types/ # TypeScript type definitions
worker/
src/
archive/ # Archive detection, multipart grouping, byte-level splitting
db/ # Prisma queries for packages, progress tracking
preview/ # Preview image matching
tdlib/ # TDLib client, channel scanning, topic/forum handling
upload/ # Telegram upload logic
util/ # Config, logger
worker.ts # Main processing pipeline
index.ts # Entry point + scheduler
bot/
src/
commands.ts # Bot command handlers (/search, /link, /subscribe, etc.)
send-listener.ts # pg_notify listener for send requests + subscriptions
tdlib/ # TDLib client with bot token auth
db/ # Database queries for links, packages, subscriptions
util/ # Config, logger
index.ts # Entry point
prisma/
schema.prisma # Database schema
seed.ts # Seed data
Configuration
Environment variables (see .env.example):
Application
| Variable | Description | Default |
|---|---|---|
DATABASE_URL |
PostgreSQL connection string | Required |
AUTH_SECRET |
NextAuth secret key | Required |
AUTH_TRUST_HOST |
Trust the host header | true |
AUTH_GITHUB_ID |
GitHub OAuth client ID | Optional |
AUTH_GITHUB_SECRET |
GitHub OAuth client secret | Optional |
NEXT_PUBLIC_APP_URL |
Public application URL | http://localhost:3000 |
SEED_DATABASE |
Seed the database on app container start | false |
Telegram Worker
| Variable | Description | Default |
|---|---|---|
TELEGRAM_API_ID |
Telegram API ID (from my.telegram.org) | Required |
TELEGRAM_API_HASH |
Telegram API hash | Required |
WORKER_INTERVAL_MINUTES |
Scan interval in minutes | 60 |
WORKER_TEMP_DIR |
Temp directory for downloads | /tmp/zips |
TDLIB_STATE_DIR |
TDLib session state persistence directory | /data/tdlib |
WORKER_MAX_ZIP_SIZE_MB |
Max archive size to process (MB) | 4096 |
MULTIPART_TIMEOUT_HOURS |
Max time span for multipart set parts (0 = no limit) | 0 |
LOG_LEVEL |
Worker log level (debug, info, warn, error) |
info |
Telegram Bot
| Variable | Description | Default |
|---|---|---|
BOT_TOKEN |
Bot token from @BotFather | Optional (bot disabled if unset) |
TELEGRAM_API_ID |
Same API ID as worker | Required (if bot enabled) |
TELEGRAM_API_HASH |
Same API hash as worker | Required (if bot enabled) |
BOT_TDLIB_STATE_DIR |
TDLib state directory for bot | /data/tdlib_bot |
LOG_LEVEL |
Bot log level | info |
Health Check
The application exposes a health check endpoint at /api/health that verifies database connectivity.
curl http://localhost:3000/api/health
Contributing
- Fork the repository
- Create a feature branch (
git checkout -b feature/amazing-feature) - Commit your changes (
git commit -m 'Add amazing feature') - Push to the branch (
git push origin feature/amazing-feature) - Open a Pull Request
License
This project is licensed under the MIT License - see the LICENSE file for details.