666306b416
- Updated medication schema to replace stripsPerPack and tabsPerStrip with blistersPerPack and pillsPerBlister. - Adjusted medication routes to handle new blister and pill structure, including calculations for total pills. - Modified frontend components to reflect changes in medication data structure and ensure compatibility with new backend logic. - Updated reminder scheduler and share routes to utilize the new medication model. - Enhanced Docker configuration for better permissions handling during development.
115 lines
6.9 KiB
TypeScript
115 lines
6.9 KiB
TypeScript
import { sqliteTable, text, integer } from "drizzle-orm/sqlite-core";
|
|
import { sql } from "drizzle-orm";
|
|
|
|
// =============================================================================
|
|
// Users - Simple auth, no roles (every user is equal)
|
|
// =============================================================================
|
|
export const users = sqliteTable("users", {
|
|
id: integer("id").primaryKey({ autoIncrement: true }),
|
|
username: text("username", { length: 100 }).notNull().unique(),
|
|
passwordHash: text("password_hash", { length: 255 }),
|
|
avatarUrl: text("avatar_url", { length: 255 }),
|
|
authProvider: text("auth_provider", { length: 50 }).notNull().default("local"),
|
|
oidcSubject: text("oidc_subject", { length: 255 }), // OIDC provider's unique user ID (sub claim)
|
|
isActive: integer("is_active", { mode: "boolean" }).notNull().default(true),
|
|
lastLoginAt: integer("last_login_at", { mode: "timestamp" }),
|
|
createdAt: integer("created_at", { mode: "timestamp" }).notNull().default(sql`CURRENT_TIMESTAMP`),
|
|
updatedAt: integer("updated_at", { mode: "timestamp" }).notNull().default(sql`CURRENT_TIMESTAMP`),
|
|
});
|
|
|
|
// =============================================================================
|
|
// Medications - Per user
|
|
// =============================================================================
|
|
export const medications = sqliteTable("medications", {
|
|
id: integer("id").primaryKey({ autoIncrement: true }),
|
|
userId: integer("user_id").notNull().references(() => users.id, { onDelete: "cascade" }),
|
|
name: text("name", { length: 100 }).notNull(),
|
|
genericName: text("generic_name", { length: 100 }),
|
|
takenByJson: text("taken_by_json").notNull().default("[]"), // JSON array of person names
|
|
packCount: integer("pack_count").notNull().default(1),
|
|
blistersPerPack: integer("blisters_per_pack").notNull().default(1),
|
|
pillsPerBlister: integer("pills_per_blister").notNull().default(1),
|
|
looseTablets: integer("loose_tablets").notNull().default(0),
|
|
pillWeightMg: integer("pill_weight_mg"),
|
|
usageJson: text("usage_json").notNull().default("[]"),
|
|
everyJson: text("every_json").notNull().default("[]"),
|
|
startJson: text("start_json").notNull().default("[]"),
|
|
imageUrl: text("image_url"),
|
|
expiryDate: text("expiry_date"),
|
|
notes: text("notes"),
|
|
intakeRemindersEnabled: integer("intake_reminders_enabled", { mode: "boolean" }).notNull().default(false),
|
|
updatedAt: integer("updated_at", { mode: "timestamp" }).notNull().default(sql`CURRENT_TIMESTAMP`),
|
|
});
|
|
|
|
// =============================================================================
|
|
// User Settings - Per user (email, push, thresholds, language)
|
|
// =============================================================================
|
|
export const userSettings = sqliteTable("user_settings", {
|
|
id: integer("id").primaryKey({ autoIncrement: true }),
|
|
userId: integer("user_id").notNull().unique().references(() => users.id, { onDelete: "cascade" }),
|
|
// Email notifications
|
|
emailEnabled: integer("email_enabled", { mode: "boolean" }).notNull().default(false),
|
|
notificationEmail: text("notification_email"),
|
|
emailStockReminders: integer("email_stock_reminders", { mode: "boolean" }).notNull().default(true),
|
|
emailIntakeReminders: integer("email_intake_reminders", { mode: "boolean" }).notNull().default(true),
|
|
// Push notifications (shoutrrr/ntfy)
|
|
shoutrrrEnabled: integer("shoutrrr_enabled", { mode: "boolean" }).notNull().default(false),
|
|
shoutrrrUrl: text("shoutrrr_url"),
|
|
shoutrrrStockReminders: integer("shoutrrr_stock_reminders", { mode: "boolean" }).notNull().default(true),
|
|
shoutrrrIntakeReminders: integer("shoutrrr_intake_reminders", { mode: "boolean" }).notNull().default(true),
|
|
// Reminder settings
|
|
reminderDaysBefore: integer("reminder_days_before").notNull().default(7),
|
|
repeatDailyReminders: integer("repeat_daily_reminders", { mode: "boolean" }).notNull().default(false),
|
|
// Stock thresholds (days)
|
|
lowStockDays: integer("low_stock_days").notNull().default(30),
|
|
normalStockDays: integer("normal_stock_days").notNull().default(90),
|
|
highStockDays: integer("high_stock_days").notNull().default(180),
|
|
// UI preferences
|
|
language: text("language", { length: 10 }).notNull().default("en"),
|
|
// Stock calculation mode: "automatic" (schedule-based) or "manual" (only marked doses)
|
|
stockCalculationMode: text("stock_calculation_mode", { length: 20 }).notNull().default("automatic"),
|
|
// Last notification tracking
|
|
lastAutoEmailSent: text("last_auto_email_sent"),
|
|
lastNotificationType: text("last_notification_type"),
|
|
lastNotificationChannel: text("last_notification_channel"),
|
|
// Timestamps
|
|
updatedAt: integer("updated_at", { mode: "timestamp" }).notNull().default(sql`CURRENT_TIMESTAMP`),
|
|
});
|
|
|
|
// =============================================================================
|
|
// Refresh Tokens - For JWT rotation
|
|
// =============================================================================
|
|
export const refreshTokens = sqliteTable("refresh_tokens", {
|
|
id: integer("id").primaryKey({ autoIncrement: true }),
|
|
userId: integer("user_id").notNull().references(() => users.id, { onDelete: "cascade" }),
|
|
tokenId: text("token_id", { length: 255 }).notNull().unique(),
|
|
expiresAt: integer("expires_at", { mode: "timestamp" }).notNull(),
|
|
rotatedAt: integer("rotated_at", { mode: "timestamp" }),
|
|
revoked: integer("revoked", { mode: "boolean" }).notNull().default(false),
|
|
createdAt: integer("created_at", { mode: "timestamp" }).notNull().default(sql`CURRENT_TIMESTAMP`),
|
|
});
|
|
|
|
// =============================================================================
|
|
// Share Tokens - For public schedule sharing by takenBy person
|
|
// =============================================================================
|
|
export const shareTokens = sqliteTable("share_tokens", {
|
|
id: integer("id").primaryKey({ autoIncrement: true }),
|
|
userId: integer("user_id").notNull().references(() => users.id, { onDelete: "cascade" }),
|
|
token: text("token", { length: 64 }).notNull().unique(),
|
|
takenBy: text("taken_by", { length: 100 }).notNull(),
|
|
scheduleDays: integer("schedule_days").notNull().default(30),
|
|
createdAt: integer("created_at", { mode: "timestamp" }).notNull().default(sql`CURRENT_TIMESTAMP`),
|
|
expiresAt: integer("expires_at", { mode: "timestamp" }), // NULL = never expires
|
|
});
|
|
|
|
// =============================================================================
|
|
// Dose Tracking - Tracks when doses are marked as taken
|
|
// =============================================================================
|
|
export const doseTracking = sqliteTable("dose_tracking", {
|
|
id: integer("id").primaryKey({ autoIncrement: true }),
|
|
userId: integer("user_id").notNull().references(() => users.id, { onDelete: "cascade" }),
|
|
doseId: text("dose_id", { length: 255 }).notNull(), // e.g. "med-5-1-86400000-1735200000000"
|
|
takenAt: integer("taken_at", { mode: "timestamp" }).notNull().default(sql`CURRENT_TIMESTAMP`),
|
|
markedBy: text("marked_by", { length: 100 }), // null = user, "Daniel" = via share link
|
|
});
|