From fe53693df93b4618a493ecbb8270dcddc487ee6b Mon Sep 17 00:00:00 2001 From: Nikita Fedkin Date: Tue, 12 Nov 2024 23:09:45 +0100 Subject: [PATCH 01/11] =?UTF-8?q?=D0=92=D1=81=D0=BF=D0=BB=D1=8B=D0=B2?= =?UTF-8?q?=D0=B0=D1=8E=D1=89=D0=B0=D1=8F=20=D0=BF=D0=BE=D0=B4=D1=81=D0=BA?= =?UTF-8?q?=D0=B0=D0=B7=D0=BA=D0=B0=20=D0=B8=20=D0=BF=D0=B5=D1=80=D0=B5?= =?UTF-8?q?=D1=85=D0=BE=D0=B4=20=D0=BA=20=D0=BE=D0=BF=D1=80=D0=B5=D0=B4?= =?UTF-8?q?=D0=B5=D0=BB=D0=B5=D0=BD=D0=B8=D1=8F=D0=BC=20=D0=B4=D0=BB=D1=8F?= =?UTF-8?q?=20=D0=B0=D0=BD=D0=BD=D0=BE=D1=82=D0=B0=D1=86=D0=B8=D0=B9=20?= =?UTF-8?q?=D0=B2=20OneScript?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../context/symbol/AnnotationSymbol.java | 86 +++++ .../AnnotationSymbolMarkupContentBuilder.java | 106 ++++++ .../hover/DescriptionFormatter.java | 356 ++++++++++++++++++ .../MethodSymbolMarkupContentBuilder.java | 262 +------------ .../VariableSymbolMarkupContentBuilder.java | 57 +-- ...rceDefinedMethodCallInlayHintSupplier.java | 11 +- .../references/AnnotationReferenceFinder.java | 79 ++++ ...ies => DescriptionFormatter_en.properties} | 4 +- ...ies => DescriptionFormatter_ru.properties} | 4 +- ...leSymbolMarkupContentBuilder_en.properties | 2 - ...leSymbolMarkupContentBuilder_ru.properties | 2 - .../AnnotationReferenceFinderTest.java | 60 +++ .../references/AnnotationReferenceFinder.os | 6 + 13 files changed, 727 insertions(+), 308 deletions(-) create mode 100644 src/main/java/com/github/_1c_syntax/bsl/languageserver/context/symbol/AnnotationSymbol.java create mode 100644 src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/AnnotationSymbolMarkupContentBuilder.java create mode 100644 src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter.java create mode 100644 src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java rename src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/{MethodSymbolMarkupContentBuilder_en.properties => DescriptionFormatter_en.properties} (75%) rename src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/{MethodSymbolMarkupContentBuilder_ru.properties => DescriptionFormatter_ru.properties} (77%) delete mode 100644 src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder_en.properties delete mode 100644 src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder_ru.properties create mode 100644 src/test/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinderTest.java create mode 100644 src/test/resources/references/AnnotationReferenceFinder.os diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/context/symbol/AnnotationSymbol.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/context/symbol/AnnotationSymbol.java new file mode 100644 index 00000000000..5a7cd0851c3 --- /dev/null +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/context/symbol/AnnotationSymbol.java @@ -0,0 +1,86 @@ +/* + * 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.context.symbol; + +import com.github._1c_syntax.bsl.languageserver.context.DocumentContext; +import com.github._1c_syntax.bsl.languageserver.context.symbol.annotations.Annotation; +import com.github._1c_syntax.bsl.languageserver.context.symbol.description.MethodDescription; +import com.github._1c_syntax.bsl.languageserver.context.symbol.variable.VariableDescription; +import lombok.Builder; +import lombok.EqualsAndHashCode; +import lombok.Getter; +import lombok.Setter; +import lombok.ToString; +import lombok.Value; +import lombok.experimental.NonFinal; +import org.eclipse.lsp4j.Range; +import org.eclipse.lsp4j.SymbolKind; + +import java.util.ArrayList; +import java.util.List; +import java.util.Optional; + +@Value +@Builder +@EqualsAndHashCode(onlyExplicitlyIncluded = true) +@ToString(exclude = {"children", "parent"}) +public class AnnotationSymbol implements SourceDefinedSymbol, Describable { + + String name; + + SymbolKind symbolKind; + + @EqualsAndHashCode.Include + DocumentContext owner; + + Range range; + + @EqualsAndHashCode.Include + Range selectionRange; + + @Setter + @NonFinal + @Builder.Default + Optional parent = Optional.empty(); + + @Builder.Default + List children = new ArrayList<>(); + + Optional description; + + @Override + public void accept(SymbolTreeVisitor visitor) { + // no-op + } + + public static AnnotationSymbol from(String name, MethodSymbol methodSymbol) { + return AnnotationSymbol.builder() + .name(name) + .symbolKind(SymbolKind.TypeParameter) + .owner(methodSymbol.getOwner()) + .range(methodSymbol.getRange()) + .selectionRange(methodSymbol.getSelectionRange()) + .description(methodSymbol.getDescription()) + .parent(Optional.of(methodSymbol)) + .build(); + } +} diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/AnnotationSymbolMarkupContentBuilder.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/AnnotationSymbolMarkupContentBuilder.java new file mode 100644 index 00000000000..39abf339813 --- /dev/null +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/AnnotationSymbolMarkupContentBuilder.java @@ -0,0 +1,106 @@ +/* + * 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.hover; + +import com.github._1c_syntax.bsl.languageserver.context.symbol.AnnotationSymbol; +import com.github._1c_syntax.bsl.languageserver.context.symbol.MethodSymbol; +import com.github._1c_syntax.bsl.languageserver.context.symbol.ParameterDefinition; +import com.github._1c_syntax.bsl.languageserver.context.symbol.description.MethodDescription; +import com.github._1c_syntax.bsl.languageserver.context.symbol.description.ParameterDescription; +import com.github._1c_syntax.bsl.languageserver.context.symbol.description.TypeDescription; +import com.github._1c_syntax.bsl.languageserver.utils.MdoRefBuilder; +import com.github._1c_syntax.bsl.languageserver.utils.Resources; +import lombok.RequiredArgsConstructor; +import org.eclipse.lsp4j.MarkupContent; +import org.eclipse.lsp4j.MarkupKind; +import org.eclipse.lsp4j.SymbolKind; +import org.springframework.stereotype.Component; + +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.StringJoiner; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +/** + * Построитель контента для всплывающего окна для {@link AnnotationSymbol}. + */ +@Component +@RequiredArgsConstructor +public class AnnotationSymbolMarkupContentBuilder implements MarkupContentBuilder { + + private final DescriptionFormatter descriptionFormatter; + + @Override + public MarkupContent getContent(AnnotationSymbol symbol) { + var maybeMethodSymbol = symbol.getParent(); + if (maybeMethodSymbol.filter(MethodSymbol.class::isInstance).isEmpty()) { + return new MarkupContent(MarkupKind.MARKDOWN, ""); + } + + var markupBuilder = new StringJoiner("\n"); + var methodSymbol = (MethodSymbol) maybeMethodSymbol.get(); + + // сигнатура + // местоположение метода + // описание метода + // параметры + // примеры + // варианты вызова + + // сигнатура + String signature = descriptionFormatter.getSignature(symbol, methodSymbol); + descriptionFormatter.addSectionIfNotEmpty(markupBuilder, signature); + + // местоположение метода + String methodLocation = descriptionFormatter.getLocation(methodSymbol); + descriptionFormatter.addSectionIfNotEmpty(markupBuilder, methodLocation); + + // описание метода + String purposeSection = descriptionFormatter.getPurposeSection(methodSymbol); + descriptionFormatter.addSectionIfNotEmpty(markupBuilder, purposeSection); + + // параметры + String parametersSection = descriptionFormatter.getParametersSection(methodSymbol); + descriptionFormatter.addSectionIfNotEmpty(markupBuilder, parametersSection); + + // примеры + String examplesSection = descriptionFormatter.getExamplesSection(methodSymbol); + descriptionFormatter.addSectionIfNotEmpty(markupBuilder, examplesSection); + + // варианты вызова + String callOptionsSection = descriptionFormatter.getCallOptionsSection(methodSymbol); + descriptionFormatter.addSectionIfNotEmpty(markupBuilder, callOptionsSection); + + String content = markupBuilder.toString(); + + return new MarkupContent(MarkupKind.MARKDOWN, content); + } + + @Override + public SymbolKind getSymbolKind() { + return SymbolKind.TypeParameter; + } + +} diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter.java new file mode 100644 index 00000000000..d483c6f7a04 --- /dev/null +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter.java @@ -0,0 +1,356 @@ +/* + * 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.hover; + +import com.github._1c_syntax.bsl.languageserver.context.symbol.AnnotationSymbol; +import com.github._1c_syntax.bsl.languageserver.context.symbol.MethodSymbol; +import com.github._1c_syntax.bsl.languageserver.context.symbol.ParameterDefinition; +import com.github._1c_syntax.bsl.languageserver.context.symbol.VariableSymbol; +import com.github._1c_syntax.bsl.languageserver.context.symbol.description.MethodDescription; +import com.github._1c_syntax.bsl.languageserver.context.symbol.description.ParameterDescription; +import com.github._1c_syntax.bsl.languageserver.context.symbol.description.TypeDescription; +import com.github._1c_syntax.bsl.languageserver.utils.MdoRefBuilder; +import com.github._1c_syntax.bsl.languageserver.utils.Resources; +import lombok.RequiredArgsConstructor; +import org.eclipse.lsp4j.SymbolKind; +import org.springframework.stereotype.Component; + +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Optional; +import java.util.StringJoiner; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +@Component +@RequiredArgsConstructor +public class DescriptionFormatter { + + private static final String PROCEDURE_KEY = "procedure"; + private static final String FUNCTION_KEY = "function"; + private static final String ANNOTATION_KEY = "annotation"; + private static final String EXPORT_KEY = "export"; + private static final String VAL_KEY = "val"; + private static final String VARIABLE_KEY = "var"; + private static final String PARAMETERS_KEY = "parameters"; + private static final String RETURNED_VALUE_KEY = "returnedValue"; + private static final String EXAMPLES_KEY = "examples"; + private static final String CALL_OPTIONS_KEY = "callOptions"; + private static final String PARAMETER_TEMPLATE = "* **%s**: %s"; + + private final Resources resources; + + public void addSectionIfNotEmpty(StringJoiner markupBuilder, String newContent) { + if (!newContent.isEmpty()) { + markupBuilder.add(newContent); + markupBuilder.add(""); + markupBuilder.add("---"); + } + } + + public String getPurposeSection(MethodSymbol methodSymbol) { + return methodSymbol.getDescription() + .map(MethodDescription::getPurposeDescription) + .orElse(""); + } + + public String getParametersSection(MethodSymbol methodSymbol) { + var result = new StringJoiner(" \n"); // два пробела + methodSymbol.getParameters().forEach(parameterDefinition -> + result.add(parameterToString(parameterDefinition)) + ); + + var parameters = result.toString(); + + if (!parameters.isBlank()) { + var parametersSection = new StringJoiner("\n"); + String header = "**" + getResourceString(PARAMETERS_KEY) + ":**"; + parametersSection.add(header); + parametersSection.add(""); + parametersSection.add(parameters); + return parametersSection.toString(); + } + + return ""; + } + + public String getReturnedValueSection(MethodSymbol methodSymbol) { + var result = new StringJoiner(" \n"); // два пробела + methodSymbol.getDescription().ifPresent((MethodDescription methodDescription) -> { + Map typesMap = typesToMap(methodDescription.getReturnedValue(), 0); + result.add(typesMapToString(typesMap, 1)); + }); + + var returnedValue = result.toString(); + + if (!returnedValue.isEmpty()) { + returnedValue = "**" + getResourceString(RETURNED_VALUE_KEY) + ":**\n\n" + returnedValue; + } + + return returnedValue; + } + + public String getExamplesSection(MethodSymbol methodSymbol) { + var examples = methodSymbol.getDescription() + .map(MethodDescription::getExamples) + .orElseGet(Collections::emptyList); + return getSectionWithCodeFences(examples, EXAMPLES_KEY); + } + + public String getCallOptionsSection(MethodSymbol methodSymbol) { + var callOptions = methodSymbol.getDescription() + .map(MethodDescription::getCallOptions) + .orElseGet(Collections::emptyList); + return getSectionWithCodeFences(callOptions, CALL_OPTIONS_KEY); + } + + public String getSectionWithCodeFences(List codeBlocks, String resourceKey) { + String codeFences = codeBlocks + .stream() + .map(codeBlock -> "```bsl\n" + codeBlock + "\n```") + .collect(Collectors.joining("\n")); + + if (!codeFences.isEmpty()) { + codeFences = "**" + getResourceString(resourceKey) + ":**\n\n" + codeFences; + } + + return codeFences; + } + + public String getLocation(MethodSymbol symbol) { + var documentContext = symbol.getOwner(); + var startPosition = symbol.getSelectionRange().getStart(); + String mdoRef = MdoRefBuilder.getMdoRef(documentContext); + + return String.format( + "[%s](%s#%d)", + mdoRef, + documentContext.getUri(), + startPosition.getLine() + 1 + ); + } + + public String getLocation(VariableSymbol symbol) { + var documentContext = symbol.getOwner(); + var startPosition = symbol.getSelectionRange().getStart(); + String mdoRef = MdoRefBuilder.getMdoRef(documentContext); + + String parentPostfix = symbol.getRootParent(SymbolKind.Method) + .map(sourceDefinedSymbol -> "." + sourceDefinedSymbol.getName()) + .orElse(""); + mdoRef += parentPostfix; + + return String.format( + "[%s](%s#%d)", + mdoRef, + documentContext.getUri(), + startPosition.getLine() + 1 + ); + } + + public String getSignature(MethodSymbol methodSymbol) { + String signatureTemplate = "```bsl\n%s %s(%s)%s%s\n```"; + + String methodKind; + if (methodSymbol.isFunction()) { + methodKind = getResourceString(FUNCTION_KEY); + } else { + methodKind = getResourceString(PROCEDURE_KEY); + } + String methodName = methodSymbol.getName(); + + var parameters = getParametersDescriptionPart(methodSymbol); + var returnedValueType = getReturnedValueTypeDescriptionPart(methodSymbol); + String export = methodSymbol.isExport() ? (" " + getResourceString(EXPORT_KEY)) : ""; + + return String.format( + signatureTemplate, + methodKind, + methodName, + parameters, + export, + returnedValueType + ); + } + + public String getSignature(AnnotationSymbol symbol, MethodSymbol methodSymbol) { + String signatureTemplate = "```bsl\n%s &%s(%s)\n```"; + + String annotationKind = getResourceString(ANNOTATION_KEY); + String annotationName = symbol.getName(); + + var parameters = getParametersDescriptionPart(methodSymbol); + + return String.format( + signatureTemplate, + annotationKind, + annotationName, + parameters + ); + } + + public String getSignature(VariableSymbol symbol) { + String signatureTemplate = "```bsl\n%s %s%s\n```"; + + String varKey = getResourceString(VARIABLE_KEY); + String name = symbol.getName(); + String export = symbol.isExport() ? (" " + getResourceString(EXPORT_KEY)) : ""; + + return String.format( + signatureTemplate, + varKey, + name, + export + ); + } + + private String getParametersDescriptionPart(MethodSymbol methodSymbol) { + var parametersDescription = new StringJoiner(", "); + methodSymbol.getParameters().forEach((ParameterDefinition parameterDefinition) -> { + var parameter = ""; + var parameterName = parameterDefinition.getName(); + + if (parameterDefinition.isByValue()) { + parameter = parameter + getResourceString(VAL_KEY) + " "; + } + parameter += parameterName; + + var parameterTypes = parameterDefinition.getDescription() + .map(ParameterDescription::getTypes) + .map(DescriptionFormatter::getTypes) + .orElse(""); + + if (!parameterTypes.isEmpty()) { + parameter += ": " + parameterTypes; + } + + if (parameterDefinition.isOptional()) { + parameter += " = "; + parameter += parameterDefinition.getDefaultValue().getValue(); + } + + parametersDescription.add(parameter); + }); + + return parametersDescription.toString(); + } + + private static String getReturnedValueTypeDescriptionPart(MethodSymbol methodSymbol) { + String returnedValueType = methodSymbol.getDescription() + .map(MethodDescription::getReturnedValue) + .map(DescriptionFormatter::getTypes) + .orElse(""); + if (!returnedValueType.isEmpty()) { + returnedValueType = ": " + returnedValueType; + } + return returnedValueType; + } + + private static String getTypes(List typeDescriptions) { + return typeDescriptions.stream() + .map(TypeDescription::getName) + .flatMap(parameterType -> Stream.of(parameterType.split(","))) + .map(String::trim) + .collect(Collectors.joining(" | ")); + } + + + public String parameterToString(ParameterDescription parameter, int level) { + var result = new StringJoiner(" \n"); // два пробела + Map typesMap = typesToMap(parameter.getTypes(), level); + var parameterTemplate = " ".repeat(level) + PARAMETER_TEMPLATE; + + if (typesMap.size() == 1) { + result.add(String.format(parameterTemplate, + parameter.getName(), + typesMapToString(typesMap, 0))); + } else { + result.add(String.format(parameterTemplate, parameter.getName(), "")); + result.add(typesMapToString(typesMap, level + 1)); + } + return result.toString(); + } + + public String parameterToString(ParameterDefinition parameterDefinition) { + int level = 0; + var parameterDescription = parameterDefinition.getDescription(); + if (parameterDescription.isPresent()) { + return parameterToString(parameterDescription.get(), level); + } + + return String.format(PARAMETER_TEMPLATE, parameterDefinition.getName(), ""); + } + + private Map typesToMap(List parameterTypes, int level) { + Map types = new HashMap<>(); + + parameterTypes.forEach((TypeDescription type) -> { + var typeDescription = typeToString(type, level); + String typeName; + if (type.isHyperlink()) { + typeName = String.format("[%s](%s)", type.getName(), type.getLink()); + } else { + typeName = String.format("`%s`", type.getName()); + } + + types.merge(typeDescription, typeName, (oldValue, newValue) -> String.format("%s | %s", oldValue, newValue)); + }); + return types; + } + + private String typesMapToString(Map types, int level) { + var result = new StringJoiner(" \n"); // два пробела + var indent = "  ".repeat(level); + types.forEach((String key, String value) -> { + if (key.isBlank()) { + result.add(value); + } else { + result.add(String.format("%s%s %s", indent, value, key)); + } + }); + return result.toString(); + } + + private String typeToString(TypeDescription type, int level) { + var result = new StringJoiner(" \n"); // два пробела + var description = type.getDescription().replace("\n", "
" + "  ".repeat(level + 1)); + + if (!description.isBlank()) { + description = "- " + description; + } + if (!type.getParameters().isEmpty()) { + description += ":"; + } + + result.add(description); + type.getParameters().forEach((ParameterDescription parameter) -> + result.add(parameterToString(parameter, level + 1))); + return result.toString(); + } + + private String getResourceString(String key) { + return resources.getResourceString(getClass(), key); + } + +} diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder.java index de16d79d37f..1161ba95bcd 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder.java @@ -60,7 +60,7 @@ public class MethodSymbolMarkupContentBuilder implements MarkupContentBuilder - result.add(parameterToString(parameterDefinition)) - ); - - var parameters = result.toString(); - - if (!parameters.isBlank()) { - var parametersSection = new StringJoiner("\n"); - String header = "**" + getResourceString(PARAMETERS_KEY) + ":**"; - parametersSection.add(header); - parametersSection.add(""); - parametersSection.add(parameters); - return parametersSection.toString(); - } - - return ""; - } - - private String getReturnedValueSection(MethodSymbol methodSymbol) { - var result = new StringJoiner(" \n"); // два пробела - methodSymbol.getDescription().ifPresent((MethodDescription methodDescription) -> { - Map typesMap = typesToMap(methodDescription.getReturnedValue(), 0); - result.add(typesMapToString(typesMap, 1)); - }); - - var returnedValue = result.toString(); - - if (!returnedValue.isEmpty()) { - returnedValue = "**" + getResourceString(RETURNED_VALUE_KEY) + ":**\n\n" + returnedValue; - } - - return returnedValue; - } - - private String getExamplesSection(MethodSymbol methodSymbol) { - var examples = methodSymbol.getDescription() - .map(MethodDescription::getExamples) - .orElseGet(Collections::emptyList); - return getSectionWithCodeFences(examples, EXAMPLES_KEY); - } - - private String getCallOptionsSection(MethodSymbol methodSymbol) { - var callOptions = methodSymbol.getDescription() - .map(MethodDescription::getCallOptions) - .orElseGet(Collections::emptyList); - return getSectionWithCodeFences(callOptions, CALL_OPTIONS_KEY); - } - - private String getSectionWithCodeFences(List codeBlocks, String resourceKey) { - String codeFences = codeBlocks - .stream() - .map(codeBlock -> "```bsl\n" + codeBlock + "\n```") - .collect(Collectors.joining("\n")); - - if (!codeFences.isEmpty()) { - codeFences = "**" + getResourceString(resourceKey) + ":**\n\n" + codeFences; - } - - return codeFences; - } - - private static String getLocation(MethodSymbol symbol) { - var documentContext = symbol.getOwner(); - var startPosition = symbol.getSelectionRange().getStart(); - String mdoRef = MdoRefBuilder.getMdoRef(documentContext); - - return String.format( - "[%s](%s#%d)", - mdoRef, - documentContext.getUri(), - startPosition.getLine() + 1 - ); - } - - private String getSignature(MethodSymbol methodSymbol) { - String signatureTemplate = "```bsl\n%s %s(%s)%s%s\n```"; - - String methodKind; - if (methodSymbol.isFunction()) { - methodKind = getResourceString(FUNCTION_KEY); - } else { - methodKind = getResourceString(PROCEDURE_KEY); - } - String methodName = methodSymbol.getName(); - - var parametersDescription = new StringJoiner(", "); - methodSymbol.getParameters().forEach((ParameterDefinition parameterDefinition) -> { - var parameter = ""; - var parameterName = parameterDefinition.getName(); - - if (parameterDefinition.isByValue()) { - parameter = parameter + getResourceString(VAL_KEY) + " "; - } - parameter += parameterName; - - var parameterTypes = parameterDefinition.getDescription() - .map(ParameterDescription::getTypes) - .map(MethodSymbolMarkupContentBuilder::getTypes) - .orElse(""); - - if (!parameterTypes.isEmpty()) { - parameter += ": " + parameterTypes; - } - - if (parameterDefinition.isOptional()) { - parameter += " = "; - parameter += parameterDefinition.getDefaultValue().getValue(); - } - - parametersDescription.add(parameter); - }); - var parameters = parametersDescription.toString(); - - String returnedValueType = methodSymbol.getDescription() - .map(MethodDescription::getReturnedValue) - .map(MethodSymbolMarkupContentBuilder::getTypes) - .orElse(""); - if (!returnedValueType.isEmpty()) { - returnedValueType = ": " + returnedValueType; - } - - String export = methodSymbol.isExport() ? (" " + getResourceString(EXPORT_KEY)) : ""; - - return String.format( - signatureTemplate, - methodKind, - methodName, - parameters, - export, - returnedValueType - ); - } - - private static String getTypes(List typeDescriptions) { - return typeDescriptions.stream() - .map(TypeDescription::getName) - .flatMap(parameterType -> Stream.of(parameterType.split(","))) - .map(String::trim) - .collect(Collectors.joining(" | ")); - } - - private String getResourceString(String key) { - return Resources.getResourceString(configuration.getLanguage(), getClass(), key); - } - - public static String parameterToString(ParameterDescription parameter, int level) { - var result = new StringJoiner(" \n"); // два пробела - Map typesMap = typesToMap(parameter.getTypes(), level); - var parameterTemplate = " ".repeat(level) + PARAMETER_TEMPLATE; - - if (typesMap.size() == 1) { - result.add(String.format(parameterTemplate, - parameter.getName(), - typesMapToString(typesMap, 0))); - } else { - result.add(String.format(parameterTemplate, parameter.getName(), "")); - result.add(typesMapToString(typesMap, level + 1)); - } - return result.toString(); - } - - public static String parameterToString(ParameterDefinition parameterDefinition) { - int level = 0; - if (parameterDefinition.getDescription().isPresent()) { - return parameterToString(parameterDefinition.getDescription().get(), level); - } - - return String.format(PARAMETER_TEMPLATE, parameterDefinition.getName(), ""); - } - - private static Map typesToMap(List parameterTypes, int level) { - Map types = new HashMap<>(); - - parameterTypes.forEach((TypeDescription type) -> { - var typeDescription = typeToString(type, level); - String typeName; - if (type.isHyperlink()) { - typeName = String.format("[%s](%s)", type.getName(), type.getLink()); - } else { - typeName = String.format("`%s`", type.getName()); - } - - types.merge(typeDescription, typeName, (oldValue, newValue) -> String.format("%s | %s", oldValue, newValue)); - }); - return types; - } - - private static String typesMapToString(Map types, int level) { - var result = new StringJoiner(" \n"); // два пробела - var indent = "  ".repeat(level); - types.forEach((String key, String value) -> { - if (key.isBlank()) { - result.add(value); - } else { - result.add(String.format("%s%s %s", indent, value, key)); - } - }); - return result.toString(); - } - - private static String typeToString(TypeDescription type, int level) { - var result = new StringJoiner(" \n"); // два пробела - var description = type.getDescription().replace("\n", "
" + "  ".repeat(level + 1)); - - if (!description.isBlank()) { - description = "- " + description; - } - if (!type.getParameters().isEmpty()) { - description += ":"; - } - - result.add(description); - type.getParameters().forEach((ParameterDescription parameter) -> - result.add(parameterToString(parameter, level + 1))); - return result.toString(); - } } diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder.java index 0b442a85e0f..d9fedd1df7d 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder.java @@ -42,6 +42,7 @@ public class VariableSymbolMarkupContentBuilder implements MarkupContentBuilder< private static final String EXPORT_KEY = "export"; private final LanguageServerConfiguration configuration; + private final DescriptionFormatter descriptionFormatter; @Override public MarkupContent getContent(VariableSymbol symbol) { @@ -52,22 +53,22 @@ public MarkupContent getContent(VariableSymbol symbol) { // описание переменной // сигнатура - String signature = getSignature(symbol); - addSectionIfNotEmpty(markupBuilder, signature); + String signature = descriptionFormatter.getSignature(symbol); + descriptionFormatter.addSectionIfNotEmpty(markupBuilder, signature); // местоположение переменной - String location = getLocation(symbol); - addSectionIfNotEmpty(markupBuilder, location); + String location = descriptionFormatter.getLocation(symbol); + descriptionFormatter.addSectionIfNotEmpty(markupBuilder, location); // описание переменной symbol.getDescription() .map(VariableDescription::getPurposeDescription) - .ifPresent(description -> addSectionIfNotEmpty(markupBuilder, description)); + .ifPresent(description -> descriptionFormatter.addSectionIfNotEmpty(markupBuilder, description)); symbol.getDescription() .flatMap(VariableDescription::getTrailingDescription) .map(VariableDescription::getPurposeDescription) - .ifPresent(trailingDescription -> addSectionIfNotEmpty(markupBuilder, trailingDescription)); + .ifPresent(trailingDescription -> descriptionFormatter.addSectionIfNotEmpty(markupBuilder, trailingDescription)); String content = markupBuilder.toString(); @@ -79,48 +80,4 @@ public SymbolKind getSymbolKind() { return SymbolKind.Variable; } - private String getSignature(VariableSymbol symbol) { - String signatureTemplate = "```bsl\n%s %s%s\n```"; - - String varKey = getResourceString(VARIABLE_KEY); - String name = symbol.getName(); - String export = symbol.isExport() ? (" " + getResourceString(EXPORT_KEY)) : ""; - - return String.format( - signatureTemplate, - varKey, - name, - export - ); - } - - private static String getLocation(VariableSymbol symbol) { - var documentContext = symbol.getOwner(); - var startPosition = symbol.getSelectionRange().getStart(); - String mdoRef = MdoRefBuilder.getMdoRef(symbol.getOwner()); - - String parentPostfix = symbol.getRootParent(SymbolKind.Method) - .map(sourceDefinedSymbol -> "." + sourceDefinedSymbol.getName()) - .orElse(""); - mdoRef += parentPostfix; - - return String.format( - "[%s](%s#%d)", - mdoRef, - documentContext.getUri(), - startPosition.getLine() + 1 - ); - } - - private static void addSectionIfNotEmpty(StringJoiner markupBuilder, String newContent) { - if (!newContent.isEmpty()) { - markupBuilder.add(newContent); - markupBuilder.add(""); - markupBuilder.add("---"); - } - } - - private String getResourceString(String key) { - return Resources.getResourceString(configuration.getLanguage(), getClass(), key); - } } diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/inlayhints/SourceDefinedMethodCallInlayHintSupplier.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/inlayhints/SourceDefinedMethodCallInlayHintSupplier.java index d47e73023be..2edb4006c3a 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/inlayhints/SourceDefinedMethodCallInlayHintSupplier.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/inlayhints/SourceDefinedMethodCallInlayHintSupplier.java @@ -25,6 +25,7 @@ import com.github._1c_syntax.bsl.languageserver.context.DocumentContext; import com.github._1c_syntax.bsl.languageserver.context.symbol.MethodSymbol; import com.github._1c_syntax.bsl.languageserver.context.symbol.ParameterDefinition; +import com.github._1c_syntax.bsl.languageserver.hover.DescriptionFormatter; import com.github._1c_syntax.bsl.languageserver.hover.MethodSymbolMarkupContentBuilder; import com.github._1c_syntax.bsl.languageserver.references.ReferenceIndex; import com.github._1c_syntax.bsl.languageserver.references.model.Reference; @@ -63,6 +64,7 @@ public class SourceDefinedMethodCallInlayHintSupplier implements InlayHintSuppli private final ReferenceIndex referenceIndex; private final LanguageServerConfiguration configuration; + private final DescriptionFormatter descriptionFormatter; @Override @@ -81,7 +83,7 @@ public List getInlayHints(DocumentContext documentContext, InlayHintP .filter(Reference::isSourceDefinedSymbolReference) .map(this::toInlayHints) .flatMap(Collection::stream) - .collect(Collectors.toList()); + .toList(); } @@ -129,7 +131,7 @@ private List toInlayHints(Reference reference) { return hints; }) .flatMap(Collection::stream) - .collect(Collectors.toList()); + .toList(); } @@ -164,9 +166,8 @@ private static void setPosition(InlayHint inlayHint, BSLParser.CallParamContext inlayHint.setPosition(position); } - private static void setTooltip(InlayHint inlayHint, ParameterDefinition parameter) { - // todo: refactor - var markdown = MethodSymbolMarkupContentBuilder.parameterToString(parameter); + private void setTooltip(InlayHint inlayHint, ParameterDefinition parameter) { + var markdown = descriptionFormatter.parameterToString(parameter); var tooltip = new MarkupContent(MarkupKind.MARKDOWN, markdown); inlayHint.setTooltip(tooltip); } diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java new file mode 100644 index 00000000000..1a960a9471b --- /dev/null +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java @@ -0,0 +1,79 @@ +/* + * 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.references; + +import com.github._1c_syntax.bsl.languageserver.context.DocumentContext; +import com.github._1c_syntax.bsl.languageserver.context.FileType; +import com.github._1c_syntax.bsl.languageserver.context.ServerContext; +import com.github._1c_syntax.bsl.languageserver.context.symbol.AnnotationSymbol; +import com.github._1c_syntax.bsl.languageserver.references.model.Reference; +import com.github._1c_syntax.bsl.languageserver.utils.Ranges; +import com.github._1c_syntax.bsl.languageserver.utils.Trees; +import com.github._1c_syntax.bsl.parser.BSLParser; +import lombok.RequiredArgsConstructor; +import org.antlr.v4.runtime.tree.TerminalNode; +import org.apache.commons.lang3.tuple.Pair; +import org.eclipse.lsp4j.Location; +import org.eclipse.lsp4j.Position; +import org.springframework.stereotype.Component; + +import java.net.URI; +import java.util.Optional; +import java.util.stream.Collectors; + +@Component +@RequiredArgsConstructor +public class AnnotationReferenceFinder implements ReferenceFinder { + + private final ServerContext serverContext; + + @Override + public Optional findReference(URI uri, Position position) { + DocumentContext document = serverContext.getDocument(uri); + if (document == null || document.getFileType() != FileType.OS) { + return Optional.empty(); + } + + var registeredAnnotations = serverContext.getDocuments().values().stream() + .filter(documentContext -> documentContext.getFileType() == FileType.OS) + .map(DocumentContext::getSymbolTree) + .flatMap(symbolTree -> symbolTree.getMethodSymbol("ПриСозданииОбъекта").stream()) + .filter(methodSymbol -> methodSymbol.getAnnotations().stream().anyMatch(annotation -> annotation.getName().equals("Аннотация"))) + .map(methodSymbol -> Pair.of(methodSymbol, methodSymbol.getAnnotations().stream().filter(annotation -> annotation.getName().equals("Аннотация")).findFirst().get())) + .collect(Collectors.toMap(methodSymbolAnnotationPair -> methodSymbolAnnotationPair.getRight().getParameters().get(0).getValue(), Pair::getLeft)); + + return Trees.findTerminalNodeContainsPosition(document.getAst(), position) + .filter(node -> node.getParent().getRuleContext().getRuleIndex() == BSLParser.RULE_annotationName) + .flatMap((TerminalNode annotationNode) -> { + var annotationName = annotationNode.getText(); + var foundAnnotationDeclaration = registeredAnnotations.get(annotationName); + if (foundAnnotationDeclaration == null) { + return Optional.empty(); + } + return Optional.of(Reference.of( + document.getSymbolTree().getModule(), + AnnotationSymbol.from(annotationName, foundAnnotationDeclaration), + new Location(uri.toString(), Ranges.create(annotationNode.getParent().getParent())) + )); + }); + } +} diff --git a/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder_en.properties b/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter_en.properties similarity index 75% rename from src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder_en.properties rename to src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter_en.properties index dd87e8b007d..02473699a8b 100644 --- a/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder_en.properties +++ b/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter_en.properties @@ -1,3 +1,4 @@ +annotation=Annotation callOptions=Сall options examples=Example export=Export @@ -5,4 +6,5 @@ function=Function parameters=Parameters procedure=Procedure returnedValue=Returns -val=Val \ No newline at end of file +val=Val +var=Var \ No newline at end of file diff --git a/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder_ru.properties b/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter_ru.properties similarity index 77% rename from src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder_ru.properties rename to src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter_ru.properties index da11d16cb00..648351fc42b 100644 --- a/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder_ru.properties +++ b/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter_ru.properties @@ -1,3 +1,4 @@ +annotation=Аннотация callOptions=Варианты вызова examples=Пример export=Экспорт @@ -5,4 +6,5 @@ function=Функция parameters=Параметры procedure=Процедура returnedValue=Возвращаемое значение -val=Знач \ No newline at end of file +val=Знач +var=Перем \ No newline at end of file diff --git a/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder_en.properties b/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder_en.properties deleted file mode 100644 index f352ad82aa2..00000000000 --- a/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder_en.properties +++ /dev/null @@ -1,2 +0,0 @@ -var=Var -export=Export diff --git a/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder_ru.properties b/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder_ru.properties deleted file mode 100644 index 4ef82d6d56b..00000000000 --- a/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder_ru.properties +++ /dev/null @@ -1,2 +0,0 @@ -var=Перем -export=Экспорт diff --git a/src/test/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinderTest.java b/src/test/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinderTest.java new file mode 100644 index 00000000000..e38d1ac9b50 --- /dev/null +++ b/src/test/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinderTest.java @@ -0,0 +1,60 @@ +/* + * 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.references; + +import com.github._1c_syntax.bsl.languageserver.util.TestUtils; +import com.github._1c_syntax.bsl.languageserver.utils.Ranges; +import org.eclipse.lsp4j.Position; +import org.eclipse.lsp4j.SymbolKind; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; + +import static org.assertj.core.api.Assertions.assertThat; + +@SpringBootTest +class AnnotationReferenceFinderTest { + + @Autowired + private AnnotationReferenceFinder referenceFinder; + + @Test + void findReference() { + // given + var documentContext = TestUtils.getDocumentContextFromFile("./src/test/resources/references/AnnotationReferenceFinder.os"); + var module = documentContext.getSymbolTree().getModule(); + var method = documentContext.getSymbolTree().getMethods().get(0); + + // when + var optionalReference = referenceFinder.findReference(documentContext.getUri(), new Position(0, 2)); + + // then + assertThat(optionalReference) + .isPresent() + .hasValueSatisfying(reference -> assertThat(reference.getFrom()).isEqualTo(module)) + .hasValueSatisfying(reference -> assertThat(reference.getSymbol().getName()).isEqualTo("ТестоваяАннотация")) + .hasValueSatisfying(reference -> assertThat(reference.getSymbol().getSymbolKind()).isEqualTo(SymbolKind.TypeParameter)) + .hasValueSatisfying(reference -> assertThat(reference.getSelectionRange()).isEqualTo(Ranges.create(0, 0, 18))) + .hasValueSatisfying(reference -> assertThat(reference.getSourceDefinedSymbol().orElseThrow().getSelectionRange()).isEqualTo(method.getSelectionRange())) + ; + } +} \ No newline at end of file diff --git a/src/test/resources/references/AnnotationReferenceFinder.os b/src/test/resources/references/AnnotationReferenceFinder.os new file mode 100644 index 00000000000..774f950d7c7 --- /dev/null +++ b/src/test/resources/references/AnnotationReferenceFinder.os @@ -0,0 +1,6 @@ +&ТестоваяАннотация +Перем ТестоваяПеременная; + +&Аннотация("ТестоваяАннотация") +Процедура ПриСозданииОбъекта() +КонецПроцедуры \ No newline at end of file From 6e0a2aade59e4a74426157b5a41a837f5cd0d834 Mon Sep 17 00:00:00 2001 From: Nikita Fedkin Date: Mon, 23 Dec 2024 20:06:03 +0100 Subject: [PATCH 02/11] Update src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter_en.properties Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com> --- .../bsl/languageserver/hover/DescriptionFormatter_en.properties | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter_en.properties b/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter_en.properties index 02473699a8b..10ca8a49e46 100644 --- a/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter_en.properties +++ b/src/main/resources/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter_en.properties @@ -1,5 +1,5 @@ annotation=Annotation -callOptions=Сall options +callOptions=Call options examples=Example export=Export function=Function From bd1e564a81846011da1143242ad0f33433647685 Mon Sep 17 00:00:00 2001 From: Nikita Fedkin Date: Fri, 27 Dec 2024 15:32:53 +0100 Subject: [PATCH 03/11] =?UTF-8?q?=D0=A0=D0=B5=D1=84=D0=B0=D0=BA=D1=82?= =?UTF-8?q?=D0=BE=D1=80=D0=B8=D0=BD=D0=B3,=20=D0=BA=D1=8D=D1=88=D0=B8?= =?UTF-8?q?=D1=80=D0=BE=D0=B2=D0=B0=D0=BD=D0=B8=D0=B5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../references/AnnotationReferenceFinder.java | 69 +++++++++++++++---- .../bsl/languageserver/utils/Methods.java | 7 ++ .../AbstractServerContextAwareTest.java | 29 ++++++++ .../diagnostics/AbstractDiagnosticTest.java | 16 +---- .../AnnotationReferenceFinderTest.java | 7 +- 5 files changed, 101 insertions(+), 27 deletions(-) create mode 100644 src/test/java/com/github/_1c_syntax/bsl/languageserver/context/AbstractServerContextAwareTest.java diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java index 1a960a9471b..c3e27810588 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java @@ -24,27 +24,69 @@ import com.github._1c_syntax.bsl.languageserver.context.DocumentContext; import com.github._1c_syntax.bsl.languageserver.context.FileType; import com.github._1c_syntax.bsl.languageserver.context.ServerContext; +import com.github._1c_syntax.bsl.languageserver.context.events.DocumentContextContentChangedEvent; +import com.github._1c_syntax.bsl.languageserver.context.events.ServerContextPopulatedEvent; import com.github._1c_syntax.bsl.languageserver.context.symbol.AnnotationSymbol; +import com.github._1c_syntax.bsl.languageserver.context.symbol.MethodSymbol; +import com.github._1c_syntax.bsl.languageserver.context.symbol.SymbolTree; +import com.github._1c_syntax.bsl.languageserver.context.symbol.annotations.Annotation; import com.github._1c_syntax.bsl.languageserver.references.model.Reference; +import com.github._1c_syntax.bsl.languageserver.utils.Methods; import com.github._1c_syntax.bsl.languageserver.utils.Ranges; import com.github._1c_syntax.bsl.languageserver.utils.Trees; import com.github._1c_syntax.bsl.parser.BSLParser; import lombok.RequiredArgsConstructor; import org.antlr.v4.runtime.tree.TerminalNode; +import org.apache.commons.collections4.map.CaseInsensitiveMap; import org.apache.commons.lang3.tuple.Pair; import org.eclipse.lsp4j.Location; import org.eclipse.lsp4j.Position; +import org.springframework.context.event.EventListener; import org.springframework.stereotype.Component; import java.net.URI; +import java.util.Map; import java.util.Optional; -import java.util.stream.Collectors; @Component @RequiredArgsConstructor public class AnnotationReferenceFinder implements ReferenceFinder { private final ServerContext serverContext; + private final Map registeredAnnotations = new CaseInsensitiveMap<>(); + + @EventListener + public void handleContextRefresh(ServerContextPopulatedEvent event) { + registeredAnnotations.clear(); + serverContext.getDocuments() + .values() + .forEach(this::findAndRegisterAnnotation); + } + + @EventListener + public void handleDocumentContextChange(DocumentContextContentChangedEvent event) { + DocumentContext documentContext = event.getSource(); + var uri = documentContext.getUri(); + + registeredAnnotations.values().stream() + .filter(annotationSymbol -> annotationSymbol.getOwner().getUri().equals(uri)) + .forEach(annotationSymbol -> registeredAnnotations.remove(annotationSymbol.getName())); + + findAndRegisterAnnotation(documentContext); + } + + private void findAndRegisterAnnotation(DocumentContext documentContext) { + if (documentContext.getFileType() != FileType.OS) { + return; + } + + var symbolTree = documentContext.getSymbolTree(); + + Methods.getOscriptClassConstructor(symbolTree) + .flatMap(AnnotationReferenceFinder::findAnnotation) + .map(methodSymbolAnnotationPair -> AnnotationSymbol.from(getAnnotationName(methodSymbolAnnotationPair.getRight()), methodSymbolAnnotationPair.getLeft())) + .ifPresent(annotationSymbol -> registeredAnnotations.put(annotationSymbol.getName(), annotationSymbol)); + } @Override public Optional findReference(URI uri, Position position) { @@ -53,27 +95,30 @@ public Optional findReference(URI uri, Position position) { return Optional.empty(); } - var registeredAnnotations = serverContext.getDocuments().values().stream() - .filter(documentContext -> documentContext.getFileType() == FileType.OS) - .map(DocumentContext::getSymbolTree) - .flatMap(symbolTree -> symbolTree.getMethodSymbol("ПриСозданииОбъекта").stream()) - .filter(methodSymbol -> methodSymbol.getAnnotations().stream().anyMatch(annotation -> annotation.getName().equals("Аннотация"))) - .map(methodSymbol -> Pair.of(methodSymbol, methodSymbol.getAnnotations().stream().filter(annotation -> annotation.getName().equals("Аннотация")).findFirst().get())) - .collect(Collectors.toMap(methodSymbolAnnotationPair -> methodSymbolAnnotationPair.getRight().getParameters().get(0).getValue(), Pair::getLeft)); - return Trees.findTerminalNodeContainsPosition(document.getAst(), position) .filter(node -> node.getParent().getRuleContext().getRuleIndex() == BSLParser.RULE_annotationName) .flatMap((TerminalNode annotationNode) -> { var annotationName = annotationNode.getText(); - var foundAnnotationDeclaration = registeredAnnotations.get(annotationName); - if (foundAnnotationDeclaration == null) { + var annotationSymbol = registeredAnnotations.get(annotationName); + if (annotationSymbol == null) { return Optional.empty(); } return Optional.of(Reference.of( document.getSymbolTree().getModule(), - AnnotationSymbol.from(annotationName, foundAnnotationDeclaration), + annotationSymbol, new Location(uri.toString(), Ranges.create(annotationNode.getParent().getParent())) )); }); } + + private static Optional> findAnnotation(MethodSymbol methodSymbol) { + return methodSymbol.getAnnotations().stream() + .filter(annotation -> annotation.getName().equalsIgnoreCase("Аннотация")) + .findFirst() + .map(annotation -> Pair.of(methodSymbol, annotation)); + } + + private static String getAnnotationName(Annotation annotation) { + return annotation.getParameters().get(0).getValue(); + } } diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/utils/Methods.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/utils/Methods.java index f223a5aa390..3a55bf8e80c 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/utils/Methods.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/utils/Methods.java @@ -21,6 +21,8 @@ */ package com.github._1c_syntax.bsl.languageserver.utils; +import com.github._1c_syntax.bsl.languageserver.context.symbol.MethodSymbol; +import com.github._1c_syntax.bsl.languageserver.context.symbol.SymbolTree; import com.github._1c_syntax.bsl.parser.BSLParser; import lombok.experimental.UtilityClass; import org.antlr.v4.runtime.Token; @@ -82,4 +84,9 @@ public static Optional getMethodName(BSLParser.LValueContext lValueContex .flatMap(Methods::getMethodName); } + public static Optional getOscriptClassConstructor(SymbolTree symbolTree) { + return symbolTree.getMethodSymbol("ПриСозданииОбъекта") + .or(() -> symbolTree.getMethodSymbol("OnObjectCreate")); + } + } diff --git a/src/test/java/com/github/_1c_syntax/bsl/languageserver/context/AbstractServerContextAwareTest.java b/src/test/java/com/github/_1c_syntax/bsl/languageserver/context/AbstractServerContextAwareTest.java new file mode 100644 index 00000000000..b483d6b638a --- /dev/null +++ b/src/test/java/com/github/_1c_syntax/bsl/languageserver/context/AbstractServerContextAwareTest.java @@ -0,0 +1,29 @@ +package com.github._1c_syntax.bsl.languageserver.context; + +import com.github._1c_syntax.utils.Absolute; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; + +import javax.annotation.PostConstruct; +import java.nio.file.Path; + +@SpringBootTest +public abstract class AbstractServerContextAwareTest { + @Autowired + protected ServerContext context; + + @PostConstruct + public void init() { + context.clear(); + } + + protected void initServerContext(String path) { + var configurationRoot = Absolute.path(path); + initServerContext(configurationRoot); + } + + protected void initServerContext(Path configurationRoot) { + context.setConfigurationRoot(configurationRoot); + context.populateContext(); + } +} diff --git a/src/test/java/com/github/_1c_syntax/bsl/languageserver/diagnostics/AbstractDiagnosticTest.java b/src/test/java/com/github/_1c_syntax/bsl/languageserver/diagnostics/AbstractDiagnosticTest.java index 8d845642a63..68e3cea83db 100644 --- a/src/test/java/com/github/_1c_syntax/bsl/languageserver/diagnostics/AbstractDiagnosticTest.java +++ b/src/test/java/com/github/_1c_syntax/bsl/languageserver/diagnostics/AbstractDiagnosticTest.java @@ -22,6 +22,7 @@ package com.github._1c_syntax.bsl.languageserver.diagnostics; import com.github._1c_syntax.bsl.languageserver.configuration.LanguageServerConfiguration; +import com.github._1c_syntax.bsl.languageserver.context.AbstractServerContextAwareTest; import com.github._1c_syntax.bsl.languageserver.context.DocumentContext; import com.github._1c_syntax.bsl.languageserver.context.ServerContext; import com.github._1c_syntax.bsl.languageserver.diagnostics.infrastructure.DiagnosticObjectProvider; @@ -46,13 +47,11 @@ import java.util.List; @SpringBootTest -abstract class AbstractDiagnosticTest { +abstract class AbstractDiagnosticTest extends AbstractServerContextAwareTest { @Autowired private DiagnosticObjectProvider diagnosticObjectProvider; @Autowired - protected ServerContext context; - @Autowired protected LanguageServerConfiguration configuration; private final Class diagnosticClass; @@ -65,20 +64,9 @@ abstract class AbstractDiagnosticTest { @PostConstruct public void init() { diagnosticInstance = diagnosticObjectProvider.get(diagnosticClass); - context.clear(); configuration.reset(); } - protected void initServerContext(String path) { - var configurationRoot = Absolute.path(path); - initServerContext(configurationRoot); - } - - protected void initServerContext(Path configurationRoot) { - context.setConfigurationRoot(configurationRoot); - context.populateContext(); - } - protected List getDiagnostics(DocumentContext documentContext) { return diagnosticInstance.getDiagnostics(documentContext); } diff --git a/src/test/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinderTest.java b/src/test/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinderTest.java index e38d1ac9b50..82b9a0af985 100644 --- a/src/test/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinderTest.java +++ b/src/test/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinderTest.java @@ -21,6 +21,8 @@ */ package com.github._1c_syntax.bsl.languageserver.references; +import com.github._1c_syntax.bsl.languageserver.context.AbstractServerContextAwareTest; +import com.github._1c_syntax.bsl.languageserver.util.CleanupContextBeforeClassAndAfterClass; import com.github._1c_syntax.bsl.languageserver.util.TestUtils; import com.github._1c_syntax.bsl.languageserver.utils.Ranges; import org.eclipse.lsp4j.Position; @@ -32,7 +34,8 @@ import static org.assertj.core.api.Assertions.assertThat; @SpringBootTest -class AnnotationReferenceFinderTest { +@CleanupContextBeforeClassAndAfterClass +class AnnotationReferenceFinderTest extends AbstractServerContextAwareTest { @Autowired private AnnotationReferenceFinder referenceFinder; @@ -40,7 +43,9 @@ class AnnotationReferenceFinderTest { @Test void findReference() { // given + initServerContext(TestUtils.PATH_TO_METADATA); var documentContext = TestUtils.getDocumentContextFromFile("./src/test/resources/references/AnnotationReferenceFinder.os"); + var module = documentContext.getSymbolTree().getModule(); var method = documentContext.getSymbolTree().getMethods().get(0); From a155eb2451caae5fd10a14a9fe54ae8be9f436c7 Mon Sep 17 00:00:00 2001 From: Nikita Fedkin Date: Fri, 27 Dec 2024 15:56:46 +0100 Subject: [PATCH 04/11] =?UTF-8?q?=D0=98=D1=81=D0=BF=D1=80=D0=B0=D0=B2?= =?UTF-8?q?=D0=BB=D0=B5=D0=BD=D0=B8=D1=8F?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../context/symbol/AnnotationSymbol.java | 9 ++++++--- .../references/AnnotationReferenceFinder.java | 10 ++++------ .../references/AnnotationReferenceFinderTest.java | 5 ++--- .../resources/references/AnnotationReferenceFinder.os | 4 +++- ...320\276\321\202\320\260\321\206\320\270\321\217.os" | 3 +++ ...20\276\321\202\320\260\321\206\320\270\321\2172.os" | 3 +++ 6 files changed, 21 insertions(+), 13 deletions(-) create mode 100644 "src/test/resources/references/annotations/\320\242\320\265\321\201\321\202\320\276\320\262\320\260\321\217\320\220\320\275\320\275\320\276\321\202\320\260\321\206\320\270\321\217.os" create mode 100644 "src/test/resources/references/annotations/\320\242\320\265\321\201\321\202\320\276\320\262\320\260\321\217\320\220\320\275\320\275\320\276\321\202\320\260\321\206\320\270\321\2172.os" diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/context/symbol/AnnotationSymbol.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/context/symbol/AnnotationSymbol.java index 5a7cd0851c3..0ae0fda074f 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/context/symbol/AnnotationSymbol.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/context/symbol/AnnotationSymbol.java @@ -36,6 +36,7 @@ import org.eclipse.lsp4j.SymbolKind; import java.util.ArrayList; +import java.util.Collections; import java.util.List; import java.util.Optional; @@ -62,11 +63,13 @@ public class AnnotationSymbol implements SourceDefinedSymbol, Describable { @Builder.Default Optional parent = Optional.empty(); - @Builder.Default - List children = new ArrayList<>(); - Optional description; + @Override + public List getChildren() { + return Collections.emptyList(); + } + @Override public void accept(SymbolTreeVisitor visitor) { // no-op diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java index c3e27810588..28649de507f 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java @@ -28,7 +28,6 @@ import com.github._1c_syntax.bsl.languageserver.context.events.ServerContextPopulatedEvent; import com.github._1c_syntax.bsl.languageserver.context.symbol.AnnotationSymbol; import com.github._1c_syntax.bsl.languageserver.context.symbol.MethodSymbol; -import com.github._1c_syntax.bsl.languageserver.context.symbol.SymbolTree; import com.github._1c_syntax.bsl.languageserver.context.symbol.annotations.Annotation; import com.github._1c_syntax.bsl.languageserver.references.model.Reference; import com.github._1c_syntax.bsl.languageserver.utils.Methods; @@ -37,7 +36,6 @@ import com.github._1c_syntax.bsl.parser.BSLParser; import lombok.RequiredArgsConstructor; import org.antlr.v4.runtime.tree.TerminalNode; -import org.apache.commons.collections4.map.CaseInsensitiveMap; import org.apache.commons.lang3.tuple.Pair; import org.eclipse.lsp4j.Location; import org.eclipse.lsp4j.Position; @@ -47,13 +45,14 @@ import java.net.URI; import java.util.Map; import java.util.Optional; +import java.util.concurrent.ConcurrentHashMap; @Component @RequiredArgsConstructor public class AnnotationReferenceFinder implements ReferenceFinder { private final ServerContext serverContext; - private final Map registeredAnnotations = new CaseInsensitiveMap<>(); + private final Map registeredAnnotations = new ConcurrentHashMap<>(); @EventListener public void handleContextRefresh(ServerContextPopulatedEvent event) { @@ -68,9 +67,8 @@ public void handleDocumentContextChange(DocumentContextContentChangedEvent event DocumentContext documentContext = event.getSource(); var uri = documentContext.getUri(); - registeredAnnotations.values().stream() - .filter(annotationSymbol -> annotationSymbol.getOwner().getUri().equals(uri)) - .forEach(annotationSymbol -> registeredAnnotations.remove(annotationSymbol.getName())); + registeredAnnotations.values() + .removeIf(annotationSymbol -> annotationSymbol.getOwner().getUri().equals(uri)); findAndRegisterAnnotation(documentContext); } diff --git a/src/test/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinderTest.java b/src/test/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinderTest.java index 82b9a0af985..0f639341f1e 100644 --- a/src/test/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinderTest.java +++ b/src/test/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinderTest.java @@ -43,11 +43,10 @@ class AnnotationReferenceFinderTest extends AbstractServerContextAwareTest { @Test void findReference() { // given - initServerContext(TestUtils.PATH_TO_METADATA); + initServerContext("./src/test/resources/references/annotations"); var documentContext = TestUtils.getDocumentContextFromFile("./src/test/resources/references/AnnotationReferenceFinder.os"); var module = documentContext.getSymbolTree().getModule(); - var method = documentContext.getSymbolTree().getMethods().get(0); // when var optionalReference = referenceFinder.findReference(documentContext.getUri(), new Position(0, 2)); @@ -59,7 +58,7 @@ void findReference() { .hasValueSatisfying(reference -> assertThat(reference.getSymbol().getName()).isEqualTo("ТестоваяАннотация")) .hasValueSatisfying(reference -> assertThat(reference.getSymbol().getSymbolKind()).isEqualTo(SymbolKind.TypeParameter)) .hasValueSatisfying(reference -> assertThat(reference.getSelectionRange()).isEqualTo(Ranges.create(0, 0, 18))) - .hasValueSatisfying(reference -> assertThat(reference.getSourceDefinedSymbol().orElseThrow().getSelectionRange()).isEqualTo(method.getSelectionRange())) + .hasValueSatisfying(reference -> assertThat(reference.getSourceDefinedSymbol().orElseThrow().getSelectionRange()).isEqualTo(Ranges.create(1, 10, 28))) ; } } \ No newline at end of file diff --git a/src/test/resources/references/AnnotationReferenceFinder.os b/src/test/resources/references/AnnotationReferenceFinder.os index 774f950d7c7..a6bb131844e 100644 --- a/src/test/resources/references/AnnotationReferenceFinder.os +++ b/src/test/resources/references/AnnotationReferenceFinder.os @@ -1,6 +1,8 @@ &ТестоваяАннотация Перем ТестоваяПеременная; -&Аннотация("ТестоваяАннотация") +&ТестоваяАннотация2 +Перем ТестоваяПеременная2; + Процедура ПриСозданииОбъекта() КонецПроцедуры \ No newline at end of file diff --git "a/src/test/resources/references/annotations/\320\242\320\265\321\201\321\202\320\276\320\262\320\260\321\217\320\220\320\275\320\275\320\276\321\202\320\260\321\206\320\270\321\217.os" "b/src/test/resources/references/annotations/\320\242\320\265\321\201\321\202\320\276\320\262\320\260\321\217\320\220\320\275\320\275\320\276\321\202\320\260\321\206\320\270\321\217.os" new file mode 100644 index 00000000000..e3b4f507ffa --- /dev/null +++ "b/src/test/resources/references/annotations/\320\242\320\265\321\201\321\202\320\276\320\262\320\260\321\217\320\220\320\275\320\275\320\276\321\202\320\260\321\206\320\270\321\217.os" @@ -0,0 +1,3 @@ +&Аннотация("ТестоваяАннотация") +Процедура ПриСозданииОбъекта() +КонецПроцедуры \ No newline at end of file diff --git "a/src/test/resources/references/annotations/\320\242\320\265\321\201\321\202\320\276\320\262\320\260\321\217\320\220\320\275\320\275\320\276\321\202\320\260\321\206\320\270\321\2172.os" "b/src/test/resources/references/annotations/\320\242\320\265\321\201\321\202\320\276\320\262\320\260\321\217\320\220\320\275\320\275\320\276\321\202\320\260\321\206\320\270\321\2172.os" new file mode 100644 index 00000000000..1bbb37f66e2 --- /dev/null +++ "b/src/test/resources/references/annotations/\320\242\320\265\321\201\321\202\320\276\320\262\320\260\321\217\320\220\320\275\320\275\320\276\321\202\320\260\321\206\320\270\321\2172.os" @@ -0,0 +1,3 @@ +&Аннотация("ТестоваяАннотация2") +Процедура ПриСозданииОбъекта() +КонецПроцедуры \ No newline at end of file From e87c1329b2064990f2071114e7a0cd7c3486aea2 Mon Sep 17 00:00:00 2001 From: Nikita Fedkin Date: Fri, 27 Dec 2024 16:01:34 +0100 Subject: [PATCH 05/11] =?UTF-8?q?=D0=9B=D0=B8=D1=86=D0=B5=D0=BD=D0=B7?= =?UTF-8?q?=D0=B8=D1=8F?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../AbstractServerContextAwareTest.java | 21 +++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/src/test/java/com/github/_1c_syntax/bsl/languageserver/context/AbstractServerContextAwareTest.java b/src/test/java/com/github/_1c_syntax/bsl/languageserver/context/AbstractServerContextAwareTest.java index b483d6b638a..07dfbad07fd 100644 --- a/src/test/java/com/github/_1c_syntax/bsl/languageserver/context/AbstractServerContextAwareTest.java +++ b/src/test/java/com/github/_1c_syntax/bsl/languageserver/context/AbstractServerContextAwareTest.java @@ -1,3 +1,24 @@ +/* + * 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.context; import com.github._1c_syntax.utils.Absolute; From 2393a2ba0956d45d77a7025fcd493ec048925dc5 Mon Sep 17 00:00:00 2001 From: Nikita Fedkin Date: Fri, 27 Dec 2024 16:05:06 +0100 Subject: [PATCH 06/11] =?UTF-8?q?=D0=97=D0=B0=D1=89=D0=B8=D1=82=D0=B0=20?= =?UTF-8?q?=D0=BE=D1=82=20=D0=BD=D0=B5=D0=B2=D0=B5=D1=80=D0=BD=D0=BE=D0=B3?= =?UTF-8?q?=D0=BE=20=D0=BA=D0=BE=D0=BB=D0=B8=D1=87=D0=B5=D1=81=D1=82=D0=B2?= =?UTF-8?q?=D0=B0=20=D0=BF=D0=B0=D1=80=D0=B0=D0=BC=D0=B5=D1=82=D1=80=D0=BE?= =?UTF-8?q?=D0=B2=20=D0=B0=D0=BD=D0=BD=D0=BE=D1=82=D0=B0=D1=86=D0=B8=D0=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../bsl/languageserver/references/AnnotationReferenceFinder.java | 1 + 1 file changed, 1 insertion(+) diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java index 28649de507f..7916aa8cd24 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java @@ -113,6 +113,7 @@ private static Optional> findAnnotation(MethodSym return methodSymbol.getAnnotations().stream() .filter(annotation -> annotation.getName().equalsIgnoreCase("Аннотация")) .findFirst() + .filter(annotation -> annotation.getParameters().size() == 1) .map(annotation -> Pair.of(methodSymbol, annotation)); } From 3ae01fd63bb146bc69019970e33139048cfa776c Mon Sep 17 00:00:00 2001 From: Nikita Fedkin Date: Fri, 27 Dec 2024 16:12:32 +0100 Subject: [PATCH 07/11] =?UTF-8?q?=D0=9D=D0=B5=D0=B1=D0=BE=D0=BB=D1=8C?= =?UTF-8?q?=D1=88=D0=BE=D0=B9=20=D0=BA=D0=BE=D0=BC=D0=BC=D0=B5=D0=BD=D1=82?= =?UTF-8?q?=D0=B0=D1=80=D0=B8=D0=B9=20=D0=BD=D0=B0=20=D0=B1=D1=83=D0=B4?= =?UTF-8?q?=D1=83=D1=89=D0=B5=D0=B5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../languageserver/references/AnnotationReferenceFinder.java | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java index 7916aa8cd24..96988a0e413 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/references/AnnotationReferenceFinder.java @@ -74,6 +74,11 @@ public void handleDocumentContextChange(DocumentContextContentChangedEvent event } private void findAndRegisterAnnotation(DocumentContext documentContext) { + // In normal case this method may be called twice per each document context: + // 1. When the document context is created during the server context population or document opening + // 2. When server context is fully populated. + // This can lead to the situation when annotations registered from opened documents are cleared after populateContext step. + // Due to limitation of mechanism to only OS files, we can leave it as is for now, but it should be refactored in the future. if (documentContext.getFileType() != FileType.OS) { return; } From 97831e3f2d1012cf75ae97f0eb7eaccc088d6126 Mon Sep 17 00:00:00 2001 From: Nikita Fedkin Date: Fri, 27 Dec 2024 16:17:49 +0100 Subject: [PATCH 08/11] Fix tests --- .../languageserver/context/AbstractServerContextAwareTest.java | 2 +- .../bsl/languageserver/diagnostics/AbstractDiagnosticTest.java | 3 --- 2 files changed, 1 insertion(+), 4 deletions(-) diff --git a/src/test/java/com/github/_1c_syntax/bsl/languageserver/context/AbstractServerContextAwareTest.java b/src/test/java/com/github/_1c_syntax/bsl/languageserver/context/AbstractServerContextAwareTest.java index 07dfbad07fd..0923a7e3dd7 100644 --- a/src/test/java/com/github/_1c_syntax/bsl/languageserver/context/AbstractServerContextAwareTest.java +++ b/src/test/java/com/github/_1c_syntax/bsl/languageserver/context/AbstractServerContextAwareTest.java @@ -34,7 +34,7 @@ public abstract class AbstractServerContextAwareTest { protected ServerContext context; @PostConstruct - public void init() { + public void abstractServerContextAwareTestInit() { context.clear(); } diff --git a/src/test/java/com/github/_1c_syntax/bsl/languageserver/diagnostics/AbstractDiagnosticTest.java b/src/test/java/com/github/_1c_syntax/bsl/languageserver/diagnostics/AbstractDiagnosticTest.java index 68e3cea83db..a760546ff57 100644 --- a/src/test/java/com/github/_1c_syntax/bsl/languageserver/diagnostics/AbstractDiagnosticTest.java +++ b/src/test/java/com/github/_1c_syntax/bsl/languageserver/diagnostics/AbstractDiagnosticTest.java @@ -24,10 +24,8 @@ import com.github._1c_syntax.bsl.languageserver.configuration.LanguageServerConfiguration; import com.github._1c_syntax.bsl.languageserver.context.AbstractServerContextAwareTest; import com.github._1c_syntax.bsl.languageserver.context.DocumentContext; -import com.github._1c_syntax.bsl.languageserver.context.ServerContext; import com.github._1c_syntax.bsl.languageserver.diagnostics.infrastructure.DiagnosticObjectProvider; import com.github._1c_syntax.bsl.languageserver.util.TestUtils; -import com.github._1c_syntax.utils.Absolute; import jakarta.annotation.PostConstruct; import lombok.SneakyThrows; import org.apache.commons.io.IOUtils; @@ -42,7 +40,6 @@ import org.springframework.boot.test.context.SpringBootTest; import java.nio.charset.StandardCharsets; -import java.nio.file.Path; import java.util.Collections; import java.util.List; From 5d351e909afa8a62d72165edaffad03e270a1170 Mon Sep 17 00:00:00 2001 From: Nikita Fedkin Date: Fri, 27 Dec 2024 16:19:06 +0100 Subject: [PATCH 09/11] Fix QF --- .../bsl/languageserver/context/symbol/AnnotationSymbol.java | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/context/symbol/AnnotationSymbol.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/context/symbol/AnnotationSymbol.java index 0ae0fda074f..06319c9a443 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/context/symbol/AnnotationSymbol.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/context/symbol/AnnotationSymbol.java @@ -22,12 +22,9 @@ package com.github._1c_syntax.bsl.languageserver.context.symbol; import com.github._1c_syntax.bsl.languageserver.context.DocumentContext; -import com.github._1c_syntax.bsl.languageserver.context.symbol.annotations.Annotation; import com.github._1c_syntax.bsl.languageserver.context.symbol.description.MethodDescription; -import com.github._1c_syntax.bsl.languageserver.context.symbol.variable.VariableDescription; import lombok.Builder; import lombok.EqualsAndHashCode; -import lombok.Getter; import lombok.Setter; import lombok.ToString; import lombok.Value; @@ -35,7 +32,6 @@ import org.eclipse.lsp4j.Range; import org.eclipse.lsp4j.SymbolKind; -import java.util.ArrayList; import java.util.Collections; import java.util.List; import java.util.Optional; @@ -43,7 +39,7 @@ @Value @Builder @EqualsAndHashCode(onlyExplicitlyIncluded = true) -@ToString(exclude = {"children", "parent"}) +@ToString(exclude = {"parent"}) public class AnnotationSymbol implements SourceDefinedSymbol, Describable { String name; From 166ae66833892d8797a1a4c1712735db2509a6e6 Mon Sep 17 00:00:00 2001 From: Nikita Fedkin Date: Fri, 27 Dec 2024 16:38:09 +0100 Subject: [PATCH 10/11] Fix QF --- .../hover/AnnotationSymbolMarkupContentBuilder.java | 12 ------------ .../languageserver/hover/DescriptionFormatter.java | 1 - .../hover/MethodSymbolMarkupContentBuilder.java | 13 ------------- .../hover/VariableSymbolMarkupContentBuilder.java | 2 -- .../SourceDefinedMethodCallInlayHintSupplier.java | 2 -- 5 files changed, 30 deletions(-) diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/AnnotationSymbolMarkupContentBuilder.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/AnnotationSymbolMarkupContentBuilder.java index 39abf339813..c9724ed4099 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/AnnotationSymbolMarkupContentBuilder.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/AnnotationSymbolMarkupContentBuilder.java @@ -23,25 +23,13 @@ import com.github._1c_syntax.bsl.languageserver.context.symbol.AnnotationSymbol; import com.github._1c_syntax.bsl.languageserver.context.symbol.MethodSymbol; -import com.github._1c_syntax.bsl.languageserver.context.symbol.ParameterDefinition; -import com.github._1c_syntax.bsl.languageserver.context.symbol.description.MethodDescription; -import com.github._1c_syntax.bsl.languageserver.context.symbol.description.ParameterDescription; -import com.github._1c_syntax.bsl.languageserver.context.symbol.description.TypeDescription; -import com.github._1c_syntax.bsl.languageserver.utils.MdoRefBuilder; -import com.github._1c_syntax.bsl.languageserver.utils.Resources; import lombok.RequiredArgsConstructor; import org.eclipse.lsp4j.MarkupContent; import org.eclipse.lsp4j.MarkupKind; import org.eclipse.lsp4j.SymbolKind; import org.springframework.stereotype.Component; -import java.util.Collections; -import java.util.HashMap; -import java.util.List; -import java.util.Map; import java.util.StringJoiner; -import java.util.stream.Collectors; -import java.util.stream.Stream; /** * Построитель контента для всплывающего окна для {@link AnnotationSymbol}. diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter.java index d483c6f7a04..1eccad44d88 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter.java @@ -38,7 +38,6 @@ import java.util.HashMap; import java.util.List; import java.util.Map; -import java.util.Optional; import java.util.StringJoiner; import java.util.stream.Collectors; import java.util.stream.Stream; diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder.java index 1161ba95bcd..ed7360d9342 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/MethodSymbolMarkupContentBuilder.java @@ -21,27 +21,14 @@ */ package com.github._1c_syntax.bsl.languageserver.hover; -import com.github._1c_syntax.bsl.languageserver.configuration.LanguageServerConfiguration; import com.github._1c_syntax.bsl.languageserver.context.symbol.MethodSymbol; -import com.github._1c_syntax.bsl.languageserver.context.symbol.ParameterDefinition; -import com.github._1c_syntax.bsl.languageserver.context.symbol.description.MethodDescription; -import com.github._1c_syntax.bsl.languageserver.context.symbol.description.ParameterDescription; -import com.github._1c_syntax.bsl.languageserver.context.symbol.description.TypeDescription; -import com.github._1c_syntax.bsl.languageserver.utils.MdoRefBuilder; -import com.github._1c_syntax.bsl.languageserver.utils.Resources; import lombok.RequiredArgsConstructor; import org.eclipse.lsp4j.MarkupContent; import org.eclipse.lsp4j.MarkupKind; import org.eclipse.lsp4j.SymbolKind; import org.springframework.stereotype.Component; -import java.util.Collections; -import java.util.HashMap; -import java.util.List; -import java.util.Map; import java.util.StringJoiner; -import java.util.stream.Collectors; -import java.util.stream.Stream; /** * Построитель контента для всплывающего окна для {@link MethodSymbol}. diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder.java index d9fedd1df7d..7ca24d7b48e 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/VariableSymbolMarkupContentBuilder.java @@ -24,8 +24,6 @@ import com.github._1c_syntax.bsl.languageserver.configuration.LanguageServerConfiguration; import com.github._1c_syntax.bsl.languageserver.context.symbol.VariableSymbol; import com.github._1c_syntax.bsl.languageserver.context.symbol.variable.VariableDescription; -import com.github._1c_syntax.bsl.languageserver.utils.MdoRefBuilder; -import com.github._1c_syntax.bsl.languageserver.utils.Resources; import lombok.RequiredArgsConstructor; import org.eclipse.lsp4j.MarkupContent; import org.eclipse.lsp4j.MarkupKind; diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/inlayhints/SourceDefinedMethodCallInlayHintSupplier.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/inlayhints/SourceDefinedMethodCallInlayHintSupplier.java index 2edb4006c3a..4ee1a9c05f2 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/inlayhints/SourceDefinedMethodCallInlayHintSupplier.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/inlayhints/SourceDefinedMethodCallInlayHintSupplier.java @@ -26,7 +26,6 @@ import com.github._1c_syntax.bsl.languageserver.context.symbol.MethodSymbol; import com.github._1c_syntax.bsl.languageserver.context.symbol.ParameterDefinition; import com.github._1c_syntax.bsl.languageserver.hover.DescriptionFormatter; -import com.github._1c_syntax.bsl.languageserver.hover.MethodSymbolMarkupContentBuilder; import com.github._1c_syntax.bsl.languageserver.references.ReferenceIndex; import com.github._1c_syntax.bsl.languageserver.references.model.Reference; import com.github._1c_syntax.bsl.languageserver.utils.Ranges; @@ -48,7 +47,6 @@ import java.util.ArrayList; import java.util.Collection; import java.util.List; -import java.util.stream.Collectors; /** * Поставщик подсказок о параметрах вызываемого метода. From fff4668b1017d556a639bcd7ed8c54714a70dc68 Mon Sep 17 00:00:00 2001 From: Nikita Fedkin Date: Fri, 27 Dec 2024 16:40:16 +0100 Subject: [PATCH 11/11] Fix QF --- .../hover/DescriptionFormatter.java | 23 ++++++++++--------- 1 file changed, 12 insertions(+), 11 deletions(-) diff --git a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter.java b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter.java index 1eccad44d88..277aac1ef03 100644 --- a/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter.java +++ b/src/main/java/com/github/_1c_syntax/bsl/languageserver/hover/DescriptionFormatter.java @@ -34,6 +34,7 @@ import org.eclipse.lsp4j.SymbolKind; import org.springframework.stereotype.Component; +import java.util.Collection; import java.util.Collections; import java.util.HashMap; import java.util.List; @@ -124,7 +125,7 @@ public String getCallOptionsSection(MethodSymbol methodSymbol) { return getSectionWithCodeFences(callOptions, CALL_OPTIONS_KEY); } - public String getSectionWithCodeFences(List codeBlocks, String resourceKey) { + public String getSectionWithCodeFences(Collection codeBlocks, String resourceKey) { String codeFences = codeBlocks .stream() .map(codeBlock -> "```bsl\n" + codeBlock + "\n```") @@ -169,7 +170,7 @@ public String getLocation(VariableSymbol symbol) { } public String getSignature(MethodSymbol methodSymbol) { - String signatureTemplate = "```bsl\n%s %s(%s)%s%s\n```"; + var signatureTemplate = "```bsl\n%s %s(%s)%s%s\n```"; String methodKind; if (methodSymbol.isFunction()) { @@ -194,10 +195,10 @@ public String getSignature(MethodSymbol methodSymbol) { } public String getSignature(AnnotationSymbol symbol, MethodSymbol methodSymbol) { - String signatureTemplate = "```bsl\n%s &%s(%s)\n```"; + var signatureTemplate = "```bsl\n%s &%s(%s)\n```"; - String annotationKind = getResourceString(ANNOTATION_KEY); - String annotationName = symbol.getName(); + var annotationKind = getResourceString(ANNOTATION_KEY); + var annotationName = symbol.getName(); var parameters = getParametersDescriptionPart(methodSymbol); @@ -210,11 +211,11 @@ public String getSignature(AnnotationSymbol symbol, MethodSymbol methodSymbol) { } public String getSignature(VariableSymbol symbol) { - String signatureTemplate = "```bsl\n%s %s%s\n```"; + var signatureTemplate = "```bsl\n%s %s%s\n```"; - String varKey = getResourceString(VARIABLE_KEY); - String name = symbol.getName(); - String export = symbol.isExport() ? (" " + getResourceString(EXPORT_KEY)) : ""; + var varKey = getResourceString(VARIABLE_KEY); + var name = symbol.getName(); + var export = symbol.isExport() ? (" " + getResourceString(EXPORT_KEY)) : ""; return String.format( signatureTemplate, @@ -292,7 +293,7 @@ public String parameterToString(ParameterDescription parameter, int level) { } public String parameterToString(ParameterDefinition parameterDefinition) { - int level = 0; + var level = 0; var parameterDescription = parameterDefinition.getDescription(); if (parameterDescription.isPresent()) { return parameterToString(parameterDescription.get(), level); @@ -318,7 +319,7 @@ private Map typesToMap(List parameterTypes, int return types; } - private String typesMapToString(Map types, int level) { + private static String typesMapToString(Map types, int level) { var result = new StringJoiner(" \n"); // два пробела var indent = "  ".repeat(level); types.forEach((String key, String value) -> {