All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ai.vespa.schemals.schemadocument.resolvers.RankExpression.GenericFunction Maven / Gradle / Ivy

There is a newer version: 8.465.15
Show newest version
package ai.vespa.schemals.schemadocument.resolvers.RankExpression;

import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;

import org.eclipse.lsp4j.Diagnostic;
import org.eclipse.lsp4j.DiagnosticSeverity;

import ai.vespa.schemals.common.SchemaDiagnostic;
import ai.vespa.schemals.context.ParseContext;
import ai.vespa.schemals.index.Symbol.SymbolStatus;
import ai.vespa.schemals.index.Symbol.SymbolType;
import ai.vespa.schemals.parser.rankingexpression.ast.identifierStr;
import ai.vespa.schemals.schemadocument.resolvers.RankExpression.argument.Argument;
import ai.vespa.schemals.tree.SchemaNode;
import ai.vespa.schemals.tree.rankingexpression.RankNode;

public class GenericFunction {

    private String name;
    List signatures;
    Set properties;

    public GenericFunction(String name, List signatures) {
        this.name = name;
        this.signatures = signatures;
        this.properties = new HashSet<>();

        for (FunctionSignature signature : signatures) {
            Set addProps = signature.getProperties();
            if (addProps.size() == 0 && properties.size() > 0) {
                 properties.add("");
            } else {
                properties.addAll(addProps);
            }
            
        }
    }

    public GenericFunction(String name, FunctionSignature signature) {
        this(name, new ArrayList<>() {{
            add(signature);
        }});
    }

    public GenericFunction(String name, Argument argument, Set properties) {
        this(name, new FunctionSignature(argument, properties));
    }

    public GenericFunction(String name, List arguments, Set proerties) {
        this(name, new FunctionSignature(arguments, proerties));
    }

    public GenericFunction(String name) {
        this(name, new FunctionSignature());
    }

    public List getSignatures() {
        return List.copyOf(signatures);
    }

    public String getName() { return name; }

    public List handleArgumentList(ParseContext context, RankNode node) {
        List diagnostics = new ArrayList<>();

        Optional property = node.getProperty();
        Optional propertyString = Optional.empty();
        if (property.isPresent()) {
            propertyString = Optional.of(property.get().getText());
        }

        Optional signature = findFunctionSignature(node.getChildren(), propertyString);

        if (signature.isEmpty()) {
            List signatureStrings = signatures.stream()
                                                      .map(func -> func.toString())
                                                      .collect(Collectors.toList());
            String availableSignatures = String.join("\n", signatureStrings);
            String message = "No function matched for that sinature. Available signatures are:\n" + availableSignatures;
            diagnostics.add(new SchemaDiagnostic.Builder()
                .setRange(node.getRange())
                .setMessage(message)
                .setSeverity(DiagnosticSeverity.Error)
                .build());
            return diagnostics;
        }

        diagnostics.addAll(signature.get().handleArgumentList(context, node.getChildren()));

        Set signatureProps = signature.get().getProperties();

        if (property.isEmpty() && (signatureProps.contains("") || signatureProps.size() == 0)) {
            // This is valid
            node.setFunctionSignature(new SpecificFunction(this, signature.get()));
            return diagnostics;
        }
        
        String availableProps = (signatureProps.size() == 0) ? "No one" : String.join(", ", signatureProps);
        if (!property.isPresent()) {
            String message = "The function '" + node.getSchemaNode().getText() + "' must be used with a property. Available properties are: " + availableProps;
            diagnostics.add(new SchemaDiagnostic.Builder()
                .setRange(node.getRange())
                .setMessage(message)
                .setSeverity(DiagnosticSeverity.Error)
                .build());
            return diagnostics;
        }

        if (!properties.contains(property.get().getText())) {
            String message = "Invalid property '" + property.get().getText() + "'. Available properties are: " + availableProps;
            diagnostics.add(new SchemaDiagnostic.Builder()
                .setRange(property.get().getRange())
                .setMessage(message)
                .setSeverity(DiagnosticSeverity.Error)
                .build());
            return diagnostics;
        }

        if (!signature.get().getProperties().contains(property.get().getText())) {
            String message = "This property is not available with with this signature. Available properties are: " + availableProps;
            diagnostics.add(new SchemaDiagnostic.Builder()
                .setRange(property.get().getRange())
                .setMessage(message)
                .setSeverity(DiagnosticSeverity.Warning)
                .build());
        }

        SchemaNode symbolNode = property.get();
        while (!symbolNode.isASTInstance(identifierStr.class) && symbolNode.size() > 0) {
            symbolNode = symbolNode.get(0);
        }

        if (symbolNode.isASTInstance(identifierStr.class)) {
            symbolNode.setSymbol(SymbolType.PROPERTY, context.fileURI());
            symbolNode.setSymbolStatus(SymbolStatus.BUILTIN_REFERENCE);
        }

        node.setFunctionSignature(new SpecificFunction(this, signature.get(), propertyString));

        return diagnostics;
    }

    private static boolean propertyInSet(Optional string, Set propertiySet) {
        if (string.isEmpty() && (
            propertiySet.size() == 0 ||
            propertiySet.contains("")
        )) {
            return true;
        }

        if (string.isEmpty()) return false;

        return propertiySet.contains(string.get());
    }

    private Optional findFunctionSignature(List arguments, Optional property) {

        List bestMatches = new ArrayList<>();
        int maxScore = 0;

        for (FunctionSignature signature : signatures) {
            int score = signature.matchScore(arguments);
            if (score == maxScore) {
                bestMatches.add(signature);
            } else if (score > maxScore) {
                maxScore = score;
                bestMatches = new ArrayList<>() {{
                    add(signature);
                }};
            }
        }

        if (bestMatches.size() == 1) {
            return Optional.of(bestMatches.get(0));
        }

        // Filter by the property first
        List possibleSignatures = new ArrayList<>(bestMatches);

        for (int i = possibleSignatures.size() - 1; i >= 0; i--) {
            if (!propertyInSet(property, possibleSignatures.get(i).getProperties())) {
                possibleSignatures.remove(i);
            }
        }

        if (possibleSignatures.size() == 1) return Optional.of(possibleSignatures.get(0));

        return Optional.empty();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy