Skip to content

Commit

Permalink
исключение Возврат Дата
Browse files Browse the repository at this point in the history
улучшено описание правила
  • Loading branch information
artbear committed Dec 15, 2023
1 parent 3598cc3 commit 6d21f39
Show file tree
Hide file tree
Showing 3 changed files with 87 additions and 64 deletions.
13 changes: 13 additions & 0 deletions docs/diagnostics/MagicDate.md
Original file line number Diff line number Diff line change
Expand Up @@ -22,3 +22,16 @@
ХоверБордБудетИзобретен = Неопределено;
КонецЕсли;
```

Также хорошим решением является использование специального метода с говорящим названием, который возвращает
дату-константу

```bsl
Функция ДатаИзобретенияХовера()
Возврат '20151021';
КонецФункции
Если текДата < ДатаИзобретенияХовера() Тогда
ХоверБордБудетИзобретен = Неопределено;
КонецЕсли;
```
Original file line number Diff line number Diff line change
Expand Up @@ -72,42 +72,33 @@ public class MagicDateDiagnostic extends AbstractVisitorDiagnostic {
)
private final Set<String> authorizedDates = new HashSet<>(Arrays.asList(DEFAULT_AUTHORIZED_DATES.split(",")));

private static Optional<BSLParserRuleContext> getExpression(Optional<BSLParser.ConstValueContext> contextOptional) {
return contextOptional
.map(BSLParserRuleContext::getParent)
.filter(context -> context.getChildCount() == 1)
.map(BSLParserRuleContext::getParent)
.filter(context -> context.getChildCount() == 1)
.filter(BSLParser.ExpressionContext.class::isInstance);
@Override
public void configure(Map<String, Object> configuration) {
var authorizedDatesString = (String) configuration.getOrDefault("authorizedDates", DEFAULT_AUTHORIZED_DATES);
Set<String> authD = Arrays.stream(authorizedDatesString.split(","))
.map(String::trim)
.collect(Collectors.toSet());
authorizedDates.clear();
authorizedDates.addAll(authD);
}

private static boolean insideSimpleDateAssignment(Optional<BSLParserRuleContext> expressionContext) {
return expressionContext
.map(BSLParserRuleContext::getParent)
.filter(BSLParser.AssignmentContext.class::isInstance)
.isPresent();
}
@Override
public ParseTree visitConstValue(BSLParser.ConstValueContext ctx) {
var tNode = ctx.DATETIME();
var sNode = ctx.string();
if ((tNode != null || sNode != null) && isAccepted(ctx)) {
if (sNode != null && !isValidDate(sNode)) {
return defaultResult();
}

private static boolean insideAssignmentWithDateMethodForSimpleDate(Optional<BSLParserRuleContext> expressionContext) {
return expressionContext
.map(BSLParserRuleContext::getParent) // callParam
.filter(context -> context.getChildCount() == 1)
.map(BSLParserRuleContext::getParent) // callParamList
.filter(context -> context.getChildCount() == 1)
.map(BSLParserRuleContext::getParent) // doCall
.map(BSLParserRuleContext::getParent) // globalCall - метод Дата(ХХХ)
.filter(BSLParser.GlobalMethodCallContext.class::isInstance)
.map(BSLParser.GlobalMethodCallContext.class::cast)
.filter(context -> methodPattern.matcher(context.methodName().getText()).matches())
.map(BSLParserRuleContext::getParent) // complexId
.filter(context -> context.getChildCount() == 1)
.map(BSLParserRuleContext::getParent) // member
.filter(context -> context.getChildCount() == 1)
.map(BSLParserRuleContext::getParent) // expression
.filter(context -> context.getChildCount() == 1)
.map(BSLParserRuleContext::getParent)
.filter(BSLParser.AssignmentContext.class::isInstance)
.isPresent();
final var expressionContext = getExpression(Optional.of(ctx));
if (!insideSimpleDateAssignment(expressionContext) && !insideReturnSimpleDate(expressionContext)
&& !insideAssignmentWithDateMethodForSimpleDate(expressionContext)) {
diagnosticStorage.addDiagnostic(ctx, info.getMessage(ctx.getText()));
}
}

return defaultResult();
}

private static boolean isValidDate(BSLParser.StringContext ctx) {
Expand Down Expand Up @@ -147,35 +138,6 @@ private static int parseInt(String text) {
}
}

@Override
public void configure(Map<String, Object> configuration) {
var authorizedDatesString = (String) configuration.getOrDefault("authorizedDates", DEFAULT_AUTHORIZED_DATES);
Set<String> authD = Arrays.stream(authorizedDatesString.split(","))
.map(String::trim)
.collect(Collectors.toSet());
authorizedDates.clear();
authorizedDates.addAll(authD);
}

@Override
public ParseTree visitConstValue(BSLParser.ConstValueContext ctx) {
var tNode = ctx.DATETIME();
var sNode = ctx.string();
if ((tNode != null || sNode != null) && isAccepted(ctx)) {
if (sNode != null && !isValidDate(sNode)) {
return defaultResult();
}

final var expressionContext = getExpression(Optional.of(ctx));
if (!insideSimpleDateAssignment(expressionContext)
&& !insideAssignmentWithDateMethodForSimpleDate(expressionContext)) {
diagnosticStorage.addDiagnostic(ctx, info.getMessage(ctx.getText()));
}
}

return defaultResult();
}

private boolean isAccepted(BSLParser.ConstValueContext ctx) {
String text = ctx.getText();
return text != null && !text.isEmpty() && !isExcluded(text);
Expand All @@ -185,4 +147,52 @@ private boolean isExcluded(String text) {
String s = nonNumberPattern.matcher(text).replaceAll("");
return authorizedDates.contains(s);
}

private static Optional<BSLParser.ExpressionContext> getExpression(Optional<BSLParser.ConstValueContext> constValue) {
return constValue
.map(BSLParserRuleContext::getParent)
.filter(context -> context.getChildCount() == 1)
.map(BSLParserRuleContext::getParent)
.filter(context -> context.getChildCount() == 1)
.filter(BSLParser.ExpressionContext.class::isInstance)
.map(BSLParser.ExpressionContext.class::cast);
}

private static boolean insideSimpleDateAssignment(Optional<BSLParser.ExpressionContext> expression) {
return insideContext(expression, BSLParser.AssignmentContext.class);
}

private static boolean insideContext(Optional<BSLParser.ExpressionContext> expression,
Class<? extends BSLParserRuleContext> assignmentContextClass) {
return expression
.map(BSLParserRuleContext::getParent)
.filter(assignmentContextClass::isInstance)
.isPresent();
}

private static boolean insideReturnSimpleDate(Optional<BSLParser.ExpressionContext> expression) {
return insideContext(expression, BSLParser.ReturnStatementContext.class);
}

private static boolean insideAssignmentWithDateMethodForSimpleDate(Optional<BSLParser.ExpressionContext> expression) {
return expression
.map(BSLParserRuleContext::getParent) // callParam
.filter(context -> context.getChildCount() == 1)
.map(BSLParserRuleContext::getParent) // callParamList
.filter(context -> context.getChildCount() == 1)
.map(BSLParserRuleContext::getParent) // doCall
.map(BSLParserRuleContext::getParent) // globalCall - метод Дата(ХХХ)
.filter(BSLParser.GlobalMethodCallContext.class::isInstance)
.map(BSLParser.GlobalMethodCallContext.class::cast)
.filter(context -> methodPattern.matcher(context.methodName().getText()).matches())
.map(BSLParserRuleContext::getParent) // complexId
.filter(context -> context.getChildCount() == 1)
.map(BSLParserRuleContext::getParent) // member
.filter(context -> context.getChildCount() == 1)
.map(BSLParserRuleContext::getParent) // expression
.filter(context -> context.getChildCount() == 1)
.map(BSLParserRuleContext::getParent)
.filter(BSLParser.AssignmentContext.class::isInstance)
.isPresent();
}
}
4 changes: 2 additions & 2 deletions src/test/resources/diagnostics/MagicDateDiagnostic.bsl
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
Функция МаксимальнаяДата() Экспорт
//Возврат '39991231235959'; // TODO
Возврат '39991231235959';
КонецФункции

Функция МаксимальнаяДатаПриПродолжении()
//Возврат '39990101000000'; // TODO
Возврат '39990101000000';
КонецФункции

Процедура Тест3()
Expand Down

0 comments on commit 6d21f39

Please sign in to comment.