Skip to content

selesse/jxlint

Repository files navigation

jxlint

Build status

jxlint is a Java framework for performing static analysis. You create your custom validation logic in Java, specify which files to perform this validation on, and add some documentation. The following "boring" stuff is taken care of for you:

usage: jxlint [flags] <directory>
 -h,--help                     Usage information, help message.
 -v,--version                  Output version information.
 -p,--profile                  Measure time every rule takes to complete.
 -l,--list                     Lists lint rules with a short, summary
                               explanation.
 -b,--web <port>               Run in the background, as a website.
                               (default port: 8380)
 -r,--rules                    Prints a Markdown dump of the program's
                               rules.
 -s,--show <RULE[s]>           Lists a verbose rule explanation.
 -c,--check <RULE[s]>          Only check for these rules.
 -d,--disable <RULE[s]>        Disable the list of rules.
 -e,--enable <RULE[s]>         Enable the list of rules.
 -y,--category <CATEGORY[s]>   Run all rules of a certain category.
 -w,--nowarn                   Only check for errors; ignore warnings.
 -Wall,--Wall                  Check all warnings, including those off by
                               default.
 -Werror,--Werror              Treat all warnings as errors.
 -q,--quiet                    Don't output any progress or reports.
 -t,--html <filename>          Create an HTML report.
 -x,--xml <filename>           Create an XML (!!) report.
 -j,--jenkins-xml <filename>   Create an XML Jenkins format (!!) report.
 -sp,--srcpath <PATH-PREFIX>   Local or remote path to the source
                               directory, if not set a relative path to
                               the local file will be computed.

<RULE[s]> should be comma separated, without spaces.
<PATH-PREFIX>:
Links to the source code files will use this value as value as prefix.
Possible values:
 - relative path: ../../my-project/
 - absolute path: file:///C:/work/my-project/
 - online path: https://github.com/selesse/jxlint/blob/master/jxlint-impl/

Exit Status:
0                     Success
1                     Failed
2                     Command line error

Motivation

The goal of this framework is to provide an easy, intuitive way to perform file validation. A user specifies a set of "rules" (i.e. validations) that should be executed against a directory. Rules contain three major pieces of information: which files to validate, what the actual validation logic is, and documentation explaining its purpose.

Another goal is to have an architecture that encouraged rules to be self-contained. Anybody interested in verifying, adding, or removing existing rules should be able to do so quickly (i.e. everything is located where you'd expect), and efficiently (i.e. adding new rules is intuitive and simple, requiring little overhead). Additionally, adding tests to validate rules should be straightforward.

Potential Users

  • If you are working with a framework that relies on XML for various parts of the system and you'd like to verify that certain configurations are correct before running the system, you might want to use this framework.

  • If you are trying to create a lint/static analysis tool, you might want to use this framework so you don't have to reinvent the wheel. Various components are already implemented for you. See the architecture for more details.

Requirements

Gradle is required to build the code. If Gradle is not already installed, you can use the wrapper to install it for you.

The required libraries/dependencies can be found in the Gradle build file.

Quick Start

  1. Specify com.selesse:jxlint:1.7.0 as a Maven dependency.
  2. Make customizations:

Create all your rules. It's recommended to put all the rules in one directory, as can be seen in the sample implementations.

public class XmlEncodingRule extends LintRule {
    public XmlEncodingRule () {
        super("XML encoding specified", "Encoding of the XML should be specified.",
                "The XML encoding should be specified. For example, <?xml version=\"1.0\" encoding=\"UTF-8\"?>.",
                Severity.WARNING, Category.LINT, false);
    }

    @Override
    public List<File> getFilesToValidate() {
        return FileUtils.allFilesWithExtension(getSourceDirectory(), "xml");
    }

    @Override
    public List<LintError> getLintErrors(File file) {
        List<LintError> lintErrorList = Lists.newArrayList();
        try {
            DocumentBuilderFactory documentBuilderFactory = DocumentBuilderFactory.newInstance();
            DocumentBuilder documentBuilder = documentBuilderFactory.newDocumentBuilder();
            documentBuilder.setErrorHandler(null); // silence the DOM error handler
            Document document = documentBuilder.parse(file);

            document.getDocumentElement().normalize();

            if (Strings.isNullOrEmpty(document.getXmlEncoding())) {
                lintErrorList.add(
                    LintError.with(this, file)
                        .andMessage("Encoding wasn't specified")
                        .create()
                );
            }
        }
        catch (Exception e) {
            lintErrorList.add(
                LintError.with(this, file)
                    .andMessage("Error checking rule, could not parse XML")
                    .andException(e)
                    .create()
            );
        }

        return lintErrorList;
    }
}

Add (at least) 1 positive and 1 negative test case. A (recommended) version of such a tester can be found here. See the Javadoc for instructions on how to set this up. Extending this class leads to the following positive + negative test case:

public class XmlEncodingTest extends AbstractPassFailFileXmlFileTest {
    public XmlEncodingTest() {
        super(new XmlEncodingRule());
    }
}

Set up the container by adding all your custom-defined rules.

public class MyXmlLintRulesImpl extends AbstractLintRules {
    @Override
    public void initializeLintRules() {
        // Example rule saying that XML must be valid
        lintRules.add(new ValidXmlRule());

        // Example rule saying that duplicate attribute tags within XML are bad
        lintRules.add(new UniqueAttributeRule());

        // Example (disabled-by-default) rules
        lintRules.add(new XmlVersionRule());
        lintRules.add(new XmlEncodingRule());
    }
}

In your application's Main class:

public class Main {
    public static void main(String[] args) {
        Jxlint jxlint = new Jxlint(new MyXmlLintRulesImpl(), new MyProgramSettings());
        jxlint.parseArgumentsAndDispatch(args);
    }
}
  1. Build your application.

Building the Code

To build the code, run gradle. This will create a jxlint jar. If you do not have gradle installed, type gradlew.

Examples

Sample implementations can be found here.

Versioning

This project follows Semantic Versioning as much as possible.

License

This software is licensed under the MIT License.

About

Framework for doing static analysis in Java

Resources

Stars

Watchers

Forks

Packages

No packages published

Contributors 3

  •  
  •  
  •