Skip to content

Latest commit

 

History

History
171 lines (127 loc) · 7.83 KB

README.md

File metadata and controls

171 lines (127 loc) · 7.83 KB

Rewrite - Distributed Java Source Refactoring

Build Status Gitter Apache 2.0

The Rewrite project is a refactoring tool for Java source code. It contains a custom Abstract Syntax Tree (AST) supporting Java 8 language features that encodes the structure and formatting of your source code. The AST is printable to reconstitute the source code, including its original formatting.

Rewrite provides high-level search functions and refactoring functions that can transform the AST.

The AST is imbued with information about types (and their type hierarchies) of expressions and statements in your code.

Rewrite provides visitor support over its AST. Basic visitors for printing the AST, transforming it with refactoring operations, etc. are provided out of the box.

Rewrite provides utilities for unit testing refactoring logic and custom visitors.

At Netflix, we operate on thousands of repositories with refactoring rules written in Rewrite, issuing pull requests en masse from an Apache Spark cluster. It is possible to execute these operations as part of your build process, to incorporate them into IDE plugins, etc. depending on the needs of your organization.

To maximize the freedom of our engineers to incorporate changes at their convenience, Rewrite's objective is to facilitate the incorporation of refactoring changes by the affected team or engineer. In this way, we consider these operations to be distributed refactoring.

This is an incubating feature.

Installing

Install the dependency from Maven Central or JCenter with:

<dependency>
    <groupId>com.netflix.devinsight.rewrite</groupId>
    <artifactId>rewrite-core</artifactId>
    <version>1.2.0</version>
</dependency>

or

compile 'com.netflix.devinsight.rewrite:rewrite-core:1.2.0'

Add the Maven or Gradle classifier jdkbundle to fetch a version of the package that package relocates and shades the relevant parts of the JDK needed for parsing into the distribution.

Features

Example usage

Below is a simple example of a refactoring operation that changes the name of a method.

public class ChangeMethodNameTestJava {
    Parser parser = new OracleJdkParser(); // pass binary dependencies to this constructor on a real project

    @Test
    public void refactorMethodName() {
        String a = "class A {{ B.foo(0); }}";
        String b = "class B { static void foo(int n) {} }";

        Tr.CompilationUnit cu = parser.parse(a, /* which depends on */ b);

        Refactor refactor = cu.refactor()
          .changeMethodName(cu.findMethodCalls("B foo(int)"), "bar");

        Tr.CompilationUnit fixed = refactor.fix();

        assertEquals(fixed.print(), "class A {{ B.bar(0); }}");
    }
}

First, we construct a Parser instance, in this case an OracleJdkParser which will tightly control the parsing and type attribution phases of the standard Oracle JDK to produce an abstract syntax tree (AST) that we can work with. If we were working with a real project, we would pass a List<Path> of the binary dependencies of the project to the OracleJdkParser constructor.

Next, we use the parser to parse some source code. Typically, you would pass a List<Path> of all the source files in the project, and parse would return a List<Tr.CompilationUnit> representing the ASTs of each source file in order. Here we are using a convenience utility that is especially handy while writing tests that allows us to pass any number of strings each representing a different Java source file, and we will receive back a single Tr.CompilationUnit for the first source string in the list. Note that a compilation unit is a combination of package declaration, imports, and all of the types (classes, interfaces, enums, etc.) defined in the file. Remember that, while usually there is one type per file whose name matches the file name, Java does allow additional non-public types to be defined inside a single Java source file.

At this point, we can use the Tr.CompilationUnit to either do a type-aware deep dive on the code or perform a refactoring operation. Here, we begin a refactoring operation by calling refactor on our compilation unit. We search for method invocations matching a certain signature (using the AspectJ pointcut grammar), and for each matching invocation, change the name to a method called bar.

Next, we call fix to return a copy of the original AST with the refactoring changes made.

Lastly, we call print on the AST to emit the source code for the resulting change. Notice how the original style of the class was preserved!

To cut down a bit on the ceremony, we can shorten the process of setting up and executing a refactor operation into one call chain:

Tr.CompilationUnit fixed = cu.refactor().changeName(cu.findMethodCalls("B foo(int)"), "bar").fix();

Generating a git-style patch

Rather than calling fix on a Refactor instance as described above, you may also call diff() which generates a git-style patch that can be used to generate a pull request or submit a patch for review and integration later. We also use diff when performing a refactoring operation across thousands of projects to wrap our heads around what kinds of changes are going to be made to the source code and sanity-check that our refactoring logic makes sense.

Let's adjust our definition of the A class in our above example to incorporate some newlines, so that it looks like this:

class A {
   {
      B.foo(0);
   }
}

Performing the same refactoring operation, but calling diff instead would yield:

diff --git a//home/A.java b//home/A.java
index 9b034e8..0234fb8 100644
--- a//home/A.java
+++ b//home/A.java
@@ -1,5 +1,5 @@
 class A {
    {
-      B.foo(0);
+      B.bar(0);
    }
 }
\ No newline at end of file

License

Copyright 2015-2016 Netflix, Inc.

Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at

http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.