Share code snippets and images with syntax highlighting, privacy options, and self-destructing content
- π Privacy-Focused: Password protection and burn-after-reading options
- π Syntax Highlighting: Automatic language detection for multiple programming languages
- β±οΈ Expiration Control: Set pastes to expire after 1 hour, 1 day, 7 days, or 30 days
- ποΈ Content Compression: Efficient storage with Brotli compression
- π€ AI-Generated Metadata: Automatic title and description generation using Google's Gemini API
- π Real-time Updates: Server-Sent Events (SSE) for live metadata generation status
- π± Responsive Design: Works seamlessly on desktop and mobile devices
- π Dark Mode: Automatic theme based on system preferences
- π No Registration: Create and share pastes without an account
- π· Image Support: Upload and share images with automatic compression
- π EXIF Data Extraction: View camera, photo, and location metadata from images
- π Format Conversion: Convert images between formats (JPEG, PNG, WebP, etc.)
- Contributing Guidelines - How to contribute to the project
- Development Guide - Development workflow and coding standards
- Code of Conduct - Guidelines for community participation
- Security Policy - Security practices and vulnerability reporting
- Changelog - Record of all notable changes
- License - MIT License
- Frontend: Next.js 15, React 19, TypeScript, Tailwind CSS, Shadcn UI
- Backend: Next.js API Routes with TypeScript
- Database: PostgreSQL with Prisma ORM
- Code Editor: CodeMirror for syntax highlighting and editing
- Validation: Zod for robust schema validation
- Compression: Brotli algorithm for efficient content storage
- Authentication: Password hashing with bcrypt for protected pastes
- AI: Google's Gemini API for generating titles and descriptions
- Real-time Updates: Server-Sent Events (SSE) for live status updates
- Image Processing: Sharp for image compression and format conversion
- EXIF Extraction: exif-reader for extracting image metadata
- Storage: Cloudflare R2 (S3-compatible) for image storage
Dustebin uses Google's Gemini API to automatically generate titles and descriptions for pastes:
- Asynchronous Processing: Metadata is generated in the background after paste creation
- Real-time Updates: Status updates are delivered via Server-Sent Events (SSE)
- Language-Aware: Generates context-appropriate titles and descriptions based on the code language
- Fallback Handling: Graceful degradation if the AI service is unavailable
The application uses SSE to provide real-time updates on metadata generation:
- Live Status Updates: Clients receive updates as metadata generation progresses
- Efficient Communication: One-way server-to-client communication without polling
- Graceful Timeouts: Automatic connection management with appropriate timeouts
- Error Handling: Robust error handling for failed connections or generation issues
- Node.js 20+ (see .nvmrc)
- pnpm (or npm/yarn)
- PostgreSQL database
- Clone the repository
git clone https://github.com/lukabudik/dustebin.git
cd dustebin
- Install dependencies
pnpm install
- Set up environment variables
cp .env.example .env
# Edit .env with your database credentials and API keys
Required environment variables:
DATABASE_URL
: PostgreSQL connection stringCLEANUP_API_KEY
: API key for the admin cleanup endpointGEMINI_API_KEY
: Google Gemini API key for AI-generated titles and descriptionsR2_ACCOUNT_ID
: Cloudflare account ID for R2 storageR2_ACCESS_KEY_ID
: Access key ID for R2 storageR2_SECRET_ACCESS_KEY
: Secret access key for R2 storageR2_BUCKET_NAME
: Bucket name for R2 storageR2_PUBLIC_URL
: Public URL for R2 storage
- Set up the database
# Run the SQL script to create the database
psql -U username -f prisma/init.sql
# Apply migrations
npx prisma migrate dev
- Start the development server
pnpm dev
- Open http://localhost:3000 in your browser
For development with Docker:
# Start the development environment
docker-compose up
For production with Docker:
# Build the Docker image
docker build -t dustebin .
# Run the container
docker run -p 3000:3000 --env-file .env dustebin
For production deployment:
# Build the application
pnpm build
# Start the production server
pnpm start
Make sure to set up the environment variables in your production environment.
Dustebin allows users to share images with the same privacy and expiration features as text pastes:
- Image Upload: Upload images up to 50MB in size
- Format Support: JPEG, PNG, WebP, GIF, HEIC, AVIF, TIFF, and BMP
- High Quality Preservation: Images maintain their original dimensions with high-quality compression
- Format Conversion: Convert images between formats for download
- EXIF Data Extraction: View camera information, photo details, and location data from images
- Privacy Protection: EXIF data is only visible to those with access to the paste
The EXIF data viewer provides detailed information about images:
- Camera Information: Make, model, software, and dimensions
- Photo Details: Date taken, exposure time, aperture, ISO, and focal length
- Location Data: GPS coordinates with Google Maps integration (if available)
- Privacy Controls: EXIF data is only shown when explicitly requested
Images are processed to ensure optimal quality:
- Original Dimensions: Images maintain their original dimensions without automatic resizing
- High-Quality Compression: Images are compressed with high-quality settings (90% quality)
- Format Preservation: Original format is preserved while offering conversion options
- Efficient Storage: Images are stored in Cloudflare R2 with appropriate caching headers
Dustebin automatically removes expired pastes through multiple mechanisms:
When a paste is accessed and found to be expired, it is automatically deleted.
For production environments, set up a scheduled job to call the cleanup endpoint:
# Example cron job (runs every hour)
0 * * * * curl -X POST https://dustebin.com/api/admin/cleanup -H "Authorization: Bearer YOUR_API_KEY"
You can configure the API key by setting the CLEANUP_API_KEY
environment variable.
A GitHub Actions workflow is included to automatically run the cleanup process:
- Add your deployment URL as a GitHub secret named
DUSTEBIN_URL
- Add your API key as a GitHub secret named
CLEANUP_API_KEY
- Enable GitHub Actions in your repository
Pastes can be protected with passwords. The passwords are:
- Never stored in plain text
- Hashed using bcrypt with appropriate salt rounds
- Verified server-side before content is delivered
The "Burn After Reading" feature:
- Creates pastes that are deleted from the database after being viewed once
- Requires explicit confirmation before viewing to prevent accidental deletion
- Provides a warning to users that the content will be permanently deleted
The application includes rate limiting to prevent abuse:
- Maximum of 60 requests per minute per IP address
- Maximum of 20 paste creations per hour per IP address
dustebin/
βββ prisma/ # Database schema and migrations
βββ public/ # Static assets
βββ src/
β βββ app/ # Next.js App Router
β β βββ api/ # API routes
β β βββ [id]/ # Paste view page
β β βββ ... # Other pages
β βββ components/ # React components
β βββ lib/ # Utility functions and services
βββ .github/ # GitHub configuration
βββ ... # Configuration files
-
POST /api/pastes
- Create a new paste- Body:
{ content, language, expiration, password?, image?, pasteType?, originalFormat? }
- Returns:
{ id, language, createdAt, expiresAt, hasPassword, hasImage?, imageUrl?, pasteType? }
- Body:
-
GET /api/pastes/[id]
- Get a paste by ID- Headers:
X-Password
(optional) - Query:
password
(optional) - Returns: Paste object with content or image URL
- Headers:
-
GET /api/pastes/[id]/raw
- Get raw paste content- Query:
password
(optional),confirm
(for burn-after-read) - Returns: Plain text content
- Query:
-
POST /api/pastes/[id]/burn
- Burn a paste after reading- Returns:
{ success: true, message: 'Paste has been burned' }
- Returns:
-
GET /api/pastes/[id]/metadata
- Get real-time metadata generation status via SSE- Returns: Server-Sent Events with status updates (
pending
,completed
,failed
, etc.) - When completed, includes generated title and description
- Returns: Server-Sent Events with status updates (
-
GET /api/pastes/[id]/image
- Get the image for a paste- Query:
password
(optional) - Returns: Image file
- Query:
-
GET /api/pastes/[id]/download
- Download an image in a specific format- Query:
format
(original, jpg, png, webp),password
(optional) - Returns: Image file in the requested format
- Query:
-
GET /api/pastes/[id]/formats
- Get available image formats and sizes- Returns:
{ formats: [{ id, name, size, extension }] }
- Returns:
GET /api/languages
- Get supported languages- Returns:
{ languages: [{ value, label }] }
- Returns:
POST /api/admin/cleanup
- Clean up expired pastes (admin only)- Headers:
Authorization: Bearer YOUR_API_KEY
- Returns:
{ success: true, deleted: count, timestamp }
- Headers:
We welcome contributions! Please see our Contributing Guidelines for details on how to get started, coding standards, and more.
This project is licensed under the MIT License - see the LICENSE file for details.