Skip to content
This repository has been archived by the owner on Aug 8, 2024. It is now read-only.

starknet-edu/cairo-coverage

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

27 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Cairo coverage

This package allows you to have a small coverage report for your cairo files. For now it's a bit hacky but with cairo 0.10.1 it'll be easier.

How to make it work

To install it run:

pip install .

That's the hacky part and i'll explain later why it's needed. You'll need to go into your python packages folder to modify the cairo_runner.py file. To find it easily run which starknet it should display something like BASE_PATH/bin/starknet. The part that we need is BASE_PATH. You'll need to use your preferred editor and follow the steps. I chose to use nano but feel free to use vim or vs code.

  1. nano BASE_PATH/python3.X/site-packages/starkware/cairo/lang/vm/cairo_runner.py
  2. go to the function initialize_vm on line 254 It should look like this:
def initialize_vm(
        self, hint_locals, static_locals: Optional[Dict[str, Any]] = None, vm_class=VirtualMachine
    ):
        context = RunContext(
            pc=self.initial_pc,
            ap=self.initial_ap,
            fp=self.initial_fp,
            memory=self.memory,
            prime=self.program.prime,
        )

We're now going to modify the default argument for the vm_class so it looks like this:

def initialize_vm(
        self, hint_locals, static_locals: Optional[Dict[str, Any]] = None, vm_class=None
    ):
        if vm_class is None:
            vm_class = VirtualMachine
        context = RunContext(
            pc=self.initial_pc,
            ap=self.initial_ap,
            fp=self.initial_fp,
            memory=self.memory,
            prime=self.program.prime,
        )

To understand why all this is necessary we'll take a look on how the coverage works and how python works.

To run the examples:

poetry run python3 -m pytest examples/ -s -W ignore::DeprecationWarning

How cairo coverage works

The first step to create cairo coverage was to find a way on how to know which instruction has been ran and to save them. The way cairo works is that every time you run some cairo code it creates a VM to execute the code (which is pretty obvious I know) but it implies that every transaction will need a new VM (also obvious). But this is a problem for us because we want to know all the pc (program counter) that have been touched by our tests and we can't just ask the VM at the end of the tests because it's wiped at each new transaction. So we would need to find a way to save what pc has been touched for what file and to map back the pc to a cairo line. In order to do that we'll override the default VM and create our own that has all the functionalities we want. Now to override the default VM we can monkey patch it basically cairo_runner.VirtualMachine = CustomVm. This would replace the cairo_runner VirtualMachine by our own but since it's a default value here

def initialize_vm(
        self, hint_locals, static_locals: Optional[Dict[str, Any]] = None, vm_class=VirtualMachine
    ):

VirtualMachine can't be modified once the script has started because this value is set at python's compile time (yes python is compiled). So in order to prevent this we modify the default value to None so the VirtualMachine class can be patched at runtime.

So now we know how to override the VM now let's understand what the VM is actually doing. The first important thing is to save all the pc touched by the tests across all the files we would need a variable that's shared between all the class instances (spoiler we need to find a hacky way to do it easily). Fortunately for us we can do that

@staticmethod
def covered(val: defaultdict(list) = defaultdict(list)) -> defaultdict(list):
    return val

What's happening there is that if you don't supply a value for val it'll use a defaultdict(list) (it's just a dict that returns a list for all the keys even not initialized). The thing is that this dict will be initialized once and then it'll reuse the same instance (because it's set at compile time) so we can share this value between all the class instances. We can then get all the pc touched during the cairo run and at the end of this run save them in the default dict shared between all the instances. Even though pc are interesting we want the cairo lines. In order to do that we'll use the debug_info of each file to map the pc to the cairo line. Once we have all this all we need to do is format it and print it in the terminal (no shame on the output I had to format everything myself)

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •