mirror of
https://github.com/simstudioai/sim.git
synced 2026-04-28 03:00:29 -04:00
* feat(audit-log): add audit events for templates, billing, credentials, env, deployments, passwords * improvement(audit-log): add actorName/actorEmail to all recordAudit calls * fix(audit-log): resolve user for password reset, add CREDENTIAL_SET_INVITATION_RESENT action * fix(audit-log): add workspaceId to deployment activation audit * improvement(audit-log): use better-auth callback for password reset audit, remove cast - Move password reset audit to onPasswordReset callback in auth config instead of coupling to better-auth's verification table internals - Remove ugly double-cast on workflowData.workspaceId in deployment activation * fix(audit-log): add missing actorName/actorEmail to workflow duplicate * improvement(audit-log): add resourceName to credential set invitation accept
173 lines
5.7 KiB
TypeScript
173 lines
5.7 KiB
TypeScript
import { db } from '@sim/db'
|
|
import { workflow } from '@sim/db/schema'
|
|
import { createLogger } from '@sim/logger'
|
|
import { eq } from 'drizzle-orm'
|
|
import { type NextRequest, NextResponse } from 'next/server'
|
|
import { z } from 'zod'
|
|
import { AuditAction, AuditResourceType, recordAudit } from '@/lib/audit/log'
|
|
import { checkSessionOrInternalAuth } from '@/lib/auth/hybrid'
|
|
import { generateRequestId } from '@/lib/core/utils/request'
|
|
import { authorizeWorkflowByWorkspacePermission } from '@/lib/workflows/utils'
|
|
import type { Variable } from '@/stores/panel/variables/types'
|
|
|
|
const logger = createLogger('WorkflowVariablesAPI')
|
|
|
|
const VariableSchema = z.object({
|
|
id: z.string(),
|
|
workflowId: z.string(),
|
|
name: z.string(),
|
|
type: z.enum(['string', 'number', 'boolean', 'object', 'array', 'plain']),
|
|
value: z.union([
|
|
z.string(),
|
|
z.number(),
|
|
z.boolean(),
|
|
z.record(z.unknown()),
|
|
z.array(z.unknown()),
|
|
]),
|
|
})
|
|
|
|
const VariablesSchema = z.object({
|
|
variables: z.record(z.string(), VariableSchema),
|
|
})
|
|
|
|
export async function POST(req: NextRequest, { params }: { params: Promise<{ id: string }> }) {
|
|
const requestId = generateRequestId()
|
|
const workflowId = (await params).id
|
|
|
|
try {
|
|
const auth = await checkSessionOrInternalAuth(req, { requireWorkflowId: false })
|
|
if (!auth.success || !auth.userId) {
|
|
logger.warn(`[${requestId}] Unauthorized workflow variables update attempt`)
|
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })
|
|
}
|
|
const userId = auth.userId
|
|
|
|
const authorization = await authorizeWorkflowByWorkspacePermission({
|
|
workflowId,
|
|
userId,
|
|
action: 'write',
|
|
})
|
|
const workflowData = authorization.workflow
|
|
|
|
if (!workflowData) {
|
|
logger.warn(`[${requestId}] Workflow not found: ${workflowId}`)
|
|
return NextResponse.json({ error: 'Workflow not found' }, { status: 404 })
|
|
}
|
|
const isAuthorized = authorization.allowed
|
|
|
|
if (!isAuthorized) {
|
|
logger.warn(
|
|
`[${requestId}] User ${userId} attempted to update variables for workflow ${workflowId} without permission`
|
|
)
|
|
return NextResponse.json(
|
|
{ error: authorization.message || 'Access denied' },
|
|
{ status: authorization.status || 403 }
|
|
)
|
|
}
|
|
|
|
const body = await req.json()
|
|
|
|
try {
|
|
const { variables } = VariablesSchema.parse(body)
|
|
|
|
// Variables are already in Record format - use directly
|
|
// The frontend is the source of truth for what variables should exist
|
|
await db
|
|
.update(workflow)
|
|
.set({
|
|
variables,
|
|
updatedAt: new Date(),
|
|
})
|
|
.where(eq(workflow.id, workflowId))
|
|
|
|
recordAudit({
|
|
workspaceId: workflowData.workspaceId ?? null,
|
|
actorId: userId,
|
|
actorName: auth.userName,
|
|
actorEmail: auth.userEmail,
|
|
action: AuditAction.WORKFLOW_VARIABLES_UPDATED,
|
|
resourceType: AuditResourceType.WORKFLOW,
|
|
resourceId: workflowId,
|
|
resourceName: workflowData.name ?? undefined,
|
|
description: `Updated workflow variables`,
|
|
request: req,
|
|
})
|
|
|
|
return NextResponse.json({ success: true })
|
|
} catch (validationError) {
|
|
if (validationError instanceof z.ZodError) {
|
|
logger.warn(`[${requestId}] Invalid workflow variables data`, {
|
|
errors: validationError.errors,
|
|
})
|
|
return NextResponse.json(
|
|
{ error: 'Invalid request data', details: validationError.errors },
|
|
{ status: 400 }
|
|
)
|
|
}
|
|
throw validationError
|
|
}
|
|
} catch (error) {
|
|
logger.error(`[${requestId}] Error updating workflow variables`, error)
|
|
return NextResponse.json({ error: 'Failed to update workflow variables' }, { status: 500 })
|
|
}
|
|
}
|
|
|
|
export async function GET(req: NextRequest, { params }: { params: Promise<{ id: string }> }) {
|
|
const requestId = generateRequestId()
|
|
const workflowId = (await params).id
|
|
|
|
try {
|
|
const auth = await checkSessionOrInternalAuth(req, { requireWorkflowId: false })
|
|
if (!auth.success || !auth.userId) {
|
|
logger.warn(`[${requestId}] Unauthorized workflow variables access attempt`)
|
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })
|
|
}
|
|
const userId = auth.userId
|
|
|
|
const authorization = await authorizeWorkflowByWorkspacePermission({
|
|
workflowId,
|
|
userId,
|
|
action: 'read',
|
|
})
|
|
const workflowData = authorization.workflow
|
|
|
|
if (!workflowData) {
|
|
logger.warn(`[${requestId}] Workflow not found: ${workflowId}`)
|
|
return NextResponse.json({ error: 'Workflow not found' }, { status: 404 })
|
|
}
|
|
const isAuthorized = authorization.allowed
|
|
|
|
if (!isAuthorized) {
|
|
logger.warn(
|
|
`[${requestId}] User ${userId} attempted to access variables for workflow ${workflowId} without permission`
|
|
)
|
|
return NextResponse.json(
|
|
{ error: authorization.message || 'Access denied' },
|
|
{ status: authorization.status || 403 }
|
|
)
|
|
}
|
|
|
|
// Return variables if they exist
|
|
const variables = (workflowData.variables as Record<string, Variable>) || {}
|
|
|
|
// Add cache headers to prevent frequent reloading
|
|
const variableHash = JSON.stringify(variables).length
|
|
const headers = new Headers({
|
|
'Cache-Control': 'max-age=30, stale-while-revalidate=300', // Cache for 30 seconds, stale for 5 min
|
|
ETag: `"variables-${workflowId}-${variableHash}"`,
|
|
})
|
|
|
|
return NextResponse.json(
|
|
{ data: variables },
|
|
{
|
|
status: 200,
|
|
headers,
|
|
}
|
|
)
|
|
} catch (error) {
|
|
logger.error(`[${requestId}] Workflow variables fetch error`, error)
|
|
const errorMessage = error instanceof Error ? error.message : 'Unknown error'
|
|
return NextResponse.json({ error: errorMessage }, { status: 500 })
|
|
}
|
|
}
|