Purpose of Static Analysis Verification and Evaluation (SAVE) project
Usage of static analyzers - is a very important part of development each and every software product. All human beings can make a mistake in their code even when a software developer is writing all kinds of tests and has a very good test-coverage. All these issues can lead to potential money losses of companies. Static analysis of programs helps to reduce the number of such bugs and issues that cannot be found by validations on the compiler's side.
There are different kinds and purposes of static analysis: it can be simple analysis using AST (abstract syntax tree), it can be more complex CFA (control-flow analysis), interprocedural analysis, context sensitive analysis, e.t.c. Static analyzers can check code style, find potential issues on the runtime in the logic of an application, check for code smells and suggest best practices. But what exactly should static analyzers do? How their functionality can be measured? What is an acceptance criteria for Which functionality do developers really need when they are writing a brand new analyzer? These questions are still remain not answered, even after decades of development of static analyzers.
Problematics
Each and every creator of static analyzers in the beginning of his development journey starts from the very simple thing: types of issues that his tool will detect. This leads to a searching of existing lists of potential issues or test packages that can be used to measure the result of his work or can be used for TDD (test driven development). In other areas of system programming such benchmarks and test sets already exists, for example SPEC.org benchmarks are used all over the world to test the functionality, evaluate and measure the performance of different applications and hardware: from compilers to CPUs, from web-servers to Java Clients. But there are no test sets and even strict standards for detection of issues that can be found in popular programming languages. There were some guidelines of coding on C/C++ done by MISRA, but there are no analogues of it even for the most popular languages in the world like Python and Java.
In this situation each and every new developer that reinvents his new code style or mechanism of static analysis each time reinvents his brand new test framework and writting test sets that have been written already thousands of times for his analyzer/linter. Someone uses existing guidelines like Google code style or using PMD rules. But in all cases lot of time will be spent on reinventing, writing and debuging tests.
What is SAVE?
SAVE - is an eco-system (see also save-cloud) for measuring, testing and certification of static analyzers. Instead of writing your test framework SAVE will provide you a command line (and cloud) application with a test sets for the language that you are developing analyzer for. It provides you also a service that can be used to determine the readiness of your tool. SAVE has a committee of static analysis experts that regularly updates tests and discuss the best practices for particular programming languages.
SAVE supports 5 different categories of static analysis and can be used for testing tools that are used both for checking and for autofixing the code. All of them have the description and special test packages:
- code style issues
- functional bugs
- security issues
- code smells
- best practices
Save can be used not only with static analyzers, but can be used as a test framework for writing functional tests for compilers (as testing principles remain the same).
How to start
- Prepare and configure your test base in the proper format. See test_detection and plugins
- Run the following: TBD (FixMe)
How to configure
SAVE has a command line interface that runs the framework and your executable. What you need is simply to configure the output of your static analyzer so SAVE will be able to check if the proper error was raised on the proper line of test code.
// FixMe: specify options here To check that the warning is correct for SAVE - your static analyzer must print the result to stderr/stdout or to some log file.
// FixMe: specify options here For save-cloud it is extremely important to detect properties file for SAVE, so in case you would like to use both cli application and cloud application we suggest putting the file with name save.properties in the root folder of your repository.
Configuration file can be also provided to SAVE via command line: save -prop $PATH_TO_PROPS/save.properties
Example of save.properties file (for full description and possible values, refer to the corresponding section):
resultOutput=stdout
parallelMode=true
language=c++
testRootPath=~/my_dir/dir_to_tests_root
reportDir=~/report 
OR you can pass these arguments directly in command line:
save --mode=single --language=java 
For the complete list of supported options refer to the options table.
SAVE framework will detect tests, run your analyzer on these tests, will calculate the pass-rate and test results.
Test detection and save.toml file
To make SAVE detect your test suites you need to put save.toml file in each directory where you have tests that should be run. Note, that these configuration files inherit configurations from the previous level of directories.
For example, in case of the following hierarchy of directories:
| A
  | save.toml
  | B
    | save.toml
save.toml from the directory B will inherit settings and properties from directory A.
Please note, that SAVE will detect all files with Test postfix and will automatically use configuration from save.toml file that is placed in the directory. Tests are named by the test file resource name without a suffix 'Test'. In case SAVE will detect a file with Test postfix in test resources and will not be able to find any save.toml configurations in the hierarchy of directories - it will raise an error.
For example, the following example is invalid and will cause an error, because SAVE framework will not be able to find save.toml configuration file:
| A
  | B
  | myTest.java
As described above, save.toml is needed to configure tests. The idea is to have only one configuration file for a directory with tests (one to many relation). Such directories we will call test suites. We decided to have only one configuration file as we have many times seen that for such tests there is a duplication of configuration in the same test suite.
save.toml configuration file
Save configuration uses toml format. As it was told above, save.toml can be imported from the directory hierarchy. The configuration file has [general] table and [plugins] table. To see more information about plugins, read this section. In this section we will give information only about the [general] table that can be used in all plugins.
[general]
# your custom tags that will be used to detect groups of tests (optional)
tags = parsing, null-pointer, e.t.c
# custom free text that describes the test suite (required)
description = "My suite description"
# Simple suite name (required)
suiteName = DocsCheck, CaseCheck, NpeTests, e.t.c 
# excluded tests in the suite (optional). Here you can provide names of excluded tests, separated by comma. By the default no tests are excluded. 
# names can have 'Test' postifix or can be without a postfix - this does not matter. Save will detect tests for both cases
excludedTests = NameWithTestPostfix, NameWithoutPostfix, e.t.c
# tests included in the suite (optional). Here you can provide names of tests, that are included in this suite.
 By the default all tests in the same directory with 'save.toml' file are included in the suite. 
includedTests = NameWithTestPostfix, NameWithoutPostfix, e.t.c
Using plugins for specific test-scenarios
SAVE doesn't have any inspections active by default, instead the behavior of the analysis is fully configurable using plugins. Plugins are dynamic libraries (.so or .dll) and they should be provided using argument --plugins-path. Some plugins are bundled with SAVE out-of-the-box and don't require an additional setup. Here is a list of standard plugins:
- diff-plugin for testing tools for static analyzers that mutate text
- warn-plugin for testing tools that find problems in the source code and emit warnings
Extending SAVE and writing your own plugin is simple. For instructions, see corresponding README.
In case you would like to have several plugins to work in your directory with same test files, just simply add them all to save.toml config:
[general]
...
[fix]
...
[warn]
...
[other plugin]
...
SAVE output
// FixMe: add output formats and a link to a dashboard from save-cloud
Contribution
You can always contribute to the main SAVE framework or to a dashboard - just create a PR for it. But to contribute or change tests in categories you will need get approvals from the maintainer of the test package/analysis category. Please see the list of them.
 JarCasting
 JarCasting
