Skip to content
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

Fix groovy multivariable declaration #4757

Merged
merged 26 commits into from
Dec 17, 2024
Merged
Show file tree
Hide file tree
Changes from 16 commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
44c138b
Fix parsing of multiple variable assignments in one statement
Laurens-W Dec 9, 2024
e2ac9f4
Minor improvement
Laurens-W Dec 9, 2024
f9f3cd3
Add another testcase and make sure it works
Laurens-W Dec 9, 2024
9274748
Simplify
Laurens-W Dec 9, 2024
eb95dbb
Clarify
Laurens-W Dec 9, 2024
0685d0e
Revert formatting
Laurens-W Dec 9, 2024
06973e1
Apply suggestions from code review
Laurens-W Dec 9, 2024
a0da837
Fixed
Laurens-W Dec 10, 2024
d9856a8
Fixed
Laurens-W Dec 10, 2024
ff0510a
Move tests to AssignmentTest + add `final def` case
jevanlingen Dec 10, 2024
27e95f1
Should have done it this way initially ;)
Laurens-W Dec 10, 2024
4278625
Add testcase for regression
Laurens-W Dec 10, 2024
d22d5ff
Polish
Laurens-W Dec 10, 2024
3553b1f
Extra testcase
Laurens-W Dec 10, 2024
bbdd436
Add `noKeyword` test
jevanlingen Dec 10, 2024
3786a05
Return String instead of passing StringBuilder
Laurens-W Dec 10, 2024
addfd85
Further convert to using String
Laurens-W Dec 10, 2024
422df9e
Simplify
jevanlingen Dec 11, 2024
0a92ffe
Merge branch 'main' into fix-groovy-multivariable-declaration
jevanlingen Dec 11, 2024
f301c34
Merge branch 'main' into fix-groovy-multivariable-declaration
timtebeek Dec 11, 2024
f196837
Different approach using marker
Laurens-W Dec 12, 2024
9b906d2
Remove unused code and FMT
Laurens-W Dec 12, 2024
7cd916b
Merge branch 'main' into fix-groovy-multivariable-declaration
Laurens-W Dec 13, 2024
b371832
Merge branch 'main' into fix-groovy-multivariable-declaration
timtebeek Dec 15, 2024
f4bb9b7
Remove specific exception cases for VariableDeclarations
Laurens-W Dec 16, 2024
7de11c8
Change to startsWith with offset
Laurens-W Dec 17, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -39,9 +39,9 @@
import org.openrewrite.java.marker.OmitParentheses;
import org.openrewrite.java.marker.Semicolon;
import org.openrewrite.java.marker.TrailingComma;
import org.openrewrite.java.tree.*;
import org.openrewrite.java.tree.Expression;
import org.openrewrite.java.tree.Statement;
import org.openrewrite.java.tree.*;
import org.openrewrite.marker.Markers;

import java.math.BigDecimal;
Expand Down Expand Up @@ -113,16 +113,16 @@ private static boolean isOlderThanGroovy3() {
}

