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

Test Coverage Reporting #14343

Draft
wants to merge 2 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 15 additions & 1 deletion .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -21,13 +21,16 @@ permissions:

jobs:
test_linux:
name: Ubuntu 24.04, Erlang/OTP ${{ matrix.otp_version }}${{ matrix.deterministic && ' (deterministic)' || '' }}
name: Ubuntu 24.04, Erlang/OTP ${{ matrix.otp_version }}${{ matrix.deterministic && ' (deterministic)' || '' }}${{ matrix.coverage && ' (coverage)' || '' }}
strategy:
fail-fast: false
matrix:
include:
- otp_version: "27.1"
deterministic: true
- otp_version: "27.1"
erlc_opts: "warnings_as_errors"
coverage: true
- otp_version: "27.1"
otp_latest: true
erlc_opts: "warnings_as_errors"
Expand Down Expand Up @@ -65,9 +68,14 @@ jobs:
- name: Erlang test suite
run: make test_erlang
continue-on-error: ${{ matrix.development }}
if: "${{ !matrix.coverage }}"
- name: Elixir test suite
run: make test_elixir
continue-on-error: ${{ matrix.development }}
if: "${{ !matrix.coverage }}"
- name: "Calculate Coverage"
run: make cover
if: "${{ matrix.coverage }}"
- name: Build docs (ExDoc main)
if: ${{ matrix.otp_latest }}
run: |
Expand All @@ -85,6 +93,12 @@ jobs:
# Recompile System without .git
cd lib/elixir && ../../bin/elixirc -o ebin lib/system.ex && cd -
taskset 1 make check_reproducible
- name: "Upload Coverage Artifact"
if: "${{ matrix.coverage }}"
uses: actions/upload-artifact@4cec3d8aa04e39d1a68397de0c4cd6fb9dce8ec1 # v4.6.1
with:
name: TestCoverage
path: cover/*

test_windows:
name: Windows Server 2019, Erlang/OTP ${{ matrix.otp_version }}
Expand Down
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -15,3 +15,4 @@
/.eunit
.elixir.plt
erl_crash.dump
/cover/
22 changes: 22 additions & 0 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ GIT_REVISION = $(strip $(shell git rev-parse HEAD 2> /dev/null ))
GIT_TAG = $(strip $(shell head="$(call GIT_REVISION)"; git tag --points-at $$head 2> /dev/null | grep -v latest | tail -1))
SOURCE_DATE_EPOCH_PATH = lib/elixir/tmp/ebin_reproducible
SOURCE_DATE_EPOCH_FILE = $(SOURCE_DATE_EPOCH_PATH)/SOURCE_DATE_EPOCH
ROOT_DIR:=$(shell dirname $(realpath $(firstword $(MAKEFILE_LIST))))

.PHONY: install install_man build_plt clean_plt dialyze test check_reproducible clean clean_elixir clean_man format docs Docs.zip Precompiled.zip zips
.NOTPARALLEL:
Expand Down Expand Up @@ -53,6 +54,11 @@ lib/$(1)/ebin/Elixir.$(2).beam: $(wildcard lib/$(1)/lib/*.ex) $(wildcard lib/$(1
test_$(1): test_formatted $(1)
@ echo "==> $(1) (ex_unit)"
$(Q) cd lib/$(1) && ../../bin/elixir -r "test/test_helper.exs" -pr "test/**/$(TEST_FILES)";

cover/ex_unit_$(1).coverdata:
$(Q) mkdir -p "$(ROOT_DIR)/cover"
$(Q) COVER_FILE="$(ROOT_DIR)/cover/ex_unit_$(1).coverdata" $(MAKE) test_$(1)
cover/combined.coverdata: cover/ex_unit_$(1).coverdata
endef

define WRITE_SOURCE_DATE_EPOCH
Expand Down Expand Up @@ -175,6 +181,7 @@ clean: clean_man
rm -rf lib/mix/test/fixtures/git_sparse_repo/
rm -rf lib/mix/test/fixtures/archive/ebin/
rm -f erl_crash.dump
rm -rf cover

clean_elixir:
$(Q) rm -f lib/*/ebin/Elixir.*.beam
Expand Down Expand Up @@ -287,6 +294,21 @@ test_stdlib: compile
cd lib/elixir && ../../bin/elixir --sname primary -r "test/elixir/test_helper.exs" -pr "test/elixir/**/$(TEST_FILES)"; \
fi

cover/ex_unit_stdlib.coverdata:
$(Q) mkdir -p "$(ROOT_DIR)/cover"
$(Q) COVER_FILE="$(ROOT_DIR)/cover/ex_unit_stdlib.coverdata" $(MAKE) test_stdlib
cover/combined.coverdata: cover/ex_unit_stdlib.coverdata

