372 lines
9.3 KiB
TypeScript
372 lines
9.3 KiB
TypeScript
import { createClient } from '@supabase/supabase-js'
|
|
|
|
// Local development configuration
|
|
const supabaseUrl = 'http://127.0.0.1:54321'
|
|
const supabaseAnonKey = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZS1kZW1vIiwicm9sZSI6ImFub24iLCJleHAiOjE5ODM4MTI5OTZ9.CRXP1A7WOeoJeXxjNni43kdQwgnWNReilDMblYTn_I0'
|
|
|
|
export const supabase = createClient(supabaseUrl, supabaseAnonKey)
|
|
|
|
// Database types for TypeScript
|
|
export type RoleName = 'admin' | 'conductor' | 'analyst' | 'data recorder'
|
|
export type UserStatus = 'active' | 'disabled'
|
|
export type ScheduleStatus = 'pending schedule' | 'scheduled' | 'canceled' | 'aborted'
|
|
export type ResultsStatus = 'valid' | 'invalid'
|
|
|
|
export interface User {
|
|
id: string
|
|
email: string
|
|
roles: RoleName[]
|
|
status: UserStatus
|
|
created_at: string
|
|
updated_at: string
|
|
}
|
|
|
|
export interface Role {
|
|
id: string
|
|
name: RoleName
|
|
description: string
|
|
created_at: string
|
|
}
|
|
|
|
export interface Experiment {
|
|
id: string
|
|
experiment_number: number
|
|
reps_required: number
|
|
rep_number: number
|
|
soaking_duration_hr: number
|
|
air_drying_time_min: number
|
|
plate_contact_frequency_hz: number
|
|
throughput_rate_pecans_sec: number
|
|
crush_amount_in: number
|
|
entry_exit_height_diff_in: number
|
|
schedule_status: ScheduleStatus
|
|
results_status: ResultsStatus
|
|
created_at: string
|
|
updated_at: string
|
|
created_by: string
|
|
}
|
|
|
|
export interface CreateExperimentRequest {
|
|
experiment_number: number
|
|
reps_required: number
|
|
rep_number: number
|
|
soaking_duration_hr: number
|
|
air_drying_time_min: number
|
|
plate_contact_frequency_hz: number
|
|
throughput_rate_pecans_sec: number
|
|
crush_amount_in: number
|
|
entry_exit_height_diff_in: number
|
|
schedule_status?: ScheduleStatus
|
|
results_status?: ResultsStatus
|
|
}
|
|
|
|
export interface UpdateExperimentRequest {
|
|
experiment_number?: number
|
|
reps_required?: number
|
|
rep_number?: number
|
|
soaking_duration_hr?: number
|
|
air_drying_time_min?: number
|
|
plate_contact_frequency_hz?: number
|
|
throughput_rate_pecans_sec?: number
|
|
crush_amount_in?: number
|
|
entry_exit_height_diff_in?: number
|
|
schedule_status?: ScheduleStatus
|
|
results_status?: ResultsStatus
|
|
}
|
|
|
|
export interface UserRole {
|
|
id: string
|
|
user_id: string
|
|
role_id: string
|
|
assigned_at: string
|
|
assigned_by?: string
|
|
}
|
|
|
|
export interface UserProfile {
|
|
id: string
|
|
email: string
|
|
status: UserStatus
|
|
created_at: string
|
|
updated_at: string
|
|
role_id?: string // Legacy field, will be deprecated
|
|
}
|
|
|
|
export interface CreateUserRequest {
|
|
email: string
|
|
roles: RoleName[]
|
|
tempPassword?: string
|
|
}
|
|
|
|
export interface CreateUserResponse {
|
|
user_id: string
|
|
email: string
|
|
temp_password: string
|
|
roles: RoleName[]
|
|
status: UserStatus
|
|
}
|
|
|
|
// User management utility functions
|
|
export const userManagement = {
|
|
// Get all users with their roles
|
|
async getAllUsers(): Promise<User[]> {
|
|
const { data: profiles, error: profilesError } = await supabase
|
|
.from('user_profiles')
|
|
.select(`
|
|
id,
|
|
email,
|
|
status,
|
|
created_at,
|
|
updated_at
|
|
`)
|
|
|
|
if (profilesError) throw profilesError
|
|
|
|
// Get roles for each user
|
|
const usersWithRoles = await Promise.all(
|
|
profiles.map(async (profile) => {
|
|
const { data: userRoles, error: rolesError } = await supabase
|
|
.from('user_roles')
|
|
.select(`
|
|
roles!inner (
|
|
name
|
|
)
|
|
`)
|
|
.eq('user_id', profile.id)
|
|
|
|
if (rolesError) throw rolesError
|
|
|
|
return {
|
|
...profile,
|
|
roles: userRoles.map(ur => ur.roles.name as RoleName)
|
|
}
|
|
})
|
|
)
|
|
|
|
return usersWithRoles
|
|
},
|
|
|
|
// Get all available roles
|
|
async getAllRoles(): Promise<Role[]> {
|
|
const { data, error } = await supabase
|
|
.from('roles')
|
|
.select('*')
|
|
.order('name')
|
|
|
|
if (error) throw error
|
|
return data
|
|
},
|
|
|
|
// Create a new user with roles
|
|
async createUser(userData: CreateUserRequest): Promise<CreateUserResponse> {
|
|
const { data, error } = await supabase.rpc('create_user_with_roles', {
|
|
user_email: userData.email,
|
|
role_names: userData.roles,
|
|
temp_password: userData.tempPassword
|
|
})
|
|
|
|
if (error) throw error
|
|
return data
|
|
},
|
|
|
|
// Update user status (enable/disable)
|
|
async updateUserStatus(userId: string, status: UserStatus): Promise<void> {
|
|
const { error } = await supabase
|
|
.from('user_profiles')
|
|
.update({ status })
|
|
.eq('id', userId)
|
|
|
|
if (error) throw error
|
|
},
|
|
|
|
// Update user roles
|
|
async updateUserRoles(userId: string, roleNames: RoleName[]): Promise<void> {
|
|
// First, remove all existing roles for the user
|
|
const { error: deleteError } = await supabase
|
|
.from('user_roles')
|
|
.delete()
|
|
.eq('user_id', userId)
|
|
|
|
if (deleteError) throw deleteError
|
|
|
|
// Get role IDs for the new roles
|
|
const { data: roles, error: rolesError } = await supabase
|
|
.from('roles')
|
|
.select('id, name')
|
|
.in('name', roleNames)
|
|
|
|
if (rolesError) throw rolesError
|
|
|
|
// Insert new role assignments
|
|
const roleAssignments = roles.map(role => ({
|
|
user_id: userId,
|
|
role_id: role.id
|
|
}))
|
|
|
|
const { error: insertError } = await supabase
|
|
.from('user_roles')
|
|
.insert(roleAssignments)
|
|
|
|
if (insertError) throw insertError
|
|
},
|
|
|
|
// Update user email
|
|
async updateUserEmail(userId: string, email: string): Promise<void> {
|
|
const { error } = await supabase
|
|
.from('user_profiles')
|
|
.update({ email })
|
|
.eq('id', userId)
|
|
|
|
if (error) throw error
|
|
},
|
|
|
|
// Get current user with roles
|
|
async getCurrentUser(): Promise<User | null> {
|
|
const { data: { user: authUser }, error: authError } = await supabase.auth.getUser()
|
|
|
|
if (authError || !authUser) return null
|
|
|
|
const { data: profile, error: profileError } = await supabase
|
|
.from('user_profiles')
|
|
.select(`
|
|
id,
|
|
email,
|
|
status,
|
|
created_at,
|
|
updated_at
|
|
`)
|
|
.eq('id', authUser.id)
|
|
.single()
|
|
|
|
if (profileError) throw profileError
|
|
|
|
const { data: userRoles, error: rolesError } = await supabase
|
|
.from('user_roles')
|
|
.select(`
|
|
roles!inner (
|
|
name
|
|
)
|
|
`)
|
|
.eq('user_id', authUser.id)
|
|
|
|
if (rolesError) throw rolesError
|
|
|
|
return {
|
|
...profile,
|
|
roles: userRoles.map(ur => ur.roles.name as RoleName)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Experiment management utility functions
|
|
export const experimentManagement = {
|
|
// Get all experiments
|
|
async getAllExperiments(): Promise<Experiment[]> {
|
|
const { data, error } = await supabase
|
|
.from('experiments')
|
|
.select('*')
|
|
.order('created_at', { ascending: false })
|
|
|
|
if (error) throw error
|
|
return data
|
|
},
|
|
|
|
// Get experiment by ID
|
|
async getExperimentById(id: string): Promise<Experiment | null> {
|
|
const { data, error } = await supabase
|
|
.from('experiments')
|
|
.select('*')
|
|
.eq('id', id)
|
|
.single()
|
|
|
|
if (error) {
|
|
if (error.code === 'PGRST116') return null // Not found
|
|
throw error
|
|
}
|
|
return data
|
|
},
|
|
|
|
// Create a new experiment
|
|
async createExperiment(experimentData: CreateExperimentRequest): Promise<Experiment> {
|
|
const { data: { user }, error: authError } = await supabase.auth.getUser()
|
|
if (authError || !user) throw new Error('User not authenticated')
|
|
|
|
const { data, error } = await supabase
|
|
.from('experiments')
|
|
.insert({
|
|
...experimentData,
|
|
created_by: user.id
|
|
})
|
|
.select()
|
|
.single()
|
|
|
|
if (error) throw error
|
|
return data
|
|
},
|
|
|
|
// Update an experiment
|
|
async updateExperiment(id: string, updates: UpdateExperimentRequest): Promise<Experiment> {
|
|
const { data, error } = await supabase
|
|
.from('experiments')
|
|
.update(updates)
|
|
.eq('id', id)
|
|
.select()
|
|
.single()
|
|
|
|
if (error) throw error
|
|
return data
|
|
},
|
|
|
|
// Delete an experiment (admin only)
|
|
async deleteExperiment(id: string): Promise<void> {
|
|
const { error } = await supabase
|
|
.from('experiments')
|
|
.delete()
|
|
.eq('id', id)
|
|
|
|
if (error) throw error
|
|
},
|
|
|
|
// Update experiment status
|
|
async updateExperimentStatus(id: string, scheduleStatus?: ScheduleStatus, resultsStatus?: ResultsStatus): Promise<Experiment> {
|
|
const updates: Partial<UpdateExperimentRequest> = {}
|
|
if (scheduleStatus) updates.schedule_status = scheduleStatus
|
|
if (resultsStatus) updates.results_status = resultsStatus
|
|
|
|
return this.updateExperiment(id, updates)
|
|
},
|
|
|
|
// Get experiments by status
|
|
async getExperimentsByStatus(scheduleStatus?: ScheduleStatus, resultsStatus?: ResultsStatus): Promise<Experiment[]> {
|
|
let query = supabase.from('experiments').select('*')
|
|
|
|
if (scheduleStatus) {
|
|
query = query.eq('schedule_status', scheduleStatus)
|
|
}
|
|
if (resultsStatus) {
|
|
query = query.eq('results_status', resultsStatus)
|
|
}
|
|
|
|
const { data, error } = await query.order('created_at', { ascending: false })
|
|
|
|
if (error) throw error
|
|
return data
|
|
},
|
|
|
|
// Check if experiment number is unique
|
|
async isExperimentNumberUnique(experimentNumber: number, excludeId?: string): Promise<boolean> {
|
|
let query = supabase
|
|
.from('experiments')
|
|
.select('id')
|
|
.eq('experiment_number', experimentNumber)
|
|
|
|
if (excludeId) {
|
|
query = query.neq('id', excludeId)
|
|
}
|
|
|
|
const { data, error } = await query
|
|
|
|
if (error) throw error
|
|
return data.length === 0
|
|
}
|
|
}
|