Fix auth trustedOrigins and improve emprestimos page layout

- Add production domain to trustedOrigins explicitly
- Add health API endpoint for diagnostics
- Improve emprestimos page layout to match tickets design
- Use correct primary color for buttons
- Fix segmented control styling with rounded borders
- Use Empty component for empty state

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
rever-tecnologia 2025-12-04 16:20:04 -03:00
parent 326da8dae6
commit e9a658341f
3 changed files with 110 additions and 42 deletions

View file

@ -0,0 +1,69 @@
import { NextResponse } from "next/server"
export const runtime = "nodejs"
export const dynamic = "force-dynamic"
export async function GET() {
const diagnostics: Record<string, unknown> = {
timestamp: new Date().toISOString(),
nodeVersion: process.version,
platform: process.platform,
arch: process.arch,
env: {
NODE_ENV: process.env.NODE_ENV,
BETTER_AUTH_URL: process.env.BETTER_AUTH_URL ? "set" : "not set",
NEXT_PUBLIC_APP_URL: process.env.NEXT_PUBLIC_APP_URL ? "set" : "not set",
DATABASE_URL: process.env.DATABASE_URL ? "set" : "not set",
BETTER_AUTH_SECRET: process.env.BETTER_AUTH_SECRET ? "set" : "not set",
},
}
// Test SQLite binding
try {
const Database = (await import("better-sqlite3")).default
const testDb = new Database(":memory:")
testDb.exec("SELECT 1")
testDb.close()
diagnostics.sqlite = { status: "ok", message: "better-sqlite3 binding works" }
} catch (error) {
diagnostics.sqlite = {
status: "error",
message: error instanceof Error ? error.message : String(error),
}
}
// Test Prisma connection
try {
const { prisma } = await import("@/lib/prisma")
await prisma.$queryRaw`SELECT 1`
diagnostics.prisma = { status: "ok", message: "Prisma connection works" }
} catch (error) {
diagnostics.prisma = {
status: "error",
message: error instanceof Error ? error.message : String(error),
}
}
// Test auth config
try {
const { auth } = await import("@/lib/auth")
diagnostics.auth = {
status: "ok",
baseURL: (auth as { options?: { baseURL?: string } }).options?.baseURL ?? "unknown",
trustedOrigins: "configured",
}
} catch (error) {
diagnostics.auth = {
status: "error",
message: error instanceof Error ? error.message : String(error),
}
}
const hasErrors = Object.values(diagnostics).some(
(v) => typeof v === "object" && v !== null && "status" in v && v.status === "error"
)
return NextResponse.json(diagnostics, {
status: hasErrors ? 500 : 200,
})
}

View file