cover/combined.coverdata:
$(Q) if [ -z "$(GITHUB_STEP_SUMMARY)" ]; then\
bin/elixir ./lib/elixir/scripts/cover.exs;\
else\
bin/elixir ./lib/elixir/scripts/cover.exs | tee "$(GITHUB_STEP_SUMMARY)";\
fi

.PHONY: cover
cover: cover/combined.coverdata

#==> Dialyzer tasks

DIALYZER_OPTS = --no_check_plt --fullpath -Werror_handling -Wunmatched_returns -Wunderspecs
Expand Down
2 changes: 2 additions & 0 deletions lib/eex/test/test_helper.exs
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@
{line_exclude, line_include} =
if line = System.get_env("LINE"), do: {[:test], [line: line]}, else: {[], []}

Code.eval_file("../../elixir/scripts/cover_record.exs", __DIR__)

ExUnit.start(
trace: !!System.get_env("TRACE"),
include: line_include,
Expand Down
42 changes: 42 additions & 0 deletions lib/elixir/scripts/cover.exs
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
#!bin/elixir

# SPDX-License-Identifier: Apache-2.0
# SPDX-FileCopyrightText: 2021 The Elixir Team

root_dir = __ENV__.file |> Path.dirname() |> Path.join("../../..")
cover_dir = Path.join(root_dir, "cover")
coverdata_inputs = cover_dir |> Path.join("ex_unit_*.coverdata") |> Path.wildcard()
coverdata_output = Path.join(cover_dir, "combined.coverdata")
ebins = root_dir |> Path.join("lib/*/ebin") |> Path.wildcard()

_ = :cover.stop()
{:ok, cover_pid} = :cover.start()

for ebin <- ebins,
result <- :cover.compile_beam_directory(String.to_charlist(ebin)) do
case result do
{:ok, _module} ->
:ok

{:error, reason} ->
raise "Failed to cover compile directory #{ebin} with reason: #{inspect(reason)}"
end
end

for file <- coverdata_inputs do
:ok = :cover.import(String.to_charlist(file))
end

:ok = :cover.export(String.to_charlist(coverdata_output))

{:ok, _} = Application.ensure_all_started(:mix)

# Silence analyse import messages emitted by cover
{:ok, string_io} = StringIO.open("")
Process.group_leader(cover_pid, string_io)

:ok =
Mix.Tasks.Test.Coverage.generate_cover_results(
output: cover_dir,
summary: [threshold: 0]
)
31 changes: 31 additions & 0 deletions lib/elixir/scripts/cover_record.exs
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
# SPDX-License-Identifier: Apache-2.0
# SPDX-FileCopyrightText: 2021 The Elixir Team

root_dir = __ENV__.file |> Path.dirname() |> Path.join("../../..")
ebins = root_dir |> Path.join("lib/*/ebin") |> Path.wildcard()

case System.fetch_env("COVER_FILE") do
{:ok, file} ->
_ = :cover.stop()
{:ok, _pid} = :cover.start()

for ebin <- ebins,
result <- :cover.compile_beam_directory(String.to_charlist(ebin)) do
case result do
{:ok, _module} ->
:ok

{:error, reason} ->
raise "Failed to cover compile directory #{ebin} with reason: #{inspect(reason)}"
end
end

System.at_exit(fn _status ->
:ok = :cover.export(String.to_charlist(file))
end)

true

:error ->
false
end
78 changes: 78 additions & 0 deletions lib/elixir/scripts/path_helpers.exs
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
# SPDX-License-Identifier: Apache-2.0
# SPDX-FileCopyrightText: 2021 The Elixir Team
# SPDX-FileCopyrightText: 2012 Plataformatec

# Beam files compiled on demand
path = Path.expand("../tmp/beams", __DIR__)
File.rm_rf!(path)
File.mkdir_p!(path)
Code.prepend_path(path)

defmodule PathHelpers do
def fixture_path() do
Path.expand("../test/elixir/fixtures", __DIR__)
end

def tmp_path() do
Path.expand("../tmp", __DIR__)
end

def fixture_path(extra) do
Path.join(fixture_path(), extra)
end

def tmp_path(extra) do
Path.join(tmp_path(), extra)
end

def elixir(args, executable_extension \\ "") do
run_cmd(elixir_executable(executable_extension), args)
end

def elixir_executable(extension \\ "") do
executable_path("elixir", extension)
end

def elixirc(args, executable_extension \\ "") do
run_cmd(elixirc_executable(executable_extension), args)
end

def elixirc_executable(extension \\ "") do
executable_path("elixirc", extension)
end

def iex(args, executable_extension \\ "") do
run_cmd(iex_executable(executable_extension), args)
end

def iex_executable(extension \\ "") do
executable_path("iex", extension)
end

def write_beam({:module, name, bin, _} = res) do
File.mkdir_p!(unquote(path))
beam_path = Path.join(unquote(path), Atom.to_string(name) <> ".beam")
File.write!(beam_path, bin)
res
end

defp run_cmd(executable, args) do
~c"#{executable} #{IO.chardata_to_string(args)}#{redirect_std_err_on_win()}"
|> :os.cmd()
|> :unicode.characters_to_binary()
end

defp executable_path(name, extension) do
Path.expand("../../../bin/#{name}#{extension}", __DIR__)
end

if match?({:win32, _}, :os.type()) do
def windows?, do: true
def executable_extension, do: ".bat"
def redirect_std_err_on_win, do: " 2>&1"
else
def windows?, do: false
def executable_extension, do: ""
def redirect_std_err_on_win, do: ""
end
end
49 changes: 39 additions & 10 deletions lib/elixir/test/elixir/exception_test.exs
Original file line number Diff line number Diff line change
Expand Up @@ -556,20 +556,36 @@ defmodule ExceptionTest do
end

test "annotates function clause errors" do
assert blame_message(Access, & &1.fetch(:foo, :bar)) =~ """
no function clause matching in Access.fetch/2
import PathHelpers

write_beam(
defmodule ExampleModule do
def fun(arg1, arg2)
def fun(:one, :one), do: :ok
def fun(:two, :two), do: :ok
end
)

:code.purge(ExampleModule)
:code.delete(ExampleModule)

message = blame_message(ExceptionTest.ExampleModule, & &1.fun(:three, :four))

assert message =~ """
no function clause matching in ExceptionTest.ExampleModule.fun/2

The following arguments were given to Access.fetch/2:
The following arguments were given to ExceptionTest.ExampleModule.fun/2:

# 1
:foo
:three

# 2
:bar
:four

Attempted function clauses (showing 5 out of 5):
Attempted function clauses (showing 2 out of 2):

def fetch(-%module{} = container-, key)
def fun(-:one-, -:one-)
def fun(-:two-, -:two-)
"""
end

Expand Down Expand Up @@ -858,15 +874,28 @@ defmodule ExceptionTest do

describe "blaming unit tests" do
test "annotates clauses errors" do
args = [%{}, :key, nil]
import PathHelpers

write_beam(
defmodule ExampleModule do
def fun(arg), do: arg
end
)

:code.purge(ExampleModule)
:code.delete(ExampleModule)

args = [nil]

{exception, stack} =
Exception.blame(:error, :function_clause, [{Keyword, :pop, args, [line: 13]}])
Exception.blame(:error, :function_clause, [{ExampleModule, :fun, args, [line: 13]}])

assert %FunctionClauseError{kind: :def, args: ^args, clauses: [_]} = exception
assert stack == [{Keyword, :pop, 3, [line: 13]}]

assert stack == [{ExampleModule, :fun, 1, [line: 13]}]
end

@tag :require_ast
test "annotates args and clauses from mfa" do
import PathHelpers

Expand Down
1 change: 1 addition & 0 deletions lib/elixir/test/elixir/kernel/dialyzer_test.exs
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ defmodule Kernel.DialyzerTest do
use ExUnit.Case, async: true

@moduletag :dialyzer
@moduletag :require_ast
import PathHelpers

setup_all do
Expand Down
2 changes: 2 additions & 0 deletions lib/elixir/test/elixir/module/types/integration_test.exs
Original file line number Diff line number Diff line change
Expand Up @@ -458,6 +458,7 @@ defmodule Module.Types.IntegrationTest do
assert_warnings(files, warnings)
end

@tag :require_ast
test "String.Chars protocol dispatch" do
files = %{
"a.ex" => """
Expand Down Expand Up @@ -520,6 +521,7 @@ defmodule Module.Types.IntegrationTest do
assert_warnings(files, warnings, consolidate_protocols: true)
end

@tag :require_ast
test "Enumerable protocol dispatch" do
files = %{
"a.ex" => """
Expand Down
16 changes: 14 additions & 2 deletions lib/elixir/test/elixir/module_test.exs
Original file line number Diff line number Diff line change
Expand Up @@ -432,9 +432,21 @@ defmodule ModuleTest do
end

test "compiles to core" do
{:ok, {Atom, [{~c"Dbgi", dbgi}]}} = Atom |> :code.which() |> :beam_lib.chunks([~c"Dbgi"])
import PathHelpers

write_beam(
defmodule ExampleModule do
end
)

:code.purge(ExampleModule)
:code.delete(ExampleModule)

{:ok, {ExampleModule, [{~c"Dbgi", dbgi}]}} =
ExampleModule |> :code.which() |> :beam_lib.chunks([~c"Dbgi"])

{:debug_info_v1, backend, data} = :erlang.binary_to_term(dbgi)
{:ok, core} = backend.debug_info(:core_v1, Atom, data, [])
{:ok, core} = backend.debug_info(:core_v1, ExampleModule, data, [])
assert is_tuple(core)
end

Expand Down
Loading
Loading