using System.Globalization; using NodePipeline.Abstractions; using NodePipeline.Abstractions.Exceptions.Validation; using NodePipeline.Abstractions.Interfaces; using NodePipeline.Abstractions.Interfaces.Nodes; using NodePipeline.Abstractions.Interfaces.Validation; using NodePipeline.Abstractions.Models.Validation; using NodePipeline.Abstractions.Validators; using NodePipeline.Configuration.Abstractions.Models.Execute; using NodePipeline.Engine.Abstractions; using NodePipeline.Engine.NodeFields; using NodePipeline.Engine.Utils; using NodePipeline.Engine.Abstractions.Validation; // ReSharper disable CheckNamespace // ReSharper disable RedundantNameQualifier // ReSharper disable UnusedParameter.Local #nullable enable namespace NodePipeline.Engine.Generated; public sealed class GeneratedNodeValidator : IPipelineNodeValidator { private Dictionary? _allPorts; public Dictionary NodeValidators { get; set; } = []; public Dictionary NodeFieldValidators { get; set; } = []; public IPipelineLocalizationProvider PipelineLocalizationProvider { get; set; } = new PipelineLocalizationProvider(); public NodeValidationResult ValidateNode(string pipelineId, INodeFactory nodeFactory, NodeConfig nodeConfig, List allNodes, Dictionary parameters, CultureInfo cultureInfo) { var node = nodeFactory.CreateNode(pipelineId, nodeConfig); nodeFactory.SetNodeParametersValues(pipelineId, node, nodeConfig.Id, parameters); var nodeConnections = allNodes.SelectMany(q => q.Inputs .Select(x => new KeyValuePair(GetNodeInputKey(q.Id, x.Key), x.Value)) ).ToDictionary(q => q.Key, q => q.Value); var ports = GetAllPorts(pipelineId, nodeFactory, allNodes); switch (node) { default: throw new UnknownNodeTypeException(nodeConfig.Type, nodeConfig.Id); } } private IReadOnlyDictionary GetAllPorts(string pipelineId, INodeFactory nodeFactory, List allNodes) { var nodeTypes = allNodes.Select(q => q.Type) .Distinct() .ToHashSet(); var nodeList = allNodes .GroupBy(q => q.Type, (type, g) => new { Type = type, NodeIds = g.Select(q => q.Id) }) .ToDictionary(q => q.Type, q => q.NodeIds); _allPorts ??= nodeFactory.GetAllPorts(pipelineId, nodeTypes, nodeList); return _allPorts.AsReadOnly(); } private INodeFieldValidator GetNodeFieldValidator(Type validatorType) { if (!NodeFieldValidators.TryGetValue(validatorType, out var validator) || validator is not INodeFieldValidator genericValidator) throw new UserDefinedNodeFieldValidatorNotFoundException(validatorType.FullName ?? validatorType.Name); return genericValidator; } private ValidationResult ValidateNodePorts(string pipelineId, INodeFactory nodeFactory, string nodeType, string nodeId, IReadOnlyDictionary nodePortConnections, IReadOnlyDictionary ports, out Dictionary> inputPortValidationResults, CultureInfo cultureInfo) { var result = ValidationResult.Valid; inputPortValidationResults = new Dictionary>(); var inputPorts = nodeFactory.GetNodeInputPorts(pipelineId, nodeType, nodeId); foreach (var inputPort in inputPorts) { var code = NodeFieldCodeGenerator.GenerateFieldCode(nodeId, inputPort.Name, inputPort.Input ? FieldDirection.Input : FieldDirection.Output); inputPortValidationResults[code] = []; var hasConnection = nodePortConnections.TryGetValue(GetNodeInputKey(nodeId, inputPort.Name), out var connection); if (inputPort.Required && !hasConnection) { inputPortValidationResults[code].Add(new NodeFieldValidationResult(ValidationResult.HasErrors, PipelineLocalizationProvider.GetLocalizedString(Constants.ValidationResults.ErrorMessages.RequiredInputPortHasNoConnectionError, cultureInfo, inputPort.Name, nodeId))); UpdateValidationResult(ValidationResult.HasErrors, ref result); continue; } if (!hasConnection || connection == null) continue; var portValidationResult = ValidateInputPort(nodeId, inputPort, connection, ports, cultureInfo); inputPortValidationResults[code].Add(portValidationResult); UpdateValidationResult(portValidationResult.Result, ref result); } return result; } private NodeFieldValidationResult ValidateInputPort(string nodeId, NodePortInfo inputPort, InputSource connection, IReadOnlyDictionary ports, CultureInfo cultureInfo) { if (!ports.TryGetValue(new NodePortKey(connection.NodeId, connection.OutputName, false), out var outputPort)) { return new NodeFieldValidationResult(ValidationResult.HasErrors, PipelineLocalizationProvider .GetLocalizedString(Constants.ValidationResults.ErrorMessages.OutputPortNotFoundError, cultureInfo, connection.OutputName, connection.NodeId, inputPort.Name, nodeId)); } if (inputPort.Type != outputPort.Type) { return new NodeFieldValidationResult(ValidationResult.HasErrors, PipelineLocalizationProvider .GetLocalizedString(Constants.ValidationResults.ErrorMessages.PortsTypeMismatchError, cultureInfo, inputPort.Name, nodeId, inputPort.Type, connection.OutputName, connection.NodeId, outputPort.Type)); } if (inputPort.IsNullable == outputPort.IsNullable) { return SuccessNodeFieldValidationResult(); } if (!inputPort.IsNullable && outputPort.IsNullable) { return new NodeFieldValidationResult( ValidationResult.HasWarnings, PipelineLocalizationProvider .GetLocalizedString(Constants.ValidationResults.ErrorMessages.NullableAndNonNullablePortsConnectionWarning, cultureInfo, inputPort.Name, nodeId, inputPort.Type, connection.OutputName, connection.NodeId, outputPort.Type)); } return SuccessNodeFieldValidationResult(); } private static string GetNodeInputKey(string nodeId, string inputPortName) { return string.Concat(nodeId, "_", inputPortName); } private INodeValidator GetNodeValidator(Type validatorType) { if (!NodeValidators.TryGetValue(validatorType, out var validator)) throw new UserDefinedNodeValidatorNotFoundException(validatorType.FullName ?? validatorType.Name); return validator; } private NodeFieldValidationResult CheckRequiredNodeParameter(string nodeId, NodeField nodeField, string nodeParameterName, CultureInfo cultureInfo) { return nodeField.Value is not null ? new NodeFieldValidationResult(ValidationResult.Valid, null) : new NodeFieldValidationResult(ValidationResult.HasErrors, PipelineLocalizationProvider .GetLocalizedString(Constants.ValidationResults.ErrorMessages.RequiredNodeParameterValidationError, cultureInfo, nodeParameterName, nodeId)); } private static NodeValidationResult SuccessNodeValidationResult(bool hasWarnings, Dictionary> fieldResults, Dictionary> portResults) => new(hasWarnings ? ValidationResult.HasWarnings : ValidationResult.Valid, string.Empty, fieldResults.AsReadOnly(), portResults.AsReadOnly()); private NodeValidationResult GetErrorValidationResult(ValidationResult validationResult, string nodeId, Dictionary> fieldResults, Dictionary> portResults, CultureInfo cultureInfo) => new(validationResult, PipelineLocalizationProvider.GetLocalizedString(Constants.ValidationResults.ErrorMessages.NodeValidationError, cultureInfo, nodeId), fieldResults.AsReadOnly(), portResults.AsReadOnly()); private static NodeFieldValidationResult SuccessNodeFieldValidationResult() => new NodeFieldValidationResult(ValidationResult.Valid, null); private static void UpdateValidationResult(ValidationResult validationResult, ref ValidationResult parametersValidationResult) { if (validationResult == parametersValidationResult || validationResult == ValidationResult.Valid) return; parametersValidationResult = validationResult switch { ValidationResult.HasWarnings when parametersValidationResult == ValidationResult.Valid => ValidationResult.HasWarnings, ValidationResult.HasErrors when parametersValidationResult != ValidationResult.HasErrors => ValidationResult.HasErrors, _ => parametersValidationResult }; } } #nullable restore