Skip to content

Latest commit

 

History

History
101 lines (81 loc) · 6.15 KB

README.md

File metadata and controls

101 lines (81 loc) · 6.15 KB

ECSTATIC

ECSTATIC (Extensible, Customizable Static Analysis Tester Informed by Configuration) is a flexible tool that can be used to test configurable static analyses on a variety of benchmarks. ECSTATIC can be extended to use alternative analyses, but currently, it can run call graph analyses on WALA, SOOT, and DOOP, as well as taint analysis on Android applications using FlowDroid.

Prerequisites

This application is written for Python version >= 3.10.0. Furthermore, ECSTATIC runs its analyses in Docker containers in order to maintain consistent environments across runs, so you must have a working Docker installation. We know ECSTATIC can successfully build on Ubuntu, Windows, and Mac (as of 2022-07-22, building on M1 Mac takes a long time, as it has to compile Z3 from scratch).

Usage

We recommend creating a virtual environment for ECSTATIC. To do so, run

python -m venv <name_of_virtual_environment>

where 'python' points to a python 3.8 or higher installation. This will create a new folder. If, for example, you named your folder 'venv', then you can activate it as follows:

source ./venv/bin/activate

In order to install ECSTATIC's dependencies, from the root directory of the repository, run

python -m pip install -r requirements.txt

Where python points to a Python executable of at least version 3.8.0. This will install all of the Python dependencies required. Then, in order to install the application, run

python -m pip install -e .

We require the -e to build in-place. Currently, omitting this option will cause the Dockerfile resolution to fail when we try to build tool-specific images.

This installation will put three executables on your system PATH: dispatcher, tester, and deltadebugger. dispatcher is the command you run from your host, while tester is the command you run from inside the Docker container (under normal usage, a user will never invoke tester themselves, but it can be useful for debugging to skip container creation.)

Simply run dispatcher --help from anywhere in order to see the helpdoc on how to invoke ECSTATIC.

Extending with New Tools

To add a new tool to ECSTATIC, you must take the following steps:

  1. Create a new Dockerfile for your tool under src/resources/tools/<tool_name>. The Dockerfile must create an image that inherits from ECSTATIC's base image, and builds the tool. See some of the existing Dockerfiles we have for examples.
  2. Specify the configuration space and grammar in src/resources/configuration_spaces/<tool_name>_config.json and src/resources/grammars/{tool_name>_grammar.json respectively. Grammars are specified in the same format as Zeller's "The Fuzzing Book" https://www.fuzzingbook.org/html/Grammars.html, and configuration spaces are specified according to the schema src/resources/schema/configuration_space.schema.json and src/resources/schema/option.schema.json. A configuration space is represented by a name and a collection of options. Options have a name, a list of settings ("levels"), and a list of partial orders ("orders"). Each partial order has a "left" element, an "order" element, and a "right" element. Order is one of MST (more sound than or equal to) or MPT (more precise than or equal to). For example, the following option
    {
      "name": "optimize",
      "default": "FALSE",
      "levels": [
        "TRUE",
        "FALSE"
      ],
      "orders": [
        {
          "left": "TRUE",
          "order": "MPT",
          "right": "FALSE"
        }
      ]
    }

from SOOT's configuration space defines an option, "optimize," with two settings: TRUE and FALSE. The default setting is "FALSE", and the partial orders are that TRUE should be at least as precise as FALSE.

  1. Add a new class that inherits from AbstractCommandLineToolRunner.py in order to run the tool. Specifically, you must override the try_run_job method. If your tool is able to be run relatively simply (i.e., only by setting command line options), then you might find it easier to extend CommandLineToolRunner.py. See SOOTRunner.py and WALARunner.py for examples of classes that extend CommandLineToolRunner, and DOOPRunner.py and FlowDroidRunner.py for examples of more complex runners that inherit from AbstractCommandLineToolRunner.py.
  2. Add logic to RunnerFactory.py to initialize your new runner given the name of the tool.
  3. Add a new class that inherits from AbstractRunner.py to read the results of your tool. The import_file method of this class accepts a file name, and returns an iterable of results. In order to detect violations, it is important that equality be defined correctly between the results.
  4. Add logic to ReaderFactory.py that will return the appropriate reader given the task and tool name.
  5. Depending on how you want to detect violations, you may need to add a new violation checker that inherits from AbstractViolationChecker.py. The default behavior is to just check equality between records. However, if you need to determine ground truths, or you need to filter results (as we do in callgraph analyses), then you might need your own class. If you implement a new class, be sure to add logic to ViolationCheckerFactory.py.
  6. That's it!

Extending with New Benchmarks

Adding a new benchmark is relatively simple.

  1. Add a new folder to src/resources/benchmarks, with the name of your benchmark.
  2. In that folder, create a build.sh script that will pull the benchmark code, build it, and put it under /benchmarks/<benchmark_name> in the Docker container. Add an index.json file specifying the programs you want to run. The resolver for this file will automatically resolve paths so long as they are unique in the /benchmarks directory.