/**
* Groovy methods can be declared with "def" AND a return type
* In these cases the "def" is semantically meaningless but needs to be preserved for source code accuracy
* If there is both a def and a return type, this method returns a RedundantDef object and advances the cursor
* position past the "def" keyword, leaving the return type to be parsed as normal.
* In any other situation an empty Optional is returned and the cursor is not advanced.
* Groovy methods can be declared with "def" AND a return type
* In these cases the "def" is semantically meaningless but needs to be preserved for source code accuracy
* If there is both a def and a return type, this method returns a RedundantDef object and advances the cursor
* position past the "def" keyword, leaving the return type to be parsed as normal.
* In any other situation an empty Optional is returned and the cursor is not advanced.
*/
private Optional<RedundantDef> maybeRedundantDef(ClassNode type, String name) {
int saveCursor = cursor;
Space defPrefix = whitespace();
if(source.startsWith("def", cursor)) {
if (source.startsWith("def", cursor)) {
skip("def");
// The def is redundant only when it is followed by the method's return type
// I hope no one puts an annotation between "def" and the return type
Expand Down Expand Up @@ -182,8 +182,8 @@ public G.CompilationUnit visit(SourceUnit unit, ModuleNode ast) throws GroovyPar
for (ClassNode aClass : ast.getClasses()) {
if (aClass.getSuperClass() == null ||
!("groovy.lang.Script".equals(aClass.getSuperClass().getName()) ||
"RewriteGradleProject".equals(aClass.getSuperClass().getName()) ||
"RewriteSettings".equals(aClass.getSuperClass().getName()))) {
"RewriteGradleProject".equals(aClass.getSuperClass().getName()) ||
"RewriteSettings".equals(aClass.getSuperClass().getName()))) {
sortedByPosition.computeIfAbsent(pos(aClass), i -> new ArrayList<>()).add(aClass);
}
}
Expand Down Expand Up @@ -822,8 +822,8 @@ public void visitArgumentlistExpression(ArgumentListExpression expression) {
// When named parameters are in use they may appear before, after, or intermixed with any positional arguments
if (unparsedArgs.size() > 1 && unparsedArgs.get(0) instanceof MapExpression &&
(unparsedArgs.get(0).getLastLineNumber() > unparsedArgs.get(1).getLastLineNumber() ||
(unparsedArgs.get(0).getLastLineNumber() == unparsedArgs.get(1).getLastLineNumber() &&
unparsedArgs.get(0).getLastColumnNumber() > unparsedArgs.get(1).getLastColumnNumber()))) {
(unparsedArgs.get(0).getLastLineNumber() == unparsedArgs.get(1).getLastLineNumber() &&
unparsedArgs.get(0).getLastColumnNumber() > unparsedArgs.get(1).getLastColumnNumber()))) {

// Figure out the source-code ordering of the expressions
MapExpression namedArgExpressions = (MapExpression) unparsedArgs.get(0);
Expand Down Expand Up @@ -1079,7 +1079,7 @@ public void visitBlockStatement(BlockStatement block) {
J expr = visit(statement);
if (i == blockStatements.size() - 1 && (expr instanceof Expression)) {
if (parent instanceof ClosureExpression || (parent instanceof MethodNode &&
!JavaType.Primitive.Void.equals(typeMapping.type(((MethodNode) parent).getReturnType())))) {
!JavaType.Primitive.Void.equals(typeMapping.type(((MethodNode) parent).getReturnType())))) {
expr = new J.Return(randomId(), expr.getPrefix(), Markers.EMPTY,
expr.withPrefix(EMPTY));
expr = expr.withMarkers(expr.getMarkers().add(new ImplicitReturn(randomId())));
Expand Down Expand Up @@ -1419,6 +1419,10 @@ public void visitNotExpression(NotExpression expression) {
@Override
public void visitDeclarationExpression(DeclarationExpression expression) {
Optional<RedundantDef> redundantDef = maybeRedundantDef(expression.getVariableExpression().getType(), expression.getVariableExpression().getName());

/* The identifier of this type is one of the following: the name of a type, `final`, `def`, `var` or `,`
The latter because MultiVariable declarations are returned as separate `DeclarationExpression` by the
Groovy AST */
TypeTree typeExpr = visitVariableExpressionType(expression.getVariableExpression());

J.VariableDeclarations.NamedVariable namedVariable;
Expand Down Expand Up @@ -1746,7 +1750,7 @@ public void visitMethodCallExpression(MethodCallExpression call) {
MethodNode methodNode = (MethodNode) call.getNodeMetaData().get(StaticTypesMarker.DIRECT_METHOD_CALL_TARGET);
JavaType.Method methodType = null;
if (methodNode == null && call.getObjectExpression() instanceof VariableExpression &&
((VariableExpression) call.getObjectExpression()).getAccessedVariable() != null) {
((VariableExpression) call.getObjectExpression()).getAccessedVariable() != null) {
// Groovy doesn't know what kind of object this method is being invoked on
// But if this invocation is inside a Closure we may have already enriched its parameters with types from the static type checker
// Use any such type information to attempt to find a matching method
Expand Down Expand Up @@ -2077,29 +2081,20 @@ public void visitPrefixExpression(PrefixExpression unary) {

public TypeTree visitVariableExpressionType(VariableExpression expression) {
JavaType type = typeMapping.type(staticType(((org.codehaus.groovy.ast.expr.Expression) expression)));
Space prefix = whitespace();
StringBuilder keyword = new StringBuilder();
Laurens-W marked this conversation as resolved.
Show resolved Hide resolved

if (expression.isDynamicTyped()) {
Space prefix = whitespace();
String keyword;
if (source.substring(cursor).startsWith("final")) {
keyword = "final";
} else {
keyword = source.substring(cursor, cursor + 3);
}
if (expression.isDynamicTyped() || source.charAt(cursor) == ',') {
keyword.append(getKeyword(expression.getName()));
} else {
keyword.append(expression.getOriginType().getUnresolvedName());
cursor += keyword.length();
return new J.Identifier(randomId(),
prefix,
Markers.EMPTY,
emptyList(),
keyword,
type, null);
}
Space prefix = sourceBefore(expression.getOriginType().getUnresolvedName());
J.Identifier ident = new J.Identifier(randomId(),
EMPTY,
Markers.EMPTY,
emptyList(),
expression.getOriginType().getUnresolvedName(),
keyword.toString(),
type, null);
if (expression.getOriginType().getGenericsTypes() != null) {
return new J.ParameterizedType(randomId(), prefix, Markers.EMPTY, ident, visitTypeParameterizations(
Expand Down Expand Up @@ -2176,6 +2171,35 @@ private <T> T pollQueue() {
}
}

private String getKeyword(String variableName) {
String retVal;
String leftover = source.substring(cursor);
Laurens-W marked this conversation as resolved.
Show resolved Hide resolved
if (leftover.startsWith("final")) {
retVal = "final";
cursor += 5;
} else if (leftover.startsWith("var")) {
retVal = "var";
cursor += 3;
} else if (leftover.startsWith("def")) {
retVal = "def";
cursor += 3;
} else if (leftover.startsWith(",")) {
retVal = ",";
cursor++;
} else {
throw new IllegalStateException("Ran into unknown or unimplemented identifier at " + leftover.substring(0, 10));
}
int saveCursor = cursor;
Space space = whitespace();
if (source.substring(cursor).startsWith(variableName)) {
cursor = saveCursor;
} else {
retVal += space.getWhitespace();
retVal += getKeyword(variableName);
}
jevanlingen marked this conversation as resolved.
Show resolved Hide resolved
return retVal;
}

// handle the obscure case where there are empty parens ahead of a closure
private Markers handlesCaseWhereEmptyParensAheadOfClosure(ArgumentListExpression args, Markers markers) {
if (args.getExpressions().size() == 1 && args.getExpressions().get(0) instanceof ClosureExpression) {
Expand Down Expand Up @@ -2646,7 +2670,7 @@ private String name() {
Can contain a J.FieldAccess, as in a variable declaration with fully qualified type parameterization:
List<java.lang.String>
*/
private JContainer<Expression> visitTypeParameterizations(GenericsType @Nullable[] genericsTypes) {
private JContainer<Expression> visitTypeParameterizations(GenericsType @Nullable [] genericsTypes) {
Space prefix = sourceBefore("<");
List<JRightPadded<Expression>> parameters;

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -295,16 +295,16 @@ public J visitLambda(J.Lambda lambda, PrintOutputCapture<P> p) {
LambdaStyle ls = lambda.getMarkers().findFirst(LambdaStyle.class)
.orElse(new LambdaStyle(null, false, !lambda.getParameters().getParameters().isEmpty()));
boolean parenthesized = lambda.getParameters().isParenthesized();
if(!ls.isJavaStyle()) {
if (!ls.isJavaStyle()) {
p.append('{');
}
visitMarkers(lambda.getParameters().getMarkers(), p);
visitSpace(lambda.getParameters().getPrefix(), Space.Location.LAMBDA_PARAMETERS_PREFIX, p);
if(parenthesized) {
if (parenthesized) {
p.append('(');
}
visitRightPadded(lambda.getParameters().getPadding().getParameters(), JRightPadded.Location.LAMBDA_PARAM, ",", p);
if(parenthesized) {
if (parenthesized) {
p.append(')');
}
if (ls.isArrow()) {
Expand All @@ -318,7 +318,7 @@ public J visitLambda(J.Lambda lambda, PrintOutputCapture<P> p) {
} else {
visit(lambda.getBody(), p);
}
if(!ls.isJavaStyle()) {
if (!ls.isJavaStyle()) {
p.append('}');
}
afterSyntax(lambda, p);
Expand Down Expand Up @@ -358,6 +358,7 @@ public J visitForEachLoop(J.ForEachLoop forEachLoop, PrintOutputCapture<P> p) {
afterSyntax(forEachLoop, p);
return forEachLoop;
}

@Override
public J visitMethodDeclaration(J.MethodDeclaration method, PrintOutputCapture<P> p) {
beforeSyntax(method, Space.Location.METHOD_DECLARATION_PREFIX, p);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,14 +16,58 @@
package org.openrewrite.groovy.tree;

import org.junit.jupiter.api.Test;
import org.openrewrite.InMemoryExecutionContext;
import org.openrewrite.Issue;
import org.openrewrite.Parser;
import org.openrewrite.groovy.GroovyParsingException;
import org.openrewrite.test.RewriteTest;
import org.openrewrite.test.SourceSpec;

import java.nio.file.Paths;
import java.util.List;

import static org.assertj.core.api.Assertions.assertThatThrownBy;
import static org.openrewrite.groovy.Assertions.groovy;
import static org.openrewrite.internal.StringUtils.trimIndentPreserveCRLF;

@SuppressWarnings({"GroovyUnusedAssignment", "GrUnnecessarySemicolon"})
class AssignmentTest implements RewriteTest {

@Test
void noKeyword() {
rewriteRun(
groovy(
"""
x = "s"
"""
)
);
}

@Test
void simple() {
rewriteRun(
groovy(
"""
def x = "s"
"""
)
);
}

@Test
void simpleWithFinal() {
rewriteRun(
groovy(
"""
final def x = "x"
def final y = "y"
Laurens-W marked this conversation as resolved.
Show resolved Hide resolved
final z = "z"
"""
Comment on lines +53 to +57
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Perhaps good to call out here that we now model final def and def final as two unique concatenated Identifiers, which seems odd to me:

----G.CompilationUnit | "G.CompilationUnit(typesInUse=org.openrewrite.java.internal.TypesInUse@745c2004, padding=org.openrewrite.groovy.tree.G..."
    |---J.VariableDeclarations | "final def x = "x""
    |   |---J.Identifier | "final def"
    |   \-------J.VariableDeclarations.NamedVariable | "x = "x""
    |           |---J.Identifier | "x"
    |           \-------J.Literal | ""x""
    |---J.VariableDeclarations | "final y = "y""
    |   |---J.Identifier | "final"
    |   \-------J.VariableDeclarations.NamedVariable | "y = "y""
    |           |---J.Identifier | "y"
    |           \-------J.Literal | ""y""
    \---J.VariableDeclarations | "final z = "z""
        |---J.Identifier | "final"
        \-------J.VariableDeclarations.NamedVariable | "z = "z""
                |---J.Identifier | "z"
                \-------J.Literal | ""z""

Note that I don't expect this to occur often, if at all, and it still prints OK; but figured comment it here for a closer look by others reviewing as well.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

While this isn't perfect either, I think I would opt for mapping all of final, def, and var to modifiers (J.Modifier) and instead deprecating the RedundantDef marker, while still supporting it in the printer to remain backwards compatible.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Implemented this. I'm not sure if we can actually mark the RedundantDef marker as deprecated though as that may also apply to declared methods, so I left that in.

As we discussed we may want to find a way to combine consecutive DeclarationExpressions into one J.VariableDeclarations. That could be a further improvement and I'm willing to work on that, but want to return focus to the parenthesis issue as well :)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

At least marking the RedundantDef marker as @deprecated should not be a problem. If you have the option to remove the maybeRedundantDef method as well, all the better. As long as you keep the RedundantDef marker in printer, you are still backwards compatible.

__
Also notice the RedundantDef marker has ben added lately (05/12/2024). So the sooner we no longer use it, the less serialised LSTs do have this marker.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@sambsnyd Since you recently added that RedundantDef, do you also want to chime in here and tell us what your thoughts are about this?

)
);
}

@Test
void concat() {
rewriteRun(
Expand Down Expand Up @@ -91,4 +135,52 @@ void baseNConversions() {
)
);
}

@Test
void multipleAssignmentsAtOneLine() {
rewriteRun(
groovy(
"""
def startItem = '| ', endItem = ' |'
def repeatLength = startItem.length() + output.length() + endItem.length()
println("\\n" + ("-" * repeatLength) + "\\n" + startItem + output + endItem + "\\n" + ("-" * repeatLength))
"""
)
);
}

@Test
void multipleAssignmentsAtOneLineSimple() {
rewriteRun(
groovy(
"""
def a = '1', b = '2'
"""
)
);
}

@Test
void multipleAssignmentsAtMultipleLineDynamicType() {
rewriteRun(
groovy(
"""
def a = '1' ,
b = '2'
"""
)
);
}

@Test
void multipleAssignmentsAtMultipleLineStaticType() {
rewriteRun(
groovy(
"""
String a = '1' ,
b = '2'
"""
)
);
}
}
Loading