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

com.github._1c_syntax.bsl.languageserver.diagnostics.AbstractExpressionTreeDiagnostic Maven / Gradle / Ivy

Go to download

Language Server Protocol implementation for 1C (BSL) - 1C:Enterprise 8 and OneScript languages.

The newest version!
/*
 * This file is a part of BSL Language Server.
 *
 * Copyright (c) 2018-2024
 * Alexey Sosnoviy , Nikita Fedkin  and contributors
 *
 * SPDX-License-Identifier: LGPL-3.0-or-later
 *
 * BSL Language Server is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 3.0 of the License, or (at your option) any later version.
 *
 * BSL Language Server is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with BSL Language Server.
 */
package com.github._1c_syntax.bsl.languageserver.diagnostics;

import com.github._1c_syntax.bsl.languageserver.context.DocumentContext;
import com.github._1c_syntax.bsl.languageserver.diagnostics.metadata.DiagnosticInfo;
import com.github._1c_syntax.bsl.languageserver.utils.expressiontree.ExpressionTreeBuildingVisitor;
import com.github._1c_syntax.bsl.languageserver.utils.expressiontree.ExpressionTreeVisitor;
import com.github._1c_syntax.bsl.parser.BSLParser;
import com.github._1c_syntax.bsl.parser.BSLParserBaseVisitor;
import lombok.Getter;
import lombok.Setter;
import org.antlr.v4.runtime.tree.ParseTree;
import org.eclipse.lsp4j.Diagnostic;

import java.util.List;

/**
 * Диагностика, анализирующая выражения BSL и предоставляющая для этого Expression Tree
 */
public abstract class AbstractExpressionTreeDiagnostic extends ExpressionTreeVisitor implements BSLDiagnostic {
  @Getter
  @Setter
  protected DiagnosticInfo info;
  protected final DiagnosticStorage diagnosticStorage = new DiagnosticStorage(this);
  protected DocumentContext documentContext;

  @Override
  public final List getDiagnostics(DocumentContext documentContext) {
    this.documentContext = documentContext;
    diagnosticStorage.clearDiagnostics();

    var expressionTreeBuilder = new ExpressionTreeBuilder();
    expressionTreeBuilder.visitFile(documentContext.getAst());

    return diagnosticStorage.getDiagnostics();
  }

  /**
   * При входе в выражение вызывается данный метод.
   * Переопределяя его можно оценить - имеет ли смысл строить дерево выражения, или данное выражение не подходит.
   * Позволяет сократить время на построение дерева, если это не требуется для данного AST.
   *
   * @param ctx - выражение, которое в данный момент посещается.
   * @return - флаг дальнейшего поведения.
   * - если надо прекратить обход в глубину и построить Expression Tree на данном выражении - надо вернуть ACCEPT
   * - если надо пройти дальше и посетить дочерние выражения, не затрагивая данное - надо вернуть VISIT_CHILDREN
   * - если надо пропустить выражение, не ходить глубже и не строить Expression Tree - надо вернуть SKIP
   */
  protected ExpressionVisitorDecision onExpressionEnter(BSLParser.ExpressionContext ctx) {
    return ExpressionVisitorDecision.ACCEPT;
  }

  /**
   * Стратегия по построению дерева выражения на основе выражения AST
   */
  protected enum ExpressionVisitorDecision {

    /**
     * Не обрабатывать выражение
     */
    SKIP,

    /**
     * Обработать данное выражение (построить для него ExpressionTree)
     */
    ACCEPT,

    /**
     * Пропустить данное выражение и обойти вложенные в него выражения
     */
    VISIT_CHILDREN
  }

  private class ExpressionTreeBuilder extends BSLParserBaseVisitor {

    @Override
    public ParseTree visitExpression(BSLParser.ExpressionContext ctx) {

      var treeBuildingVisitor = new ExpressionTreeBuildingVisitor();

      var result = onExpressionEnter(ctx);
      return switch (result) {
        case SKIP -> ctx;
        case ACCEPT -> processExpression(treeBuildingVisitor, ctx);
        case VISIT_CHILDREN -> treeBuildingVisitor.visitChildren(ctx);
      };
    }

    private BSLParser.ExpressionContext processExpression(
      ExpressionTreeBuildingVisitor treeBuildingVisitor,
      BSLParser.ExpressionContext ctx
    ) {
      treeBuildingVisitor.visitExpression(ctx);
      var expressionTree = treeBuildingVisitor.getExpressionTree();
      if (expressionTree != null) { // нашлись выражения в предложенном файле
        visitTopLevelExpression(expressionTree);
      }

      return ctx;
    }
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy