Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add code coverage support #52

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open

add code coverage support #52

wants to merge 1 commit into from

Conversation

hchauvin
Copy link

Add an offline instrumentation step with Jacoco to the Kotlin builder,
complete with some changes related to hermeticity (allows end-to-end
testing). The behavior of the native Java rules is copied as much as
possible concerning code coverage.

@hchauvin
Copy link
Author

I must add that currently rules_kotlin itself is not supposed to pass code coverage, i.e. "cd rules_kotlin && bazel coverage //:all_tests" does not work. The reasons are:

  1. The unit tests for the builder do not reference any instrumented source, tripping LcovMerger.

  2. The integration tests do not fully incorporate the runfiles of the Java executables they run. The only way I found to make them run was to re-use the collect_runfiles() python function, used here for integration tests concerning code coverage, but that's beyond the scope of this PR.

  3. And in any case, since the integration tests run Java executables that are themselves instrumented, LcovMerger ends up having to actually merge LCOV files and that is not currently supported.

However, I added some integration tests based on https://github.com/bazelbuild/bazel-integration-testing that check the generated LCOV for a simple case. There are probably many edge cases I have not considered, and things regarding, e.g. baseline coverage, that need to be taken into account, but this is what I have right now.

@hsyed
Copy link
Contributor

hsyed commented Apr 23, 2018

@hchauvin great to see someone else getting into the codebase. I need help in maintaining this repo and designing advanced features.

I assume the end goal of this PR is to test the builder layer directly ? Currently the builder layer is tested indirectly by passing positive test cases through it and validating the output.

A motivating use case for testing the builder directly from the stuff that I have been working on might be #40. Currently #40 is implemented in a poor-mans way, ideally we should have a kotlin compiler plugin just like the java builder.


bazel_integration_testing is realy slow last I checked (@buchgr has anything changed on this front ?). I purposefully avoided it. Unless optimisations have been added to bazel forbazel_integration_testing I'd prefer to keep things fast and simple and with as few moving parts as possible.

A shortcoming with the approach I am currently taking is that we can't pass through negative test cases -- but negative cases aren't important enough currently.

Can you test what you need to test without using bazel_integration_testing ?

@hchauvin
Copy link
Author

Hi,

Thank you for the interest!

So the original purpose was to add code coverage support (via bazel coverage), and I used bazel_integration_testing to check that the LCOV data files are generated as expected. I could not find any open-source example of how third-party rules can plug in the code coverage system, nor any example of how to "meta-test" a Bazel rule, so this PR has been designed, I concede, in a rather experimental way.

bazel_integration_testing has drawbacks, but I am able to have the whole integration tests run in under two minutes. Granted, I achieved that by using some tricks, but nothing out of the ordinary.

Unless I use bazel_integration_testing, the only way to find out whether the LCOV files are generated to spec is to invoke bazel in a bash script, outside of the usual invocation of tests with 'bazel test //...'. This can only be done if the CI system allows it, and currently, unless I am mistaken, your Buildkite CI system does not. If I can use a bash script instead, I can greatly simplify this step, and I can save the bazel_integration_testing for another PR, if that is still something you are willing to consider.

And yes, I haven't looked at the details, but PR #40 can definitely reuse a lot of the logic developed here for its e2e testing. It can be as simple as adding a new method to CoverageTest (and rename the whole test class to something like EndToEndTest?).

@ittaiz
Copy link
Member

ittaiz commented Apr 24, 2018

@hsyed I hope it's not too much of a derail but when you say that bazel-integration-testing is really slow what exactly do you mean? we're using it in a few places and are constantly looking for feedback to improve it (if it's too much of a derail we can continue this in an issue over in bazel-integration-testing). Thanks!

@buchgr
Copy link
Contributor

buchgr commented Apr 24, 2018

@hsyed I am not familiar with bazel-integration-testing. @ittaiz is the expert :-)!

@hchauvin
Copy link
Author

hchauvin commented Apr 24, 2018

Hi, a quick update since I got curious and did some profiling.

So on an Ubuntu box I got the integration test to pass in 31s, downloading time included (way faster than on my Mac, but my setting is typical of CI environments, so 31s is the actual testing time to expect in CI).

