UnityHelperScriptsGuide.md 10.1 KB
Newer Older
T
toby 已提交
1
# Unity Helper Scripts
2

T
toby 已提交
3
## With a Little Help From Our Friends
4 5

Sometimes what it takes to be a really efficient C programmer is a little non-C.
D
Deryew 已提交
6
The Unity project includes a couple of Ruby scripts for making your life just a tad
7 8
easier. They are completely optional. If you choose to use them, you'll need a
copy of Ruby, of course. Just install whatever the latest version is, and it is
T
toby 已提交
9 10
likely to work. You can find Ruby at [ruby-lang.org](https://ruby-labg.org/).

11

T
toby 已提交
12
### `generate_test_runner.rb`
13 14 15 16

Are you tired of creating your own `main` function in your test file? Do you
keep forgetting to add a `RUN_TEST` call when you add a new test case to your
suite? Do you want to use CMock or other fancy add-ons but don't want to figure
T
toby 已提交
17 18 19 20
out how to create your own `RUN_TEST` macro?

Well then we have the perfect script for you!

21 22 23
The `generate_test_runner` script processes a given test file and automatically
creates a separate test runner file that includes ?main?to execute the test
cases within the scanned test file. All you do then is add the generated runner
T
toby 已提交
24 25 26
to your list of files to be compiled and linked, and presto you're done!

This script searches your test file for void function signatures having a
27 28
function name beginning with "test" or "spec". It treats each of these
functions as a test case and builds up a test suite of them. For example, the
T
toby 已提交
29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49
following includes three test cases:

```C
void testVerifyThatUnityIsAwesomeAndWillMakeYourLifeEasier(void)
{
  ASSERT_TRUE(1);
}
void test_FunctionName_should_WorkProperlyAndReturn8(void) {
  ASSERT_EQUAL_INT(8, FunctionName());
}
void spec_Function_should_DoWhatItIsSupposedToDo(void) {
  ASSERT_NOT_NULL(Function(5));
}
```

You can run this script a couple of ways. The first is from the command line:

```Shell
ruby generate_test_runner.rb TestFile.c NameOfRunner.c
```

50
Alternatively, if you include only the test file parameter, the script will copy
51
the name of the test file and automatically append `_Runner` to the name of the
T
toby 已提交
52 53 54 55 56 57
generated file. The example immediately below will create TestFile_Runner.c.

```Shell
ruby generate_test_runner.rb TestFile.c
```

58 59 60 61
You can also add a [YAML](http://www.yaml.org/) file to configure extra options.
Conveniently, this YAML file is of the same format as that used by Unity and
CMock. So if you are using YAML files already, you can simply pass the very same
file into the generator script.
T
toby 已提交
62 63 64 65 66

```Shell
ruby generate_test_runner.rb TestFile.c my_config.yml
```

67 68
The contents of the YAML file `my_config.yml` could look something like the
example below. If you're wondering what some of these options do, you're going
T
toby 已提交
69 70 71 72 73 74 75 76 77 78 79 80
to love the next section of this document.

```YAML
:unity:
  :includes:
    - stdio.h
    - microdefs.h
  :cexception: 1
  :suit_setup: "blah = malloc(1024);"
  :suite_teardown: "free(blah);"
```

81 82
If you would like to force your generated test runner to include one or more
header files, you can just include those at the command line too. Just make sure
T
toby 已提交
83 84 85 86 87 88
these are _after_ the YAML file, if you are using one:

```Shell
ruby generate_test_runner.rb TestFile.c my_config.yml extras.h
```

89 90 91 92
Another option, particularly if you are already using Ruby to orchestrate your
builds - or more likely the Ruby-based build tool Rake - is requiring this
script directly. Anything that you would have specified in a YAML file can be
passed to the script as part of a hash. Let's push the exact same requirement
T
toby 已提交
93 94 95 96 97 98 99 100 101 102 103 104 105
set as we did above but this time through Ruby code directly:

```Ruby
require "generate_test_runner.rb"
options = {
  :includes => ["stdio.h", "microdefs.h"],
  :cexception => 1,
  :suite_setup => "blah = malloc(1024);",
  :suite_teardown => "free(blah);"
}
UnityTestRunnerGenerator.new.run(testfile, runner_name, options)
```

106 107
If you have multiple files to generate in a build script (such as a Rakefile),
you might want to instantiate a generator object with your options and call it
D
Deryew 已提交
108
to generate each runner afterwards. Like thus:
T
toby 已提交
109 110 111 112 113 114 115 116 117

```Ruby
gen = UnityTestRunnerGenerator.new(options)
test_files.each do |f|
  gen.run(f, File.basename(f,'.c')+"Runner.c"
end
```

#### Options accepted by generate_test_runner.rb:
118 119 120 121

The following options are available when executing `generate_test_runner`. You
may pass these as a Ruby hash directly or specify them in a YAML file, both of
which are described above. In the `examples` directory, Example 3's Rakefile
T
toby 已提交
122 123
demonstrates using a Ruby hash.

124

T
toby 已提交
125
##### `:includes`
126

J
John Lindgren 已提交
127
This option specifies an array of file names to be `#include`'d at the top of
128
your runner C file. You might use it to reference custom types or anything else
T
toby 已提交
129 130
universally needed in your generated runners.

131

T
toby 已提交
132
##### `:suite_setup`
133

T
toby 已提交
134 135
Define this option with C code to be executed _before any_ test cases are run.

J
John Lindgren 已提交
136 137 138 139 140
Alternatively, if your C compiler supports weak symbols, you can leave this
option unset and instead provide a `void suiteSetUp(void)` function in your test
suite.  The linker will look for this symbol and fall back to a Unity-provided
stub if it is not found.

141

T
toby 已提交
142
##### `:suite_teardown`
143

J
John Lindgren 已提交
144 145 146 147 148 149 150 151 152
Define this option with C code to be executed _after all_ test cases have
finished.  An integer variable `num_failures` is available for diagnostics.
The code should end with a `return` statement; the value returned will become
the exit code of `main`.  You can normally just return `num_failures`.

Alternatively, if your C compiler supports weak symbols, you can leave this
option unset and instead provide a `int suiteTearDown(int num_failures)`
function in your test suite.  The linker will look for this symbol and fall
back to a Unity-provided stub if it is not found.
T
toby 已提交
153

154

T
toby 已提交
155
##### `:enforce_strict_ordering`
156 157 158 159

This option should be defined if you have the strict order feature enabled in
CMock (see CMock documentation). This generates extra variables required for
everything to run smoothly. If you provide the same YAML to the generator as
T
toby 已提交
160 161
used in CMock's configuration, you've already configured the generator properly.

162 163 164 165 166 167

##### `:externc`

This option should be defined if you are mixing C and CPP and want your test
runners to automatically include extern "C" support when they are generated.

168 169 170 171 172 173
##### `:mock_prefix` and `:mock_suffix`

Unity automatically generates calls to Init, Verify and Destroy for every file
included in the main test file that starts with the given mock prefix and ends
with the given mock suffix, file extension not included. By default, Unity
assumes a `Mock` prefix and no suffix.
174

T
toby 已提交
175
##### `:plugins`
176 177 178 179

This option specifies an array of plugins to be used (of course, the array can
contain only a single plugin). This is your opportunity to enable support for
CException support, which will add a check for unhandled exceptions in each
T
toby 已提交
180 181 182 183 184 185 186 187 188 189 190 191 192
test, reporting a failure if one is detected. To enable this feature using Ruby:

```Ruby
:plugins => [ :cexception ]
```

Or as a yaml file:

```YAML
:plugins:
  -:cexception
```

193 194
If you are using CMock, it is very likely that you are already passing an array
of plugins to CMock. You can just use the same array here. This script will just
T
toby 已提交
195 196
ignore the plugins that don't require additional support.

197 198 199 200 201 202 203 204 205 206 207 208
##### `:include_extensions`

This option specifies the pattern for matching acceptable header file extensions.
By default it will accept hpp, hh, H, and h files. If you need a different combination
of files to search, update this from the default `'(?:hpp|hh|H|h)'`.

##### `:source_extensions`

This option specifies the pattern for matching acceptable source file extensions.
By default it will accept cpp, cc, C, c, and ino files. If you need a different combination
of files to search, update this from the default `'(?:cpp|cc|ino|C|c)'`.

209

T
toby 已提交
210
### `unity_test_summary.rb`
211 212 213 214 215 216 217 218 219 220 221 222

A Unity test file contains one or more test case functions. Each test case can
pass, fail, or be ignored. Each test file is run individually producing results
for its collection of test cases. A given project will almost certainly be
composed of multiple test files. Therefore, the suite of tests is comprised of
one or more test cases spread across one or more test files. This script
aggregates individual test file results to generate a summary of all executed
test cases. The output includes how many tests were run, how many were ignored,
and how many failed. In addition, the output includes a listing of which
specific tests were ignored and failed. A good example of the breadth and
details of these results can be found in the `examples` directory. Intentionally
ignored and failing tests in this project generate corresponding entries in the
T
toby 已提交
223 224
summary report.

225 226
If you're interested in other (prettier?) output formats, check into the
Ceedling build tool project (ceedling.sourceforge.net) that works with Unity and
T
toby 已提交
227 228
CMock and supports xunit-style xml as well as other goodies.

229 230 231 232 233 234 235
This script assumes the existence of files ending with the extensions
`.testpass` and `.testfail`.The contents of these files includes the test
results summary corresponding to each test file executed with the extension set
according to the presence or absence of failures for that test file. The script
searches a specified path for these files, opens each one it finds, parses the
results, and aggregates and prints a summary. Calling it from the command line
looks like this:
T
toby 已提交
236 237 238 239 240

```Shell
ruby unity_test_summary.rb build/test/
```

241 242
You can optionally specify a root path as well. This is really helpful when you
are using relative paths in your tools' setup, but you want to pull the summary
T
toby 已提交
243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275
into an IDE like Eclipse for clickable shortcuts.

```Shell
ruby unity_test_summary.rb build/test/ ~/projects/myproject/
```

Or, if you're more of a Windows sort of person:

```Shell
ruby unity_test_summary.rb build\teat\ C:\projects\myproject\
```

When configured correctly, you'll see a final summary, like so:

```Shell
--------------------------
UNITY IGNORED TEST SUMMARY
--------------------------
blah.c:22:test_sandwiches_should_HaveBreadOnTwoSides:IGNORE

-------------------------
UNITY FAILED TEST SUMMARY
-------------------------
blah.c:87:test_sandwiches_should_HaveCondiments:FAIL:Expected 1 was 0
meh.c:38:test_soda_should_BeCalledPop:FAIL:Expected "pop" was "coke"

--------------------------
OVERALL UNITY TEST SUMMARY
--------------------------
45 TOTAL TESTS 2 TOTAL FAILURES 1 IGNORED
```

How convenient is that?
M
Mark VanderVoord 已提交
276 277 278


*Find The Latest of This And More at [ThrowTheSwitch.org](https://throwtheswitch.org)*