Make sure that you have installed the minimum required version of flow-cli
:
flow version
Version: v1.3.1
Commit: 9f622977c3dff5381dbaf49fa7984805e34649d3
To install it, simply run:
sh -ci "$(curl -fsSL https://raw.githubusercontent.com/onflow/flow-cli/master/install.sh)" -- v1.3.1
To view code coverage results when running our tests, we can use:
flow test --cover tests/test_foo_contract.cdc
The output will look something like this:
Test results: "tests/test_foo_contract.cdc"
- PASS: testGetIntegerTrait
- PASS: testAddSpecialNumber
Coverage: 95.8% of statements
It looks like not all statements were covered by the test inputs. To view details for the coverage report,
we can consult the auto-generated coverage.json
file:
{
"coverage": {
"S.FooContract": {
"line_hits": {
"14": 1,
"18": 9,
"19": 1,
"20": 8,
"21": 1,
"22": 7,
"23": 1,
"24": 6,
"25": 1,
"26": 5,
"27": 0,
"30": 5,
"31": 4,
"34": 1,
"6": 1
},
"missed_lines": [
27
],
"statements": 15,
"percentage": "93.3%"
},
}
...
}
Note: We can use the --coverprofile
flag if we wish to generate the coverage report to a different file.
flow test --cover --coverprofile=codecov.json tests/test_foo_contract.cdc
We can also generate a coverage report for the LCOV format, to be used with CI/CD plugins such as Codecov and Coveralls.
flow test --cover --coverprofile=codecov.lcov tests/test_foo_contract.cdc
All we need to do is give the file the .lcov
extension.
The file will look something like this:
TN:
SF:S.FooContract
DA:6,1
DA:14,1
DA:18,9
DA:19,1
DA:20,8
DA:21,1
DA:22,7
DA:23,1
DA:24,6
DA:25,1
DA:26,5
DA:27,0
DA:30,5
DA:31,4
DA:34,1
LF:15
LH:14
end_of_record
Reading the JSON/LCOV file, we can see that for FooContract
the line 27
was missed during the tests (not covered by any of the test inputs).
To fix that, we can tweak the testInputs
Dictionary on tests/test_foo_contract.cdc
to observe how the coverage percentage changes. By uncommenting the line 14
, we now get:
flow test --cover tests/test_foo_contract.cdc
Test results: "tests/test_foo_contract.cdc"
- PASS: testGetIntegerTrait
- PASS: testAddSpecialNumber
Coverage: 97.2% of statements
For some more realistic contracts and tests:
flow test --cover tests/test_array_utils.cdc
Test results: "tests/test_array_utils.cdc"
- PASS: testRange
- PASS: testTransform
- PASS: testIterate
- PASS: testMap
- PASS: testMapStrings
- PASS: testReduce
Coverage: 94.3% of statements
Look at the files contracts/ArrayUtils.cdc
(smart contract) and tests/test_array_utils.cdc
(tests for the smart contract).
For the ArrayUtils.range
method, we have omitted the code branch where start > end
on purpose. It is left as an exercise for the reader. Look at the comment on line 26 in tests/test_array_utils.cdc
.
Note that the above examples of tests could be best described as unit tests.
An example of integration tests can be found in the tests/test_string_utils.cdc
file, which tests the functionality of the contracts/StringUtils.cdc
smart contract.
flow test --cover tests/test_string_utils.cdc
Test results: "tests/test_string_utils.cdc"
- PASS: testFormat
- PASS: testExplode
- PASS: testTrimLeft
- PASS: testTrim
- PASS: testReplaceAll
- PASS: testHasPrefix
- PASS: testHasSuffix
- PASS: testIndex
- PASS: testCount
- PASS: testContains
- PASS: testSubstringUntil
- PASS: testSplit
- PASS: testJoin
Coverage: 88.0% of statements
The generated coverage.json
file is somewhat more elaborate, for integration tests. By viewing its content, we find the following keys:
A.01cf0e2f2f715450.ArrayUtils
A.01cf0e2f2f715450.StringUtils
and some other locations.
Locations that start with A.
are contracts deployed to an account, ones that start with s.
are scripts, and ones that start with t.
are transactions.
The ArrayUtils
smart contract is imported by StringUtils
, that's why it was deployed on the integration tests, and that's why it is included in the resulting coverage report.
Note: These two contracts are taken from: https://github.com/green-goo-dao/flow-utils. They are copied here for demonstration purposes. To get the original source code, visit the above repository.
For viewing the coverage report of the StringUtils
smart contract, we can just consult the value of the A.01cf0e2f2f715450.StringUtils
key, in the coverage.json
file.
There is also a more advance example of integration tests for the ApprovalVoting
smart contract, which deals with resources, multi-sig transactions etc.
flow test --cover tests/test_approval_voting.cdc
11:31AM INF LOG: "Proposals Initialized!"
11:31AM INF LOG: "Ballot transferred to voter"
11:31AM INF LOG: "Vote cast and tallied"
Test results: "tests/test_approval_voting.cdc"
- PASS: testInitializeEmptyProposals
- PASS: testInitializeProposals
- PASS: testProposalsImmutability
- PASS: testIssueBallot
- PASS: testCastVoteOnMissingProposal
- PASS: testCastVote
- PASS: testViewVotes
Coverage: 94.1% of statements
It is also possible to view code coverage through the emulator, outside the context of testing.
All we have to do is start the emulator with the necessary flag (coverage-reporting
):
flow emulator --storage-limit=false --coverage-reporting
With this, we can use our browser and visit http://localhost:8080/emulator/codeCoverage.
This code coverage report will reflect every interaction with the emulator. For example, we can deploy contracts to the emulator, run scripts/transactions against them and view the results:
flow deploy contracts --network=emulator
flow scripts execute scripts/foo_contract_scripts.cdc --network=emulator
We can also flush/reset the collected code coverage report, with:
curl -XPUT 'http://localhost:8080/emulator/codeCoverage/reset'
Which results in the following:
All of the keys have disappeared, except for A.f8d6e0586b0a20c7.FlowServiceAccount
, which is a system contract that is essential to the operations of Flow.