Profiling shows that ~10s are spent downloading com_github_jetbrains_kotlin. That's definitely an area where things can be improved, and there are ways to pay the 10s only once, not every time the test is run (bazelbuild/bazel-integration-testing#63).

Then you have some time spent initializing the toolchains. On Ubuntu, for java+cc, that's around 4-5s max. On Mac, that's around 10-15s mainly because of the toolchain for xcode. That is not an area were things are easily improved. Concerning the remaining ~15s, it does not seem to me out of the ordinary considering you have to compile everything with code coverage then run the tests.

@hchauvin
Copy link
Author

hchauvin commented Apr 25, 2018

So I removed bazel-integration-testing and split the commit to bring clarity.

@hsyed
Copy link
Contributor

hsyed commented Apr 25, 2018

@hchauvin cool ! it's a good idea really. at the moment it's only 30 seconds because the workspace contains no dependencies. This repo currently has barely any external deps, the kotlin compiler repo is only 30ish mb. When/if we add tests that include the intellij bazel plugin repo (400 mb to download the intellij platform) and android examples things really begin to add up.

@hsyed
Copy link
Contributor

hsyed commented Apr 27, 2018

@ittaiz the bazel-in-bazel aspect is extremely slow for a local development workflow everything needs to be fetched and unpacked etc.

@ittaiz
Copy link
Member

ittaiz commented Apr 27, 2018 via email

@hchauvin hchauvin changed the title add code coverage support [wip] add code coverage support Apr 30, 2018
@hchauvin hchauvin changed the title [wip] add code coverage support add code coverage support Apr 30, 2018
@hchauvin
Copy link
Author

One development concerning Jacoco is worth mentioning here: 0.8.0 released on January 3rd introduced filtering options. One contribution to follow in particular relates to filtering out methods generated by Kotlin, such as getters/setters: jacoco/jacoco#681. This makes coverage reports for Kotlin even more relevant.

@hsyed
Copy link
Contributor

hsyed commented May 11, 2018

@hchauvin master just had a huge refactor. Most of it is directly relevant to the work you have done.

You can now test the builder directly -- see //kotlin/builder/integrationtests. You should be able to do the bulk of the tests there and then do some e2e functional verification tests using the same pattern I am using in //tests/jvm.

@hsyed
Copy link
Contributor

hsyed commented May 11, 2018

@hchauvin Also the deps are now managed via bazel-deps.

@hchauvin
Copy link
Author

hchauvin commented May 12, 2018

Thanks for for heads-up, your changes are very cool! I'm rewriting my patch.

@hchauvin
Copy link
Author

@hsyed So I completely reworked the PR and got rid of the bash integration test by getting a bit into the logic of the JacocoCoverageRunner.

@hsyed
Copy link
Contributor

hsyed commented Jul 10, 2018

@hchauvin sorry for the delay. i've rebased the changes in the branch rebase_coverage to save you some time.

When I was merging the code i saw instrumentation matching .kt files in skylark, it should also match java files I think because mixed mode compile is possible ? .

I think we need to pick up these deps from upstream Ideally.

Have a look at this. I think instrumentation will end up being another an action that is runnable after preprocessing is done.

@Kernald
Copy link
Contributor

Kernald commented Nov 13, 2018

Any news on that topic? It looks like most of the work is done?

sgammon pushed a commit to sgammon/rules_kotlin that referenced this pull request Dec 16, 2020
Builds on previous work (bazelbuild#52) to add
coverage support to Kotlin targets in Bazel.

So far, it's working (in basic form), but seems to omit major
swaths of our code when I test it on a live codebase. I'm not
sure why that happens yet.
sgammon pushed a commit to sgammon/rules_kotlin that referenced this pull request Dec 16, 2020
Builds on previous work (bazelbuild#52) to add
coverage support to Kotlin targets in Bazel.

So far, it's working (in basic form), but seems to omit major
swaths of our code when I test it on a live codebase. I'm not
sure why that happens yet.
@sgammon sgammon mentioned this pull request Dec 16, 2020
sgammon pushed a commit to sgammon/rules_kotlin that referenced this pull request Jan 13, 2021
Builds on previous work (bazelbuild#52) to add
coverage support to Kotlin targets in Bazel.

So far, it's working (in basic form), but seems to omit major
swaths of our code when I test it on a live codebase. I'm not
sure why that happens yet.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants