generated from nhcarrigan/template
983b78b0e9
## Summary This PR includes multiple feature additions and fixes to improve the library application: ### 🎨 Base64 Image Upload Support - Fixed Fastify body limit (1MB → 10MB) to accommodate base64-encoded images - Corrected base64 size calculation and validation logic - Improved error handling with proper 400 status codes and helpful messages - Removed duplicate validation that was blocking uploads - Users can now upload cover images up to 5MB (decoded size) ### 📝 Reusable Form Components - Created 6 form components: `GameForm`, `BookForm`, `MusicForm`, `ShowForm`, `MangaForm`, `ArtForm` - All forms support both 'add' and 'edit' modes with pre-population - Integrated inline editing into all detail views (edit/delete buttons) - Enhanced admin suggestions workflow with full forms instead of basic modals - Added scroll-to-top when clicking edit in list views for better UX ### 🖼️ Default Cover Image - Added beautiful library reading image as default cover for all media types - Fixed static asset serving to use correct MIME types - Updated all 12 components (6 list views + 6 detail views) to always show images ### 🔒 Tiered Rate Limiting - Unauthenticated users: 100 requests/minute - Authenticated users: 500 requests/minute (5x more lenient) - Admin users: No rate limits (complete bypass via allowList) ### 🎮 Discord Integration - Auto-assign library member role to users in NHCarrigan Discord server - Checks server membership on every login - Only assigns role if user is in server and doesn't have it yet - Graceful error handling without blocking login - Similar pattern to badge refresh flow ### 📚 Documentation - Added comprehensive CLAUDE.md with: - Project structure and tech stack - Development workflow and commands - Database schema documentation - Authentication flow details - Security features - Code style conventions - Common gotchas and solutions ## Test Plan - [x] Base64 image uploads work for cover images up to 5MB - [x] Helpful error messages appear for validation failures - [x] Edit/delete buttons appear on all detail views for admin users - [x] Inline edit forms display and save correctly - [x] Admin suggestions workflow uses full forms for all media types - [x] Scroll-to-top works when editing from list views - [x] Default cover image displays when no cover is provided - [x] Static assets serve with correct MIME types - [x] Rate limiting works correctly for different user types - [x] Discord role assignment works on login - [x] All builds pass without errors - [x] No TypeScript errors ## Related Issues Closes #65 - Base64 image upload issue ✨ This pull request was created with help from Hikari~ 🌸 Reviewed-on: #66 Co-authored-by: Hikari <hikari@nhcarrigan.com> Co-committed-by: Hikari <hikari@nhcarrigan.com>
58 lines
1.6 KiB
TypeScript
58 lines
1.6 KiB
TypeScript
import Fastify from 'fastify';
|
|
import { app } from './app/app';
|
|
import { logger } from './app/utils/logger';
|
|
|
|
const host = process.env.HOST ?? 'localhost';
|
|
const port = process.env.PORT ? Number(process.env.PORT) : 12321;
|
|
|
|
// Global error handlers
|
|
process.on('uncaughtException', (error: Error) => {
|
|
void logger.error('Uncaught Exception', error);
|
|
process.exit(1);
|
|
});
|
|
|
|
process.on('unhandledRejection', (reason: unknown) => {
|
|
const error = reason instanceof Error ? reason : new Error(String(reason));
|
|
void logger.error('Unhandled Rejection', error);
|
|
process.exit(1);
|
|
});
|
|
|
|
process.on('warning', (warning: Error) => {
|
|
void logger.log('warn', `Process Warning: ${warning.name} - ${warning.message}`);
|
|
});
|
|
|
|
process.on('SIGTERM', () => {
|
|
void logger.log('info', 'SIGTERM signal received: closing HTTP server');
|
|
server.close(() => {
|
|
void logger.log('info', 'HTTP server closed');
|
|
process.exit(0);
|
|
});
|
|
});
|
|
|
|
process.on('SIGINT', () => {
|
|
void logger.log('info', 'SIGINT signal received: closing HTTP server');
|
|
server.close(() => {
|
|
void logger.log('info', 'HTTP server closed');
|
|
process.exit(0);
|
|
});
|
|
});
|
|
|
|
// Instantiate Fastify with some config
|
|
const server = Fastify({
|
|
logger: true,
|
|
bodyLimit: 10485760, // 10MB max body size (to accommodate base64-encoded images)
|
|
});
|
|
|
|
// Register your application as a normal plugin.
|
|
server.register(app);
|
|
|
|
// Start listening.
|
|
server.listen({ port, host }, (err) => {
|
|
if (err) {
|
|
server.log.error(err);
|
|
process.exit(1);
|
|
} else {
|
|
void logger.log('info', `Server ready at http://${host}:${port}`);
|
|
}
|
|
});
|