ai.vespa.schemals.schemadocument.resolvers.IndexingLanguageResolver Maven / Gradle / Ivy
package ai.vespa.schemals.schemadocument.resolvers;
import java.util.List;
import java.util.Optional;
import org.eclipse.lsp4j.Diagnostic;
import org.eclipse.lsp4j.DiagnosticSeverity;
import com.yahoo.vespa.indexinglanguage.expressions.StatementExpression;
import ai.vespa.schemals.common.SchemaDiagnostic;
import ai.vespa.schemals.context.ParseContext;
import ai.vespa.schemals.index.FieldIndex.IndexingType;
import ai.vespa.schemals.index.Symbol;
import ai.vespa.schemals.index.Symbol.SymbolStatus;
import ai.vespa.schemals.index.Symbol.SymbolType;
import ai.vespa.schemals.parser.ast.identifierStr;
import ai.vespa.schemals.parser.ast.indexingElm;
import ai.vespa.schemals.parser.indexinglanguage.ast.ATTRIBUTE;
import ai.vespa.schemals.parser.indexinglanguage.ast.DOT;
import ai.vespa.schemals.parser.indexinglanguage.ast.INDEX;
import ai.vespa.schemals.parser.indexinglanguage.ast.SUMMARY;
import ai.vespa.schemals.parser.indexinglanguage.ast.fieldName;
import ai.vespa.schemals.parser.indexinglanguage.ast.inputExp;
import ai.vespa.schemals.parser.indexinglanguage.ast.script;
import ai.vespa.schemals.parser.indexinglanguage.ast.statement;
import ai.vespa.schemals.tree.CSTUtils;
import ai.vespa.schemals.tree.SchemaNode;
/**
* IndexingLanguageResolver
*/
public class IndexingLanguageResolver {
private ParseContext context;
private Symbol containingFieldDefinition;
/*
* Given a node containing an ILSCRIPT, resolve the language
*/
public static void resolveIndexingLanguage(ParseContext context, SchemaNode indexingLanguageNode, List diagnostics) {
SchemaNode fieldDefinitionNode = indexingLanguageNode.getParent().getParent().get(1);
while (fieldDefinitionNode.getNextSibling() != null && fieldDefinitionNode.getNextSibling().isASTInstance(identifierStr.class) && fieldDefinitionNode.getNextSibling().hasSymbol()) {
fieldDefinitionNode = fieldDefinitionNode.getNextSibling();
}
Symbol fieldDefinitionSymbol = fieldDefinitionNode.getSymbol();
if (fieldDefinitionSymbol.getStatus() == SymbolStatus.REFERENCE) {
Optional definition = context.schemaIndex().getSymbolDefinition(fieldDefinitionSymbol);
if (definition.isPresent())fieldDefinitionSymbol = definition.get();
}
if (fieldDefinitionSymbol.getStatus() != SymbolStatus.DEFINITION || fieldDefinitionSymbol.getType() != SymbolType.FIELD) {
return;
}
if (indexingLanguageNode.isASTInstance(indexingElm.class)) {
var expression = ((indexingElm)indexingLanguageNode.getOriginalSchemaNode()).expression;
if (expression != null && expression.requiredInputType() != null && !context.fieldIndex().getIsInsideDoc(fieldDefinitionSymbol)) {
diagnostics.add(new SchemaDiagnostic.Builder()
.setRange(indexingLanguageNode.get(0).getRange())
.setMessage(
"Expected " + expression.requiredInputType().getName() + " input, but no input is specified. " +
"Fields defined outside the document must start with indexing statements explicitly collecting input.")
.setSeverity(DiagnosticSeverity.Error)
.build());
}
}
IndexingLanguageResolver instance = new IndexingLanguageResolver(context, fieldDefinitionSymbol);
instance.traverse(indexingLanguageNode, diagnostics);
}
private IndexingLanguageResolver(ParseContext context, Symbol containingFieldDefinition) {
this.context = context;
this.containingFieldDefinition = containingFieldDefinition;
}
private void traverse(SchemaNode node, List diagnostics) {
if (node.isASTInstance(ATTRIBUTE.class)) {
context.fieldIndex().addFieldIndexingType(containingFieldDefinition, IndexingType.ATTRIBUTE);
}
if (node.isASTInstance(INDEX.class)) {
context.fieldIndex().addFieldIndexingType(containingFieldDefinition, IndexingType.INDEX);
}
if (node.isASTInstance(SUMMARY.class)) {
context.fieldIndex().addFieldIndexingType(containingFieldDefinition, IndexingType.SUMMARY);
}
if (node.isASTInstance(ai.vespa.schemals.parser.indexinglanguage.ast.identifierStr.class) && node.getParent().isASTInstance(inputExp.class)) {
Optional scope = CSTUtils.findScope(node);
if (scope.isPresent()) {
Optional fieldDefinition = context.schemaIndex().findSymbol(scope.get(), SymbolType.FIELD, node.getText());
if (fieldDefinition.isPresent()) {
if (!context.fieldIndex().getIsInsideDoc(fieldDefinition.get())) {
diagnostics.add(new SchemaDiagnostic.Builder()
.setRange(node.getRange())
.setMessage("Input references field " + node.getText() + " which is not a document field")
.setSeverity(DiagnosticSeverity.Error)
.build()
);
}
// we insert it as a reference anyways to enable go to definition
node.setSymbol(SymbolType.FIELD, context.fileURI(), scope.get());
node.setSymbolStatus(SymbolStatus.REFERENCE);
context.schemaIndex().insertSymbolReference(fieldDefinition.get(), node.getSymbol());
} else {
diagnostics.add(new SchemaDiagnostic.Builder()
.setRange(node.getRange())
.setMessage("Undefined symbol " + node.getText())
.setSeverity(DiagnosticSeverity.Error)
.build()
);
}
}
}
if (node.isASTInstance(statement.class)
&& (node.getParent().isASTInstance(indexingElm.class) || node.getParent().isASTInstance(script.class))) {
// Top level statement of field outside document should have some input
statement originalNode = (statement)node.getOriginalIndexingNode();
StatementExpression expression = originalNode.expression;
if (expression != null && expression.requiredInputType() != null && !context.fieldIndex().getIsInsideDoc(containingFieldDefinition)) {
diagnostics.add(new SchemaDiagnostic.Builder()
.setRange(node.getRange())
.setMessage("Expected " + expression.requiredInputType().getName() + " input, but no input is specified. Fields defined outside the document must start with indexing statements explicitly collecting input.")
.setSeverity(DiagnosticSeverity.Error)
.build());
}
}
if (node.getParent().isASTInstance(fieldName.class) && node.isASTInstance(ai.vespa.schemals.parser.indexinglanguage.ast.identifierStr.class)) {
Optional scope = CSTUtils.findScope(node);
SymbolType type = SymbolType.FIELD;
if (node.getPreviousSibling() != null && node.getPreviousSibling().isASTInstance(DOT.class)) {
type = SymbolType.SUBFIELD;
}
if (scope.isPresent()) {
node.setSymbol(type, context.fileURI(), scope.get());
} else {
node.setSymbol(type, context.fileURI());
}
node.setSymbolStatus(SymbolStatus.UNRESOLVED);
}
for (SchemaNode child : node) {
traverse(child, diagnostics);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy