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
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.
-
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.
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.
- Specify
com.selesse:jxlint:1.7.0
as a Maven dependency. - 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);
}
}
- Build your application.
To build the code, run gradle
. This will create a jxlint jar. If you do
not have gradle installed, type gradlew
.
Sample implementations can be found here.
This project follows Semantic Versioning as much as possible.
This software is licensed under the MIT License.