-
Notifications
You must be signed in to change notification settings - Fork 109
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Границы обращения к метаданным #1837
base: develop
Are you sure you want to change the base?
Changes from 16 commits
cd9ef8b
768ab7d
ea1e7dd
b17d97f
2bc45e1
f06515a
65557a8
fdad879
8912ca2
36f4e78
f05deaa
1248793
1379891
84e4d9d
7171d3f
239f1c0
5ce7014
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,67 @@ | ||
# Границы метаданных (MetadataBorders) | ||
|
||
| Тип | Поддерживаются<br>языки | Важность | Включена<br>по умолчанию | Время на<br>исправление (мин) | Теги | | ||
|:-------------:|:-----------------------------:|:----------------:|:------------------------------:|:-----------------------------------:|:--------:| | ||
| `Дефект кода` | `BSL`<br>`OS` | `Информационный` | `Нет` | `1` | `design` | | ||
|
||
## Параметры | ||
|
||
|
||
| Имя | Тип | Описание | Значение<br>по умолчанию | | ||
|:---------------------------:|:--------:|:------------------------------------------------------------------------------------------------------------:|:------------------------------:| | ||
| `metadataBordersParameters` | `Строка` | `JSON-структура для пар "регулярное выражение для операторов":"регулярное выражение для имени файла модуля"` | `` | | ||
<!-- Блоки выше заполняются автоматически, не трогать --> | ||
## Описание диагностики | ||
Диагностика обращения к метаданным за пределами разрешенных границ. Нужна, если, например, в компании установлены правила обращения к каким-то метаданным только из определенного общего модуля. | ||
|
||
## Примеры | ||
Например, получение контактной информации по внутренним правилам должно происходить только через функции общего модуля РаботаСКонтактами. Тогда заданные настройки вида | ||
|
||
```{"Регистры?Сведений.КонтактнаяИнформация":"CommonModules/РаботаСКонтактами"}``` | ||
Golovanoff marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
вызовут срабатывание диагностики на запросе с текстом | ||
|
||
``` | ||
"ВЫБРАТЬ | ||
... | ||
ИЗ | ||
РегистрСведений.КонтактнаяИнформация КАК КИ" | ||
``` | ||
|
||
если этот запрос будет не в общем модуле РаботаСКонтактами. Так же эта диагностика с такой настройкой сработает и на оператор | ||
|
||
```НаборЗаписей = РегистрыСведений.ИспользованиеКонтактов.СоздатьНаборЗаписей();``` | ||
Golovanoff marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
В настройках обе части - это регулярные выражения java. Вторая часть настройки сравнивается с путем к файлу выгрузки конфигурации. | ||
|
||
Можно запретить использование нескольких метаданных: | ||
|
||
```{"Регистры?Сведений.КонтактнаяИнформация|Справочники?.Контакты":"CommonModules/РаботаСКонтактами"}``` | ||
|
||
Настройки задаются в виде json-строки: | ||
``` | ||
Golovanoff marked this conversation as resolved.
Show resolved
Hide resolved
|
||
{ | ||
"Регистры?Сведений.КонтактнаяИнформация":"CommonModules/РаботаСКонтактами", | ||
"Справочники?.СтрокиСоединения":"DataProcessors/ВнешниеСоединения|CommonModules/ПодключениеК.*" | ||
} | ||
``` | ||
|
||
## Источники | ||
|
||
## Сниппеты | ||
|
||
<!-- Блоки ниже заполняются автоматически, не трогать --> | ||
### Экранирование кода | ||
|
||
```bsl | ||
// BSLLS:MetadataBorders-off | ||
// BSLLS:MetadataBorders-on | ||
``` | ||
|
||
### Параметр конфигурационного файла | ||
|
||
```json | ||
"MetadataBorders": { | ||
"metadataBordersParameters": "" | ||
} | ||
``` |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,44 @@ | ||
# Metadata borders (MetadataBorders) | ||
|
||
| Type | Scope | Severity | Activated<br>by default | Minutes<br>to fix | Tags | | ||
|:------------:|:-------------------:|:--------:|:-----------------------------:|:-----------------------:|:--------:| | ||
| `Code smell` | `BSL`<br>`OS` | `Info` | `No` | `1` | `design` | | ||
|
||
## Parameters | ||
|
||
|
||
| Name | Type | Description | Default value | | ||
|:---------------------------:|:--------:|:---------------------------------------------------------------------------:|:-------------:| | ||
| `metadataBordersParameters` | `String` | `JSON-structure for pairs "regex for statements":"regex for module names".` | `` | | ||
<!-- Блоки выше заполняются автоматически, не трогать --> | ||
## Description | ||
<!-- Описание диагностики заполняется вручную. Необходимо понятным языком описать смысл и схему работу --> | ||
|
||
## Examples | ||
<!-- В данном разделе приводятся примеры, на которые диагностика срабатывает, а также можно привести пример, как можно исправить ситуацию --> | ||
|
||
## Sources | ||
<!-- Необходимо указывать ссылки на все источники, из которых почерпнута информация для создания диагностики --> | ||
<!-- Примеры источников | ||
|
||
* Источник: [Стандарт: Тексты модулей](https://its.1c.ru/db/v8std#content:456:hdoc) | ||
* Полезная информация: [Отказ от использования модальных окон](https://its.1c.ru/db/metod8dev#content:5272:hdoc) | ||
* Источник: [Cognitive complexity, ver. 1.4](https://www.sonarsource.com/docs/CognitiveComplexity.pdf) --> | ||
|
||
## Snippets | ||
|
||
<!-- Блоки ниже заполняются автоматически, не трогать --> | ||
### Diagnostic ignorance in code | ||
|
||
```bsl | ||
// BSLLS:MetadataBorders-off | ||
// BSLLS:MetadataBorders-on | ||
``` | ||
|
||
### Parameter for config | ||
|
||
```json | ||
"MetadataBorders": { | ||
"metadataBordersParameters": "" | ||
} | ||
``` |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,109 @@ | ||
/* | ||
* This file is a part of BSL Language Server. | ||
* | ||
* Copyright (c) 2018-2021 | ||
* Alexey Sosnoviy <[email protected]>, Nikita Gryzlov <[email protected]> and contributors | ||
* | ||
* SPDX-License-Identifier: LGPL-3.0-or-later | ||
* | ||
* BSL Language Server is free software; you can redistribute it and/or | ||
* modify it under the terms of the GNU Lesser General Public | ||
* License as published by the Free Software Foundation; either | ||
* version 3.0 of the License, or (at your option) any later version. | ||
* | ||
* BSL Language Server is distributed in the hope that it will be useful, | ||
* but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
* Lesser General Public License for more details. | ||
* | ||
* You should have received a copy of the GNU Lesser General Public | ||
* License along with BSL Language Server. | ||
*/ | ||
package com.github._1c_syntax.bsl.languageserver.diagnostics; | ||
|
||
import com.fasterxml.jackson.core.JsonProcessingException; | ||
import com.fasterxml.jackson.databind.ObjectMapper; | ||
import com.fasterxml.jackson.databind.type.MapType; | ||
import com.github._1c_syntax.bsl.languageserver.diagnostics.metadata.DiagnosticMetadata; | ||
import com.github._1c_syntax.bsl.languageserver.diagnostics.metadata.DiagnosticParameter; | ||
import com.github._1c_syntax.bsl.languageserver.diagnostics.metadata.DiagnosticSeverity; | ||
import com.github._1c_syntax.bsl.languageserver.diagnostics.metadata.DiagnosticTag; | ||
import com.github._1c_syntax.bsl.languageserver.diagnostics.metadata.DiagnosticType; | ||
import com.github._1c_syntax.bsl.parser.BSLParser; | ||
import com.github._1c_syntax.utils.CaseInsensitivePattern; | ||
import org.antlr.v4.runtime.tree.ParseTree; | ||
|
||
import java.util.Collections; | ||
import java.util.HashMap; | ||
import java.util.List; | ||
import java.util.Map; | ||
import java.util.regex.Matcher; | ||
import java.util.regex.Pattern; | ||
import java.util.stream.Collectors; | ||
|
||
@DiagnosticMetadata( | ||
type = DiagnosticType.CODE_SMELL, | ||
severity = DiagnosticSeverity.INFO, | ||
minutesToFix = 1, | ||
activatedByDefault = false, | ||
tags = { | ||
DiagnosticTag.DESIGN | ||
} | ||
|
||
) | ||
public class MetadataBordersDiagnostic extends AbstractVisitorDiagnostic { | ||
|
||
private static final String METADATA_BORDERS_DEFAULT = ""; | ||
nixel2007 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
@DiagnosticParameter( | ||
type = String.class, | ||
defaultValue = "" | ||
) | ||
private Map<Pattern, Pattern> metadataBordersParameters = MapFromJSON(METADATA_BORDERS_DEFAULT); | ||
|
||
private List<Pattern> statementPatterns = Collections.emptyList(); | ||
|
||
private static Map<Pattern, Pattern> MapFromJSON(String userSettings) { | ||
Golovanoff marked this conversation as resolved.
Show resolved
Hide resolved
|
||
ObjectMapper mapper = new ObjectMapper(); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. создание ObjectMapper на каждый configure - довольно дорогая операция. лучше его закэшировать или просто указать его конструкторе - его туда подложит спринг. |
||
MapType mapType = mapper.getTypeFactory().constructMapType(HashMap.class, String.class, String.class); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. mapType тоже можно закэшировать в конструторе по принятому инстансу обджект-маппера |
||
try { | ||
Map<String, String> stringMap = mapper.readValue(userSettings, mapType); | ||
return stringMap.entrySet().stream() | ||
.filter(entry -> ! entry.getKey().isBlank() && ! entry.getValue().isBlank()) | ||
Golovanoff marked this conversation as resolved.
Show resolved
Hide resolved
|
||
.collect(Collectors.toMap( | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. пробовал делать включать замер производительности? диагностика не вырывается в топ? |
||
entry -> CaseInsensitivePattern.compile(entry.getKey()), | ||
entry -> CaseInsensitivePattern.compile(entry.getValue()))); | ||
} catch (JsonProcessingException e) { | ||
return Collections.emptyMap(); | ||
} | ||
} | ||
|
||
@Override | ||
public void configure(Map<String, Object> configuration) { | ||
metadataBordersParameters = MapFromJSON( | ||
(String) configuration.getOrDefault("metadataBordersParameters", METADATA_BORDERS_DEFAULT)); | ||
} | ||
|
||
@Override | ||
public ParseTree visitFile(BSLParser.FileContext ctx) { | ||
statementPatterns = metadataBordersParameters.entrySet().stream() | ||
.filter(entry -> ! entry.getValue().matcher(this.documentContext.getUri().getPath()).find()) | ||
Golovanoff marked this conversation as resolved.
Show resolved
Hide resolved
|
||
.map(Map.Entry::getKey) | ||
.collect(Collectors.toList()); | ||
|
||
return super.visitFile(ctx); | ||
} | ||
|
||
@Override | ||
public ParseTree visitStatement(BSLParser.StatementContext ctx){ | ||
|
||
statementPatterns.forEach(pattern -> { | ||
Matcher matcher = pattern.matcher(ctx.getText()); | ||
while (matcher.find()) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. вообще не понял - а цикл зачем? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. потому что матчер возвращает срабатывание по одному за раз There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. а зачем одну и туже диагностику в цикле добавлять? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 100 раз "addDiagnostic(ctx)" There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. так, стоп, тут же где-то был другой код, когда в addDiagnostic передается start() и end() от матчера... или это в соседней диагностике?) |
||
diagnosticStorage.addDiagnostic(ctx); | ||
} | ||
}); | ||
|
||
return super.visitStatement(ctx); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. если ты проверяешь текст statement целиком, то нет смысла дергать super, т.к. вложенные стэйтменты проверят тот же самый кусок текста. т.е. здесь будет фонить на вложенных конструкциях вида Если Истина Тогда
Если Истина Тогда
ЗапрещенныйВызов(); // тут кинет два срабатывания на одно и то же место
КонецЕсли;
КонецЕсли; |
||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
diagnosticMessage=Use metadata outside the borders | ||
diagnosticName=Metadata borders | ||
metadataBordersParameters=JSON-structure for pairs "regex for statements":"regex for module names". | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
diagnosticMessage=Обращение к метаданным за пределами разрешенных границ | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. может быть в сообщение стоит добавить регулярку/пару регулярок, на которой произошло срабатывание? |
||
diagnosticName=Границы метаданных | ||
metadataBordersParameters=JSON-структура для пар "регулярное выражение для операторов":"регулярное выражение для имени файла модуля" |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,86 @@ | ||
/* | ||
* This file is a part of BSL Language Server. | ||
* | ||
* Copyright (c) 2018-2021 | ||
* Alexey Sosnoviy <[email protected]>, Nikita Gryzlov <[email protected]> and contributors | ||
* | ||
* SPDX-License-Identifier: LGPL-3.0-or-later | ||
* | ||
* BSL Language Server is free software; you can redistribute it and/or | ||
* modify it under the terms of the GNU Lesser General Public | ||
* License as published by the Free Software Foundation; either | ||
* version 3.0 of the License, or (at your option) any later version. | ||
* | ||
* BSL Language Server is distributed in the hope that it will be useful, | ||
* but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
* Lesser General Public License for more details. | ||
* | ||
* You should have received a copy of the GNU Lesser General Public | ||
* License along with BSL Language Server. | ||
*/ | ||
package com.github._1c_syntax.bsl.languageserver.diagnostics; | ||
|
||
import org.eclipse.lsp4j.Diagnostic; | ||
import org.junit.jupiter.api.Test; | ||
|
||
import java.util.List; | ||
import java.util.Map; | ||
|
||
import static com.github._1c_syntax.bsl.languageserver.util.Assertions.assertThat; | ||
|
||
class MetadataBordersDiagnosticTest extends AbstractDiagnosticTest<MetadataBordersDiagnostic> { | ||
MetadataBordersDiagnosticTest() { | ||
super(MetadataBordersDiagnostic.class); | ||
} | ||
|
||
@Test | ||
void testWithoutSettings() { | ||
List<Diagnostic> diagnostics = getDiagnostics(); | ||
assertThat(diagnostics).hasSize(0); | ||
} | ||
|
||
@Test | ||
void testMatchesInProperModule() { | ||
|
||
Map<String, Object> configuration = diagnosticInstance.info.getDefaultConfiguration(); | ||
configuration.put("metadataBordersParameters", | ||
"{\"Регистры?Сведений.КонтактнаяИнформация|Справочники?.Контакты\":\"fake-uri\"}"); | ||
diagnosticInstance.configure(configuration); | ||
List<Diagnostic> diagnostics = getDiagnostics(); | ||
|
||
assertThat(diagnostics).hasSize(0); | ||
} | ||
|
||
@Test | ||
void testMatchesInWrongModule() { | ||
|
||
Map<String, Object> configuration = diagnosticInstance.info.getDefaultConfiguration(); | ||
configuration.put("metadataBordersParameters", | ||
"{\"Регистры?Сведений.КонтактнаяИнформация|Справочники?.Контакты\":\"MetadataBordersDiagnostic\"}"); | ||
diagnosticInstance.configure(configuration); | ||
List<Diagnostic> diagnostics = getDiagnostics(); | ||
|
||
assertThat(diagnostics).hasSize(3); | ||
assertThat(diagnostics, true) | ||
.hasRange(3, 0, 6, 49) | ||
.hasRange(8, 0, 8, 69) | ||
.hasRange(10, 0, 10, 103); | ||
} | ||
|
||
@Test | ||
void testSeveralSettings() { | ||
|
||
Map<String, Object> configuration = diagnosticInstance.info.getDefaultConfiguration(); | ||
configuration.put("metadataBordersParameters", | ||
"{\"Регистры?Сведений.КонтактнаяИнформация\":\"fake-uri\"," + | ||
"\"Справочники?.Контакты\":\"MetadataBordersDiagnostic\"}"); | ||
diagnosticInstance.configure(configuration); | ||
|
||
List<Diagnostic> diagnostics = getDiagnostics(); | ||
|
||
assertThat(diagnostics).hasSize(1); | ||
assertThat(diagnostics, true) | ||
.hasRange(10, 0, 10, 103); | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Где-то в ранних версиях здесь был пример заполнения. Кажется, стоит его вернуть. Плюс добавить
{}