feat: implement backend API for Pantree Phase 1 MVP
- Project setup: package.json, tsconfig.json, jest.config.js, .env.example - Config: env.ts, constants.ts (ALLOWED_UNITS, bcrypt rounds, JWT, deletion windows) - DB: Knex connection, knexfile, migrations 001-006 (users, password_reset_tokens, pantry_items, recipes+recipe_ingredients, shopping_lists+items, deleted_records) - Seeds: 10 seeded recipes with full ingredient lists - Middleware: JWT authMiddleware (validates token + user existence), errorHandler - Utils: Pino logger, JWT sign helper, Zod validators for all request shapes - Services: authService (signup/signin/google/pwd-reset/soft-delete/restore), pantryService (CRUD + case-insensitive duplicate guard), recipeService (browse+filter+scale), shoppingListService (CRUD+merge logic), syncService (delta sync + tombstone cleanup), emailService (SendGrid) - Routes: /v1/auth, /v1/pantry, /v1/recipes, /v1/shopping-lists, /v1/sync - App: Express factory (createApp), server entry point - Jobs: node-cron daily hard-delete + tombstone cleanup - Tests: validators, utils, auth, pantry, recipes, shopping lists, sync
This commit is contained in:
18
src/config/constants.ts
Normal file
18
src/config/constants.ts
Normal file
@@ -0,0 +1,18 @@
|
||||
export const ALLOWED_UNITS = [
|
||||
'cups', 'tbsp', 'tsp', 'oz', 'fl_oz',
|
||||
'g', 'kg', 'ml', 'l',
|
||||
'pieces', 'slices', 'cloves', 'pinch',
|
||||
'whole', 'can', 'package', 'bunch'
|
||||
] as const;
|
||||
|
||||
export type AllowedUnit = typeof ALLOWED_UNITS[number];
|
||||
|
||||
export const BCRYPT_ROUNDS = 12;
|
||||
export const JWT_EXPIRES_IN = '24h';
|
||||
export const PASSWORD_RESET_EXPIRES_HOURS = 1;
|
||||
export const ACCOUNT_DELETION_DAYS = 15;
|
||||
export const TOMBSTONE_RETENTION_DAYS = 30;
|
||||
export const MAX_RECIPE_SCALE = 3;
|
||||
export const MIN_RECIPE_SCALE = 1;
|
||||
export const DEFAULT_PAGE_LIMIT = 20;
|
||||
export const MAX_PAGE_LIMIT = 50;
|
||||
26
src/config/env.ts
Normal file
26
src/config/env.ts
Normal file
@@ -0,0 +1,26 @@
|
||||
import dotenv from 'dotenv';
|
||||
dotenv.config();
|
||||
|
||||
function requireEnv(key: string): string {
|
||||
const value = process.env[key];
|
||||
if (!value) {
|
||||
throw new Error(`Missing required environment variable: ${key}`);
|
||||
}
|
||||
return value;
|
||||
}
|
||||
|
||||
export const config = {
|
||||
nodeEnv: process.env.NODE_ENV ?? 'development',
|
||||
port: parseInt(process.env.PORT ?? '3000', 10),
|
||||
databaseUrl: process.env.DATABASE_URL ?? 'postgresql://postgres:postgres@localhost:5432/pantree_test',
|
||||
jwtSecret: process.env.JWT_SECRET ?? 'test_secret_do_not_use_in_production_ever',
|
||||
jwtExpiresIn: process.env.JWT_EXPIRES_IN ?? '24h',
|
||||
googleClientId: process.env.GOOGLE_CLIENT_ID ?? '',
|
||||
sendgridApiKey: process.env.SENDGRID_API_KEY ?? '',
|
||||
sendgridFromEmail: process.env.SENDGRID_FROM_EMAIL ?? 'noreply@pantree.app',
|
||||
frontendUrl: process.env.FRONTEND_URL ?? 'http://localhost:3000',
|
||||
passwordResetUrl: process.env.PASSWORD_RESET_URL ?? 'http://localhost:3000/reset-password',
|
||||
isTest: process.env.NODE_ENV === 'test',
|
||||
isDev: process.env.NODE_ENV === 'development',
|
||||
isProd: process.env.NODE_ENV === 'production',
|
||||
};
|
||||
Reference in New Issue
Block a user