@ -53,6 +53,8 @@ import { DateRangeButton, type DateRangeValue } from "@/components/date-range-bu
import { DatePicker } from "@/components/ui/date-picker"
import { Textarea } from "@/components/ui/textarea"
import { ToggleGroup, ToggleGroupItem } from "@/components/ui/toggle-group"
import { Empty, EmptyHeader, EmptyMedia, EmptyTitle, EmptyDescription } from "@/components/ui/empty"
import { Plus } from "lucide-react"
const EQUIPAMENTO_TIPOS = [
"NOTEBOOK",
@ -401,12 +403,12 @@ export function EmprestimosPageClient() {
const fieldTrigger =
"h-10 w-full rounded-2xl border border-slate-300 bg-slate-50/80 px-3 text-left text-sm font-semibold text-neutral-700 focus:ring-neutral-300 flex items-center gap-2"
const segmentedRoot =
"flex h-10 min-w-[200px] items-stretch rounded-full border border-slate-200 bg-slate-50/70 p-1 gap-1"
"flex h-10 min-w-[220px] flex-1 items-stretch rounded-full border border-slate-200 bg-slate-50/70 p-1 gap-1"
const segmentedItem =
"inline-flex h-full flex-1 items-center justify-center rounded-full px-4 text-sm font-semibold text-neutral-600 transition-colors hover:bg-slate-100 data-[state=on]:bg-cyan-600 data-[state=on]:text-white focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-neutral-300"
"inline-flex h-full flex-1 items-center justify-center rounded-full first:rounded-full last:rounded-full px-4 text-sm font-semibold text-neutral-600 transition-colors hover:bg-slate-100 data-[state=on]:bg-neutral-900 data-[state=on]:text-white focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-neutral-300"
return (
<div className="space-y-6 px-4 lg:px-6">
<div className="mx-auto max-w-6xl space-y-6 px-4 lg:px-6">
{/* Stats Cards */}
<div className="grid gap-4 md:grid-cols-4">
<div className="rounded-2xl border border-slate-200 bg-white/90 p-4 shadow-sm transition-colors hover:border-cyan-200/60 hover:bg-cyan-50/30">
@ -429,24 +431,17 @@ export function EmprestimosPageClient() {
</div>
</div>
{/* Action Button */}
<div className="flex justify-end">
<Button size="sm" onClick={() => setIsCreateDialogOpen(true)}>
<Plus className="size-4" />
Novo empréstimo
</Button>
</div>
{/* Filters Section */}
<section className="rounded-3xl border border-slate-200 bg-white/90 p-4 shadow-sm">
<div className="flex flex-col gap-4">
{/* Header with title and action */}
<div className="flex flex-col gap-3 lg:flex-row lg:items-center lg:justify-between">
<div>
<h2 className="text-lg font-semibold text-neutral-900">Empréstimos de Equipamentos</h2>
<p className="text-sm text-neutral-500">Gerencie o empréstimo de equipamentos para clientes.</p>
</div>
<Button
onClick={() => setIsCreateDialogOpen(true)}
className="h-10 gap-2 rounded-full bg-cyan-600 px-5 font-semibold text-white shadow-sm transition-colors hover:bg-cyan-700"
>
<IconPlus className="size-4" />
Novo empréstimo
</Button>
</div>
{/* Search and Date Range */}
<div className="flex flex-col gap-3 lg:flex-row lg:items-center lg:gap-4">
<div className="flex flex-1 flex-col gap-2">
@ -517,14 +512,22 @@ export function EmprestimosPageClient() {
{/* Table Section */}
<section className="rounded-3xl border border-slate-200 bg-white/90 shadow-sm overflow-hidden">
{!emprestimos ? (
<div className="flex items-center justify-center py-16">
<Spinner className="size-8" />
<div className="flex items-center justify-center gap-2 py-16">
<Spinner className="size-5 text-neutral-500" />
<span className="text-sm text-neutral-600">Carregando empréstimos...</span>
</div>
) : filteredEmprestimos.length === 0 ? (
<div className="flex flex-col items-center justify-center py-16 text-center">
<IconPackage className="mb-4 size-12 text-neutral-300" />
<p className="text-neutral-500">Nenhum empréstimo encontrado.</p>
</div>
<Empty className="m-4 border-dashed">
<EmptyHeader>
<EmptyMedia variant="icon" className="bg-slate-100 text-slate-600">
<IconPackage className="size-5" />
</EmptyMedia>
<EmptyTitle>Nenhum empréstimo encontrado</EmptyTitle>
<EmptyDescription>
Ajuste os filtros ou crie um novo empréstimo para começar.
</EmptyDescription>
</EmptyHeader>
</Empty>
) : (
<div className="overflow-x-auto">
<Table>
@ -571,14 +574,14 @@ export function EmprestimosPageClient() {
</TableCell>
<TableCell className="text-right">
{emp.status === "ATIVO" && (
<Button
size="sm"
className="h-8 gap-1.5 rounded-full bg-emerald-600 px-3 text-xs font-semibold text-white shadow-sm transition-colors hover:bg-emerald-700"
<button
type="button"
onClick={() => openDevolverDialog(emp.id)}
className="inline-flex h-9 items-center gap-2 rounded-full border border-emerald-200 bg-emerald-50 px-3 text-sm font-semibold text-emerald-700 transition hover:border-emerald-300 hover:bg-emerald-100"
>
<IconCircleCheck className="size-3.5" />
<IconCircleCheck className="size-4" />
Devolver
</Button>
</button>
)}
</TableCell>
</TableRow>
@ -767,14 +770,10 @@ export function EmprestimosPageClient() {
</div>
<DialogFooter>
<Button variant="outline" onClick={() => setIsCreateDialogOpen(false)} className="rounded-full">
<Button variant="outline" onClick={() => setIsCreateDialogOpen(false)}>
Cancelar
</Button>
<Button
onClick={handleCreate}
disabled={isSubmitting}
className="gap-2 rounded-full bg-cyan-600 font-semibold text-white hover:bg-cyan-700"
>
<Button onClick={handleCreate} disabled={isSubmitting}>
{isSubmitting ? (
<>
<Spinner className="size-4" />
@ -811,14 +810,10 @@ export function EmprestimosPageClient() {
</div>
<DialogFooter>
<Button variant="outline" onClick={() => setIsDevolverDialogOpen(false)} className="rounded-full">
<Button variant="outline" onClick={() => setIsDevolverDialogOpen(false)}>
Cancelar
</Button>
<Button
onClick={handleDevolver}
disabled={isSubmitting}
className="gap-2 rounded-full bg-emerald-600 font-semibold text-white hover:bg-emerald-700"
>
<Button onClick={handleDevolver} disabled={isSubmitting}>
{isSubmitting ? (
<>
<Spinner className="size-4" />

View file

@ -8,12 +8,16 @@ import { prisma } from "./prisma"
export const auth = betterAuth({
secret: env.BETTER_AUTH_SECRET,
baseURL: env.BETTER_AUTH_URL,
// Permite login tanto no domínio de produção quanto no localhost em DEV
// Permite login no dominio de producao e localhost em DEV
trustedOrigins: Array.from(
new Set(
[
// Dominio de producao (sempre incluido)
"https://tickets.esdrasrenan.com.br",
// URLs das variaveis de ambiente
env.BETTER_AUTH_URL,
env.NEXT_PUBLIC_APP_URL,
// Localhost apenas em desenvolvimento
process.env.NODE_ENV !== "production" ? "http://localhost:3000" : undefined,
process.env.NODE_ENV !== "production" ? "http://127.0.0.1:3000" : undefined,
process.env.NODE_ENV !== "production" ? "http://localhost:3001" : undefined,