Change ticket IDs from UUID to sequential integers

Backend:
- tickets.id: uuid → integer GENERATED ALWAYS AS IDENTITY
- transactions.ticket_id, custom_field_values.ticket_id: uuid → integer
- Routes convert string params to Number() for DB queries
- ScripEngine.prepare takes ticketId: number
- ActionPayload.ticketId: string → number

Frontend:
- Ticket.id: string → number, Transaction.ticket_id: string → number
- API functions accept number params
- formatTicketId() helper returns TKT-0001 format
- Ticket rows display TKT-XXXX, detail page uses formatTicketId

Migration: drops FKs, clears data, alters column types, re-adds FKs
This commit is contained in:
Gjermund Høsøien Wiggen
2026-06-07 23:23:05 +02:00
parent 7da52dfff6
commit 04b4e28d21
12 changed files with 979 additions and 22 deletions

View File

@@ -13,7 +13,7 @@ export interface ActionPayload {
scripName: string;
actionType: string;
actionConfig: Record<string, unknown>;
ticketId?: string;
ticketId?: number;
recipients?: string[];
subject?: string;
body?: string;
@@ -97,7 +97,7 @@ export class SetCustomField implements ActionExecutor {
async execute(payload: ActionPayload): Promise<{ success: boolean; message: string }> {
const fieldId = payload.field_id ?? String(payload.actionConfig['field_id'] ?? '');
const value = payload.value ?? String(payload.actionConfig['value'] ?? '');
const ticketId = payload.ticketId ?? String(payload.actionConfig['ticket_id'] ?? '');
const ticketId = payload.ticketId ?? Number(payload.actionConfig['ticket_id'] ?? 0);
if (!fieldId || !value || !ticketId) {
return { success: false, message: 'SetCustomField: missing field_id, value, or ticket_id' };
@@ -121,7 +121,7 @@ export class CreateTransaction implements ActionExecutor {
constructor(private db: Db) {}
async execute(payload: ActionPayload): Promise<{ success: boolean; message: string }> {
const ticketId = payload.ticketId ?? String(payload.actionConfig['ticket_id'] ?? '');
const ticketId = payload.ticketId ?? Number(payload.actionConfig['ticket_id'] ?? 0);
const transactionType = String(payload.actionConfig['transaction_type'] ?? '');
const field = payload.actionConfig['field'] as string | undefined ?? null;
const oldValue = payload.actionConfig['old_value'] as string | undefined ?? null;

View File

@@ -35,7 +35,7 @@ export class ScripEngine {
}
async prepare(
ticketId: string,
ticketId: number,
transactions: Transaction[],
): Promise<PreparedScrip[]> {
const ticketRecord = await this.db.query.tickets.findFirst({