Skip to content

Commit 84fe429

Browse files
authored
Merge pull request #8886 from obsidiansystems/flatten-tests
Move unit tests to separate directories, and document
2 parents 77adb55 + 91b6833 commit 84fe429

File tree

134 files changed

+464
-352
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

134 files changed

+464
-352
lines changed

.gitignore

+3-3
Original file line numberDiff line numberDiff line change
@@ -45,14 +45,14 @@ perl/Makefile.config
4545
/src/libexpr/parser-tab.hh
4646
/src/libexpr/parser-tab.output
4747
/src/libexpr/nix.tbl
48-
/src/libexpr/tests/libnixexpr-tests
48+
/tests/unit/libexpr/libnixexpr-tests
4949

5050
# /src/libstore/
5151
*.gen.*
52-
/src/libstore/tests/libnixstore-tests
52+
/tests/unit/libstore/libnixstore-tests
5353

5454
# /src/libutil/
55-
/src/libutil/tests/libnixutil-tests
55+
/tests/unit/libutil/libnixutil-tests
5656

5757
/src/nix/nix
5858

Makefile

+6-4
Original file line numberDiff line numberDiff line change
@@ -25,11 +25,13 @@ makefiles = \
2525
endif
2626

2727
ifeq ($(ENABLE_BUILD)_$(ENABLE_TESTS), yes_yes)
28-
UNIT_TEST_ENV = _NIX_TEST_UNIT_DATA=unit-test-data
2928
makefiles += \
30-
src/libutil/tests/local.mk \
31-
src/libstore/tests/local.mk \
32-
src/libexpr/tests/local.mk
29+
tests/unit/libutil/local.mk \
30+
tests/unit/libutil-support/local.mk \
31+
tests/unit/libstore/local.mk \
32+
tests/unit/libstore-support/local.mk \
33+
tests/unit/libexpr/local.mk \
34+
tests/unit/libexpr-support/local.mk
3335
endif
3436

3537
ifeq ($(ENABLE_TESTS), yes)

doc/internal-api/doxygen.cfg.in

+8-4
Original file line numberDiff line numberDiff line change
@@ -39,17 +39,21 @@ INPUT = \
3939
src/libcmd \
4040
src/libexpr \
4141
src/libexpr/flake \
42-
src/libexpr/tests \
43-
src/libexpr/tests/value \
42+
tests/unit/libexpr \
43+
tests/unit/libexpr/value \
44+
tests/unit/libexpr/test \
45+
tests/unit/libexpr/test/value \
4446
src/libexpr/value \
4547
src/libfetchers \
4648
src/libmain \
4749
src/libstore \
4850
src/libstore/build \
4951
src/libstore/builtins \
50-
src/libstore/tests \
52+
tests/unit/libstore \
53+
tests/unit/libstore/test \
5154
src/libutil \
52-
src/libutil/tests \
55+
tests/unit/libutil \
56+
tests/unit/libutil/test \
5357
src/nix \
5458
src/nix-env \
5559
src/nix-store

doc/manual/src/contributing/testing.md

+48-19
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@ The unit tests are defined using the [googletest] and [rapidcheck] frameworks.
2020

2121
[googletest]: https://google.github.io/googletest/
2222
[rapidcheck]: https://github.com/emil-e/rapidcheck
23+
[property testing]: https://en.wikipedia.org/wiki/Property_testing
2324

2425
### Source and header layout
2526

