101 lines
2.9 KiB
PL/PgSQL
101 lines
2.9 KiB
PL/PgSQL
-- Allow multiple check-in/out within a single duty schedule.
|
|
-- Preserves first check-in time & arrival/late status on subsequent re-check-ins.
|
|
|
|
CREATE OR REPLACE FUNCTION public.attendance_check_in(
|
|
p_duty_id uuid,
|
|
p_lat double precision,
|
|
p_lng double precision
|
|
) RETURNS uuid
|
|
LANGUAGE plpgsql SECURITY DEFINER
|
|
AS $$
|
|
DECLARE
|
|
v_schedule duty_schedules%ROWTYPE;
|
|
v_log_id uuid;
|
|
v_now timestamptz := now();
|
|
v_status text;
|
|
BEGIN
|
|
SELECT * INTO v_schedule FROM duty_schedules WHERE id = p_duty_id;
|
|
IF NOT FOUND THEN
|
|
RAISE EXCEPTION 'Duty schedule not found';
|
|
END IF;
|
|
IF v_schedule.user_id != auth.uid() THEN
|
|
RAISE EXCEPTION 'Not your duty schedule';
|
|
END IF;
|
|
|
|
IF v_now < (v_schedule.start_time - interval '2 hours') THEN
|
|
RAISE EXCEPTION 'Too early to check in (2-hour window)';
|
|
END IF;
|
|
IF v_now > v_schedule.end_time THEN
|
|
RAISE EXCEPTION 'Duty has already ended';
|
|
END IF;
|
|
|
|
-- Determine status (used only when first check-in)
|
|
IF v_now <= v_schedule.start_time THEN
|
|
v_status := 'arrival';
|
|
ELSE
|
|
v_status := 'late';
|
|
END IF;
|
|
|
|
-- Insert a new attendance log row for each check-in
|
|
INSERT INTO attendance_logs (user_id, duty_schedule_id, check_in_at, check_in_lat, check_in_lng)
|
|
VALUES (auth.uid(), p_duty_id, v_now, p_lat, p_lng)
|
|
RETURNING id INTO v_log_id;
|
|
|
|
-- Only update duty schedule on first check-in (preserve original arrival status)
|
|
UPDATE duty_schedules
|
|
SET check_in_at = COALESCE(check_in_at, v_now),
|
|
check_in_location = ST_SetSRID(ST_MakePoint(p_lng, p_lat), 4326)::geography,
|
|
status = CASE
|
|
WHEN check_in_at IS NULL THEN v_status::duty_status
|
|
ELSE status
|
|
END
|
|
WHERE id = p_duty_id;
|
|
|
|
RETURN v_log_id;
|
|
END;
|
|
$$;
|
|
|
|
-- Also update duty_check_in for consistency
|
|
CREATE OR REPLACE FUNCTION public.duty_check_in(
|
|
p_duty_id uuid,
|
|
p_lat double precision,
|
|
p_lng double precision
|
|
) RETURNS text
|
|
LANGUAGE plpgsql SECURITY DEFINER
|
|
AS $$
|
|
DECLARE
|
|
v_schedule duty_schedules%ROWTYPE;
|
|
v_now timestamptz := now();
|
|
v_status text;
|
|
BEGIN
|
|
SELECT * INTO v_schedule FROM duty_schedules WHERE id = p_duty_id;
|
|
IF NOT FOUND THEN
|
|
RAISE EXCEPTION 'Duty schedule not found';
|
|
END IF;
|
|
IF v_schedule.user_id != auth.uid() THEN
|
|
RAISE EXCEPTION 'Not your duty schedule';
|
|
END IF;
|
|
|
|
IF v_now <= v_schedule.start_time THEN
|
|
v_status := 'arrival';
|
|
ELSE
|
|
v_status := 'late';
|
|
END IF;
|
|
|
|
UPDATE duty_schedules
|
|
SET check_in_at = COALESCE(check_in_at, v_now),
|
|
check_in_location = ST_SetSRID(ST_MakePoint(p_lng, p_lat), 4326)::geography,
|
|
status = CASE
|
|
WHEN check_in_at IS NULL THEN v_status::duty_status
|
|
ELSE status
|
|
END
|
|
WHERE id = p_duty_id;
|
|
|
|
-- Create attendance log entry
|
|
INSERT INTO attendance_logs (user_id, duty_schedule_id, check_in_at, check_in_lat, check_in_lng)
|
|
VALUES (auth.uid(), p_duty_id, v_now, p_lat, p_lng);
|
|
|
|
RETURN v_status;
|
|
END;
|
|
$$;
|