mirror of
https://github.com/simstudioai/sim.git
synced 2026-04-06 03:00:16 -04:00
* hide form deployment tab from docs * progress * fix resolution * cleanup code * fix positioning * cleanup dead sockets adv mode ops * address greptile comments * fix tests plus more simplification * fix cleanup * bring back advanced mode with specific definition * revert feature flags * improvement(subblock): ui * resolver change to make all var references optional chaining * fix(webhooks/schedules): deployment version friendly * fix tests * fix credential sets with new lifecycle * prep merge * add back migration * fix display check for adv fields * fix trigger vs block scoping --------- Co-authored-by: Emir Karabeg <emirkarabeg@berkeley.edu>
177 lines
5.6 KiB
TypeScript
177 lines
5.6 KiB
TypeScript
import { createLogger } from '@sim/logger'
|
|
import { type NextRequest, NextResponse } from 'next/server'
|
|
import { generateRequestId } from '@/lib/core/utils/request'
|
|
import {
|
|
checkWebhookPreprocessing,
|
|
findAllWebhooksForPath,
|
|
formatProviderErrorResponse,
|
|
handlePreDeploymentVerification,
|
|
handleProviderChallenges,
|
|
handleProviderReachabilityTest,
|
|
parseWebhookBody,
|
|
queueWebhookExecution,
|
|
shouldSkipWebhookEvent,
|
|
verifyProviderAuth,
|
|
} from '@/lib/webhooks/processor'
|
|
import { blockExistsInDeployment } from '@/lib/workflows/persistence/utils'
|
|
|
|
const logger = createLogger('WebhookTriggerAPI')
|
|
|
|
export const dynamic = 'force-dynamic'
|
|
export const runtime = 'nodejs'
|
|
export const maxDuration = 60
|
|
|
|
export async function GET(request: NextRequest, { params }: { params: Promise<{ path: string }> }) {
|
|
const requestId = generateRequestId()
|
|
const { path } = await params
|
|
|
|
// Handle provider-specific GET verifications (Microsoft Graph, WhatsApp, etc.)
|
|
const challengeResponse = await handleProviderChallenges({}, request, requestId, path)
|
|
if (challengeResponse) {
|
|
return challengeResponse
|
|
}
|
|
|
|
return new NextResponse('Method not allowed', { status: 405 })
|
|
}
|
|
|
|
export async function POST(
|
|
request: NextRequest,
|
|
{ params }: { params: Promise<{ path: string }> }
|
|
) {
|
|
const requestId = generateRequestId()
|
|
const { path } = await params
|
|
|
|
// Handle provider challenges before body parsing (Microsoft Graph validationToken, etc.)
|
|
const earlyChallenge = await handleProviderChallenges({}, request, requestId, path)
|
|
if (earlyChallenge) {
|
|
return earlyChallenge
|
|
}
|
|
|
|
const parseResult = await parseWebhookBody(request, requestId)
|
|
|
|
// Check if parseWebhookBody returned an error response
|
|
if (parseResult instanceof NextResponse) {
|
|
return parseResult
|
|
}
|
|
|
|
const { body, rawBody } = parseResult
|
|
|
|
const challengeResponse = await handleProviderChallenges(body, request, requestId, path)
|
|
if (challengeResponse) {
|
|
return challengeResponse
|
|
}
|
|
|
|
// Find all webhooks for this path (supports credential set fan-out where multiple webhooks share a path)
|
|
const webhooksForPath = await findAllWebhooksForPath({ requestId, path })
|
|
|
|
if (webhooksForPath.length === 0) {
|
|
logger.warn(`[${requestId}] Webhook or workflow not found for path: ${path}`)
|
|
return new NextResponse('Not Found', { status: 404 })
|
|
}
|
|
|
|
// Process each webhook
|
|
// For credential sets with shared paths, each webhook represents a different credential
|
|
const responses: NextResponse[] = []
|
|
|
|
for (const { webhook: foundWebhook, workflow: foundWorkflow } of webhooksForPath) {
|
|
const authError = await verifyProviderAuth(
|
|
foundWebhook,
|
|
foundWorkflow,
|
|
request,
|
|
rawBody,
|
|
requestId
|
|
)
|
|
if (authError) {
|
|
// For multi-webhook, log and continue to next webhook
|
|
if (webhooksForPath.length > 1) {
|
|
logger.warn(`[${requestId}] Auth failed for webhook ${foundWebhook.id}, continuing to next`)
|
|
continue
|
|
}
|
|
return authError
|
|
}
|
|
|
|
const reachabilityResponse = handleProviderReachabilityTest(foundWebhook, body, requestId)
|
|
if (reachabilityResponse) {
|
|
// Reachability test should return immediately for the first webhook
|
|
return reachabilityResponse
|
|
}
|
|
|
|
let preprocessError: NextResponse | null = null
|
|
try {
|
|
preprocessError = await checkWebhookPreprocessing(foundWorkflow, foundWebhook, requestId)
|
|
if (preprocessError) {
|
|
if (webhooksForPath.length > 1) {
|
|
logger.warn(
|
|
`[${requestId}] Preprocessing failed for webhook ${foundWebhook.id}, continuing to next`
|
|
)
|
|
continue
|
|
}
|
|
return preprocessError
|
|
}
|
|
} catch (error) {
|
|
logger.error(`[${requestId}] Unexpected error during webhook preprocessing`, {
|
|
error: error instanceof Error ? error.message : String(error),
|
|
stack: error instanceof Error ? error.stack : undefined,
|
|
webhookId: foundWebhook.id,
|
|
workflowId: foundWorkflow.id,
|
|
})
|
|
|
|
if (webhooksForPath.length > 1) {
|
|
continue
|
|
}
|
|
|
|
return formatProviderErrorResponse(
|
|
foundWebhook,
|
|
'An unexpected error occurred during preprocessing',
|
|
500
|
|
)
|
|
}
|
|
|
|
if (foundWebhook.blockId) {
|
|
const blockExists = await blockExistsInDeployment(foundWorkflow.id, foundWebhook.blockId)
|
|
if (!blockExists) {
|
|
const preDeploymentResponse = handlePreDeploymentVerification(foundWebhook, requestId)
|
|
if (preDeploymentResponse) {
|
|
return preDeploymentResponse
|
|
}
|
|
|
|
logger.info(
|
|
`[${requestId}] Trigger block ${foundWebhook.blockId} not found in deployment for workflow ${foundWorkflow.id}`
|
|
)
|
|
if (webhooksForPath.length > 1) {
|
|
continue
|
|
}
|
|
return new NextResponse('Trigger block not found in deployment', { status: 404 })
|
|
}
|
|
}
|
|
|
|
if (shouldSkipWebhookEvent(foundWebhook, body, requestId)) {
|
|
continue
|
|
}
|
|
|
|
const response = await queueWebhookExecution(foundWebhook, foundWorkflow, body, request, {
|
|
requestId,
|
|
path,
|
|
})
|
|
responses.push(response)
|
|
}
|
|
|
|
// Return the last successful response, or a combined response for multiple webhooks
|
|
if (responses.length === 0) {
|
|
return new NextResponse('No webhooks processed successfully', { status: 500 })
|
|
}
|
|
|
|
if (responses.length === 1) {
|
|
return responses[0]
|
|
}
|
|
|
|
// For multiple webhooks, return success if at least one succeeded
|
|
logger.info(
|
|
`[${requestId}] Processed ${responses.length} webhooks for path: ${path} (credential set fan-out)`
|
|
)
|
|
return NextResponse.json({
|
|
success: true,
|
|
webhooksProcessed: responses.length,
|
|
})
|
|
}
|