@@ -28,34 +29,50 @@ The unit tests are defined using the [googletest] and [rapidcheck] frameworks.
2829
> ```
2930
> src
3031
> ├── libexpr
32+
> │ ├── local.mk
3133
> │ ├── value/context.hh
3234
> │ ├── value/context.cc
33-
> │ │
3435
> │ …
35-
> └── tests
36-
> │ ├── value/context.hh
37-
> │ ├── value/context.cc
38-
> │ │
39-
> │ …
4036
> │
41-
> ├── unit-test-data
42-
> │ ├── libstore
43-
> │ │ ├── worker-protocol/content-address.bin
44-
> │ │ …
37+
> ├── tests
38+
> │ │
4539
> │ …
40+
> │ └── unit
41+
> │ ├── libutil
42+
> │ │ ├── local.mk
43+
> │ │ …
44+
> │ │ └── data
45+
> │ │ ├── git/tree.txt
46+
> │ │ …
47+
> │ │
48+
> │ ├── libexpr-support
49+
> │ │ ├── local.mk
50+
> │ │ └── tests
51+
> │ │ ├── value/context.hh
52+
> │ │ ├── value/context.cc
53+
> │ │ …
54+
> │ │
55+
> │ ├── libexpr
56+
> │ … ├── local.mk
57+
> │ ├── value/context.cc
58+
> │ …
4659
> …
4760
> ```
4861
49-
The unit tests for each Nix library (`libnixexpr`, `libnixstore`, etc..) live inside a directory `src/${library_shortname}/tests` within the directory for the library (`src/${library_shortname}`).
62+
The tests for each Nix library (`libnixexpr`, `libnixstore`, etc..) live inside a directory `tests/unit/${library_name_without-nix}`.
63+
Given a interface (header) and implementation pair in the original library, say, `src/libexpr/value/context.{hh,cc}`, we write tests for it in `tests/unit/libexpr/tests/value/context.cc`, and (possibly) declare/define additional interfaces for testing purposes in `tests/unit/libexpr-support/tests/value/context.{hh,cc}`.
5064
51-
The data is in `unit-test-data`, with one subdir per library, with the same name as where the code goes.
52-
For example, `libnixstore` code is in `src/libstore`, and its test data is in `unit-test-data/libstore`.
53-
The path to the `unit-test-data` directory is passed to the unit test executable with the environment variable `_NIX_TEST_UNIT_DATA`.
65+
Data for unit tests is stored in a `data` subdir of the directory for each unit test executable.
66+
For example, `libnixstore` code is in `src/libstore`, and its test data is in `tests/unit/libstore/data`.
67+
The path to the `tests/unit/data` directory is passed to the unit test executable with the environment variable `_NIX_TEST_UNIT_DATA`.
68+
Note that each executable only gets the data for its tests.
5469
55-
> **Note**
56-
> Due to the way googletest works, downstream unit test executables will actually include and re-run upstream library tests.
57-
> Therefore it is important that the same value for `_NIX_TEST_UNIT_DATA` be used with the tests for each library.
58-
> That is why we have the test data nested within a single `unit-test-data` directory.
70+
The unit test libraries are in `tests/unit/${library_name_without-nix}-lib`.
71+
All headers are in a `tests` subdirectory so they are included with `#include "tests/"`.
72+
73+
The use of all these separate directories for the unit tests might seem inconvenient, as for example the tests are not "right next to" the part of the code they are testing.
74+
But organizing the tests this way has one big benefit:
75+
there is no risk of any build-system wildcards for the library accidentally picking up test code that should not built and installed as part of the library.
5976
6077
### Running tests
6178
@@ -69,7 +86,7 @@ See [functional characterisation testing](#characterisation-testing-functional)
6986
Like with the functional characterisation, `_NIX_TEST_ACCEPT=1` is also used.
7087
For example:
7188
```shell-session
72-
$ _NIX_TEST_ACCEPT=1 make libstore-tests-exe_RUN
89+
$ _NIX_TEST_ACCEPT=1 make libstore-tests_RUN
7390
...
7491
[ SKIPPED ] WorkerProtoTest.string_read
7592
[ SKIPPED ] WorkerProtoTest.string_write
@@ -80,6 +97,18 @@ $ _NIX_TEST_ACCEPT=1 make libstore-tests-exe_RUN
8097
will regenerate the "golden master" expected result for the `libnixstore` characterisation tests.
8198
The characterisation tests will mark themselves "skipped" since they regenerated the expected result instead of actually testing anything.
8299

100+
### Unit test support libraries
101+
102+
There are headers and code which are not just used to test the library in question, but also downstream libraries.
103+
For example, we do [property testing] with the [rapidcheck] library.
104+
This requires writing `Arbitrary` "instances", which are used to describe how to generate values of a given type for the sake of running property tests.
105+
Because types contain other types, `Arbitrary` "instances" for some type are not just useful for testing that type, but also any other type that contains it.
106+
Downstream types frequently contain upstream types, so it is very important that we share arbitrary instances so that downstream libraries' property tests can also use them.
107+
108+
It is important that these testing libraries don't contain any actual tests themselves.
109+
On some platforms they would be run as part of every test executable that uses them, which is redundant.
110+
On other platforms they wouldn't be run at all.
111+
83112
## Functional tests
84113

85114
The functional tests reside under the `tests/functional` directory and are listed in `tests/functional/local.mk`.

flake.nix

+1-1
Original file line numberDiff line numberDiff line change
@@ -93,7 +93,7 @@
9393
./misc
9494
./precompiled-headers.h
9595
./src
96-
./unit-test-data
96+
./tests/unit
9797
./COPYING
9898
./scripts/local.mk
9999
functionalTestFiles

mk/common-test.sh

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
# Remove overall test dir (at most one of the two should match) and
22
# remove file extension.
33
test_name=$(echo -n "$test" | sed \
4-
-e "s|^unit-test-data/||" \
4+
-e "s|^tests/unit/[^/]*/data/||" \
55
-e "s|^tests/functional/||" \
66
-e "s|\.sh$||" \
77
)

mk/programs.mk

+1-1
Original file line numberDiff line numberDiff line change
@@ -87,6 +87,6 @@ define build-program
8787
# Phony target to run this program (typically as a dependency of 'check').
8888
.PHONY: $(1)_RUN
8989
$(1)_RUN: $$($(1)_PATH)
90-
$(trace-test) $$(UNIT_TEST_ENV) $$($(1)_PATH)
90+
$(trace-test) $$($(1)_ENV) $$($(1)_PATH)
9191

9292
endef

src/libexpr/tests/local.mk

-23
This file was deleted.

src/libstore/tests/local.mk

-37
This file was deleted.

src/libutil/tests/local.mk

-41
This file was deleted.

tests/unit/libexpr-support/local.mk

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
libraries += libexpr-test-support
2+
3+
libexpr-test-support_NAME = libnixexpr-test-support
4+
5+
libexpr-test-support_DIR := $(d)
6+
7+
ifeq ($(INSTALL_UNIT_TESTS), yes)
8+
libexpr-test-support_INSTALL_DIR := $(checklibdir)
9+
else
10+
libexpr-test-support_INSTALL_DIR :=
11+
endif
12+
13+
libexpr-test-support_SOURCES := \
14+
$(wildcard $(d)/tests/*.cc) \
15+
$(wildcard $(d)/tests/value/*.cc)
16+
17+
libexpr-test-support_CXXFLAGS += $(libexpr-tests_EXTRA_INCLUDES)
18+
19+
libexpr-test-support_LIBS = \
20+
libstore-test-support libutil-test-support \
21+
libexpr libstore libutil
22+
23+
libexpr-test-support_LDFLAGS := -lrapidcheck
File renamed without changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
#include <rapidcheck.h>
2+
3+
#include "tests/path.hh"
4+
#include "tests/value/context.hh"
5+
6+
namespace rc {
7+
using namespace nix;
8+
9+
Gen<NixStringContextElem::DrvDeep> Arbitrary<NixStringContextElem::DrvDeep>::arbitrary()
10+
{
11+
return gen::just(NixStringContextElem::DrvDeep {
12+
.drvPath = *gen::arbitrary<StorePath>(),
13+
});
14+
}
15+
16+
Gen<NixStringContextElem> Arbitrary<NixStringContextElem>::arbitrary()
17+
{
18+
switch (*gen::inRange<uint8_t>(0, std::variant_size_v<NixStringContextElem::Raw>)) {
19+
case 0:
20+
return gen::just<NixStringContextElem>(*gen::arbitrary<NixStringContextElem::Opaque>());
21+
case 1:
22+
return gen::just<NixStringContextElem>(*gen::arbitrary<NixStringContextElem::DrvDeep>());
23+
case 2:
24+
return gen::just<NixStringContextElem>(*gen::arbitrary<NixStringContextElem::Built>());
25+
default:
26+
assert(false);
27+
}
28+
}
29+
30+
}

src/libexpr/tests/value/context.hh tests/unit/libexpr-support/tests/value/context.hh

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33

44
#include <rapidcheck/gen/Arbitrary.h>
55

6-
#include <value/context.hh>
6+
#include "value/context.hh"
77

88
namespace rc {
99
using namespace nix;
File renamed without changes.
File renamed without changes.
File renamed without changes.
File renamed without changes.

tests/unit/libexpr/local.mk

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
check: libexpr-tests_RUN
2+
3+
programs += libexpr-tests
4+
5+
libexpr-tests_NAME := libnixexpr-tests
6+
7+
libexpr-tests_ENV := _NIX_TEST_UNIT_DATA=$(d)/data
8+
9+
libexpr-tests_DIR := $(d)
10+
11+
ifeq ($(INSTALL_UNIT_TESTS), yes)
12+
libexpr-tests_INSTALL_DIR := $(checkbindir)
13+
else
14+
libexpr-tests_INSTALL_DIR :=
15+
endif
16+
17+
libexpr-tests_SOURCES := \
18+
$(wildcard $(d)/*.cc) \
19+
$(wildcard $(d)/value/*.cc)
20+
21+
libexpr-tests_EXTRA_INCLUDES = \
22+
-I tests/unit/libexpr-support \
23+
-I tests/unit/libstore-support \
24+
-I tests/unit/libutil-support \
25+
-I src/libexpr \
26+
-I src/libfetchers \
27+
-I src/libstore \
28+
-I src/libutil
29+
30+
libexpr-tests_CXXFLAGS += $(libexpr-tests_EXTRA_INCLUDES)
31+
32+
libexpr-tests_LIBS = \
33+
libexpr-test-support libstore-test-support libutils-test-support \
34+
libexpr libfetchers libstore libutil
35+
36+
libexpr-tests_LDFLAGS := -lrapidcheck $(GTEST_LIBS) -lgmock
File renamed without changes.
File renamed without changes.
File renamed without changes.

0 commit comments

Comments
 (0)