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

com.github._1c_syntax.bsl.languageserver.codeactions.DisableDiagnosticTriggeringSupplier Maven / Gradle / Ivy

/*
 * This file is a part of BSL Language Server.
 *
 * Copyright (c) 2018-2022
 * 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.codeactions;

import com.github._1c_syntax.bsl.languageserver.configuration.LanguageServerConfiguration;
import com.github._1c_syntax.bsl.languageserver.context.DocumentContext;
import com.github._1c_syntax.bsl.languageserver.diagnostics.metadata.DiagnosticCode;
import com.github._1c_syntax.bsl.languageserver.utils.Ranges;
import com.github._1c_syntax.bsl.languageserver.utils.Resources;
import org.antlr.v4.runtime.Token;
import org.eclipse.lsp4j.CodeAction;
import org.eclipse.lsp4j.CodeActionKind;
import org.eclipse.lsp4j.CodeActionParams;
import org.eclipse.lsp4j.Diagnostic;
import org.eclipse.lsp4j.Range;
import org.eclipse.lsp4j.TextEdit;
import org.eclipse.lsp4j.WorkspaceEdit;
import org.jetbrains.annotations.NotNull;
import org.springframework.stereotype.Component;

import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
import java.util.Map;
import java.util.function.Function;
import java.util.stream.Collectors;

import static com.github._1c_syntax.bsl.languageserver.providers.DiagnosticProvider.SOURCE;


@Component
public class DisableDiagnosticTriggeringSupplier implements CodeActionSupplier {

  private static final String ALL_DIAGNOSTIC_NAME = "";
  private final LanguageServerConfiguration languageServerConfiguration;

  public DisableDiagnosticTriggeringSupplier(LanguageServerConfiguration languageServerConfiguration) {
    this.languageServerConfiguration = languageServerConfiguration;
  }

  /**
   * При необходимости создает {@code CodeAction} для создания служебного комментария отключающего срабатывание
   * диагностики.
   * Может быть в трех вариантах:
   * 1. Отключаются срабатывания в конкретной строке
   * 2. Отключаются срабатывания в области между парой комментариев
   * 3. Отключаются срабатывания во всем файле
   *
   * @param params          параметры вызова генерации {@code codeAction}
   * @param documentContext представление программного модуля
   * @return {@code List} если модуль не содержит всех стандартных областей,
   * пустой {@code List} если генерация областей не требуется
   */
  @Override
  public List getCodeActions(CodeActionParams params, DocumentContext documentContext) {
    List result = new ArrayList<>();

    if (!params.getContext().getDiagnostics().isEmpty()) {
      if (params.getRange().getStart() != null && params.getRange().getEnd() != null) {
        var selectedLineNumber = params.getRange().getEnd().getLine() + 1;

        documentContext
          .getTokens()
          .stream()
          .filter(token -> token.getLine() == selectedLineNumber)
          .max(Comparator.comparingInt(Token::getCharPositionInLine))
          .ifPresent(token -> {
            if (params.getRange().getStart().getLine() == params.getRange().getEnd().getLine()) {
              result.addAll(getDisableActionForLine(params, documentContext, token));
            } else {
              result.addAll(getDisableActionForRange(params, documentContext, token));
            }
          }
        );
      }

      result.addAll(
        actionDisableDiagnostic(
          name -> createCodeAction(
            getMessage("file", name),
            createInFileTextEdits(":" + name),
            documentContext
          ),
          params
        )
      );
    }

    result.add(
      createCodeAction(
        getMessage("fileAll"),
        createInFileTextEdits(ALL_DIAGNOSTIC_NAME),
        documentContext
      )
    );
    return new ArrayList<>(result);
  }

  private List getDisableActionForLine(
    CodeActionParams params,
    DocumentContext documentContext,
    Token lastToken
  ) {
    var result = actionDisableDiagnostic(
      name -> createCodeAction(
        getMessage("line", name),
        createInLineTextEdits(":" + name, lastToken, params),
        documentContext
      ),
      params
    );
    result.add(
      createCodeAction(
        getMessage("lineAll"),
        createInLineTextEdits(ALL_DIAGNOSTIC_NAME, lastToken, params),
        documentContext
      )
    );
    return result;
  }

  private List getDisableActionForRange(
    CodeActionParams params,
    DocumentContext documentContext,
    Token lastToken
  ) {
    var result = actionDisableDiagnostic(
      name -> createCodeAction(
        getMessage("range", name),
        createInRegionTextEdits(":" + name, lastToken, params),
        documentContext
      ),
      params
    );
    result.add(
      createCodeAction(
        getMessage("rangeAll"),
        createInRegionTextEdits(ALL_DIAGNOSTIC_NAME, lastToken, params),
        documentContext
      )
    );
    return result;
  }

  private List actionDisableDiagnostic(Function func, CodeActionParams params) {
    return params.getContext()
      .getDiagnostics()
      .stream()
      .filter(diagnostic -> SOURCE.equals(diagnostic.getSource()))
      .map(Diagnostic::getCode)
      .map(DiagnosticCode::getStringValue)
      .distinct()
      .map(func)
      .collect(Collectors.toList());
  }

  private List createInRegionTextEdits(String diagnosticName, Token last, CodeActionParams params) {
    List edits = new ArrayList<>();

    Range disableRange = Ranges.create(
      params.getRange().getStart().getLine(),
      0,
      params.getRange().getStart().getLine(),
      0
    );
    TextEdit disableTextEdit = new TextEdit(disableRange, String.format("// BSLLS%s-off%n", diagnosticName));
    edits.add(disableTextEdit);

    Range enableRange = Ranges.create(
      params.getRange().getEnd().getLine(),
      last.getCharPositionInLine() + last.getText().length(),
      params.getRange().getEnd().getLine(),
      last.getCharPositionInLine() + last.getText().length()
    );
    TextEdit enableTextEdit = new TextEdit(enableRange, String.format("%n// BSLLS%s-on%n", diagnosticName));
    edits.add(enableTextEdit);
    return edits;
  }

  private List createInFileTextEdits(String diagnosticName) {
    TextEdit textEdit = new TextEdit(
      Ranges.create(0, 0, 0, 0),
      String.format("// BSLLS%s-off%n", diagnosticName)
    );
    return Collections.singletonList(textEdit);
  }

  @NotNull
  private CodeAction createCodeAction(String title, List edits, DocumentContext documentContext) {
    Map> changes = Map.of(documentContext.getUri().toString(), edits);
    WorkspaceEdit edit = new WorkspaceEdit();
    edit.setChanges(changes);

    CodeAction codeAction = new CodeAction(title);
    codeAction.setDiagnostics(new ArrayList<>());
    codeAction.setKind(CodeActionKind.Refactor);
    codeAction.setEdit(edit);
    return codeAction;
  }

  private String getMessage(String key) {
    return Resources.getResourceString(languageServerConfiguration.getLanguage(), this.getClass(), key);
  }

  private String getMessage(String key, Object... args) {
    return Resources.getResourceString(languageServerConfiguration.getLanguage(), this.getClass(), key, args);
  }

  private static List createInLineTextEdits(String diagnosticName, Token last, CodeActionParams params) {
    Range range = Ranges.create(
      params.getRange().getStart().getLine(),
      last.getCharPositionInLine() + last.getText().length(),
      params.getRange().getStart().getLine(),
      last.getCharPositionInLine() + last.getText().length()
    );

    TextEdit textEdit = new TextEdit(range, String.format(" // BSLLS%s-off", diagnosticName));
    return Collections.singletonList(textEdit);
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy