fix: expression validation for langchain nodes - skip node repo and expression validation

- Skip node repository lookup for langchain nodes (they have AI-specific validators)
- Skip expression validation for langchain nodes (different expression rules)
- Allow single-node langchain workflows for AI tool validation
- Set both node and nodeName fields in validation response for compatibility

Fixes integration test failures in AI validation suite.

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

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
czlonkowski
2025-10-07 10:36:33 +02:00
parent cffd5e8b2e
commit 36dc8b489c
3 changed files with 31 additions and 6 deletions

View File

@@ -272,13 +272,15 @@ export class WorkflowValidator {
const normalizedType = NodeTypeNormalizer.normalizeToFullForm(singleNode.type);
const isWebhook = normalizedType === 'nodes-base.webhook' ||
normalizedType === 'nodes-base.webhookTrigger';
if (!isWebhook) {
const isLangchainNode = normalizedType.startsWith('nodes-langchain.');
// Langchain nodes can be validated standalone for AI tool purposes
if (!isWebhook && !isLangchainNode) {
result.errors.push({
type: 'error',
message: 'Single-node workflows are only valid for webhook endpoints. Add at least one more connected node to create a functional workflow.'
});
} else if (Object.keys(workflow.connections).length === 0) {
} else if (isWebhook && Object.keys(workflow.connections).length === 0) {
result.warnings.push({
type: 'warning',
message: 'Webhook node has no connections. Consider adding nodes to process the webhook data.'
@@ -961,6 +963,13 @@ export class WorkflowValidator {
for (const node of workflow.nodes) {
if (node.disabled || this.isStickyNote(node)) continue;
// Skip expression validation for langchain nodes
// They have AI-specific validators and different expression rules
const normalizedType = NodeTypeNormalizer.normalizeToFullForm(node.type);
if (normalizedType.startsWith('nodes-langchain.')) {
continue;
}
// Create expression context
const context = {
availableNodes: nodeNames.filter(n => n !== node.name),