1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 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 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502
|
Puppet Labs Spec Helper
=======================
[](https://github.com/puppetlabs/puppetlabs_spec_helper/blob/main/CODEOWNERS)


The Short Version
-----------------
This repository is meant to provide a single source of truth for how to
initialize different Puppet versions for spec testing.
The common use case is a module such as
[stdlib](http://forge.puppetlabs.com/puppetlabs/stdlib) that works with many
versions of Puppet. The stdlib module should require the spec helper in this
repository, which will in turn automatically figure out the version of Puppet
being tested against and perform version specific initialization.
Other "customers" that should use this module are:
* [Facter](https://github.com/puppetlabs/facter)
* [PuppetDB](https://github.com/puppetlabs/puppetdb)
* [Mount Providers](https://github.com/puppetlabs/puppetlabs-mount_providers)
Usage
=====
When developing or testing modules, simply clone this repository and install the
gem it contains. The recommended way to do this is using [bundler](http://bundler.io/#getting-started).
Example Gemfile:
source 'https://rubygems.org'
gem 'puppetlabs_spec_helper'
Add this to your project's spec\_helper.rb:
require 'puppetlabs_spec_helper/module_spec_helper'
Add this to your project's Rakefile:
require 'puppetlabs_spec_helper/rake_tasks'
And run the spec tests:
$ cd $modulename
$ rake spec
### Parallel Fixture Downloads
Fixture downloads will now execute in parallel to speed up the testing process. Which can represent >= 600% speed increase (depending on number of threads). You can control the amount of threads by setting the `MAX_FIXTURE_THREAD_COUNT` environment variable
to a positive integer, the default is currently 10. We don't suggest going higher than 25 as the gains are marginal due to some repos taking a long time to download. Please be aware that your internal VCS system may not be able to handle a high load in which case the server would fail to clone the repository. Because of this issue, this setting is tunable via `MAX_FIXTURE_THREAD_COUNT`.
Additionally, you can also speed up cloning when using the ssh protocol by multiplexing ssh sessions. Add something similar to your ssh config.
Note: you may need to change the host if you're using an internal git server.
```shell
Host github.com
ControlMaster auto
ControlPath ~/.ssh/ssh-%r@%h:%p
ControlPersist yes
```
Note: parallel downloads are available for repositories and forge modules.
### Parallel tests
It is also possible to use the `parallel_tests` Gem via the `:parallel_spec` Rake task to run rspec commands in parallel on groups of spec files.
Use of parallelization at this level can result in large performance benefits when the Rspec examples tend to cause a number of large, CPU-intensive catalog compilations to occur. An example of where this might be the case is in a complex module with a lot of tests or a control repo with many hosts.
Be aware however that in other circumstances this parallelization can result in the tests actually taking longer to run. The best thing to do is time `rspec spec` and `rspec parallel_spec` and use the parallelization only when there is a clear benefit.
To enable this feature, add the `parallel_tests` Gem to your project's Gemfile:
gem 'parallel_tests'
And then to run spec tests in parallel:
$ rake parallel_spec
Issues
======
Please file issues against this project at the [Puppet Labs Issue Tracker](https://tickets.puppetlabs.com/browse/MODULES)
The Long Version
----------------
Purpose of this Project
=======================
This project is intended to serve two purposes:
1. To serve as a bridge between external projects and multiple versions of puppet;
in other words, if your project has a dependency on puppet, you shouldn't need
to need to worry about the details of how to initialize puppet's state for
testing, no matter what version of puppet you are testing against.
2. To provide some convenience classes / methods for doing things like creating
tempfiles, common rspec matchers, etc. These classes are in the puppetlabs\_spec
directory.
3. To provide a common set of Rake tasks so that the procedure for testing modules
is unified.
To Use this Project
===================
The most common usage scenario is that you will check out the 'main'
branch of this project from github, and install it as a rubygem.
There should be few or no cases where you would want to have any other
branch of this project besides main/HEAD.
Running on non-current ruby versions
------------------------------------
Since gem and bundler, ruby's package management tools, do not take the target ruby version into account when downloading packages, the puppetlabs_spec_helper gem can only depend on gems that are available for all supported ruby versions. If you can/want to use features from other packages, install those additional packages manually, or have a look at the Gemfile, which provides code to specify those dependencies in a more "friendly" way. This currently affects the following gems:
* puppet
* rubocop
* rubocop-rspec
* json_pure
* rack
Initializing Puppet for Testing
===============================
In most cases, your project should be able to define a spec\_helper.rb that includes
just this one simple line:
require 'puppetlabs_spec_helper/puppet_spec_helper'
Then, as long as the gem is installed, you should be all set.
If you are using rspec-puppet for module testing, you will want to include a different
library:
require 'puppetlabs_spec_helper/module_spec_helper'
NOTE that this is specifically for initializing Puppet's core. If your project does
not have any dependencies on puppet and you just want to use the utility classes,
see the next section.
A number of the Puppet parser features, controlled via configuration during a
normal puppet run, can be controlled by exporting specific environment
variables for the spec run. These are:
* ``STRICT_VARIABLES`` - Set to "no" to disable strict variable checking.
See [strict_variables](http://docs.puppetlabs.com/references/latest/configuration.html#strictvariables) in puppet.conf for details.
* ``ORDERING`` - set to the desired ordering method ("title-hash", "manifest", or "random")
to set the order of unrelated resources when applying a catalog. Leave unset for the default
behavior, currently "random". This is equivalent to setting [ordering](http://docs.puppetlabs.com/references/latest/configuration.html#ordering)
in puppet.conf.
As an example, to run spec tests with the future parser, strict variable checking,
and manifest ordering, you would:
ORDERING=manifest rake spec
Using Utility Classes
=====================
If you'd like to use the Utility classes (PuppetlabsSpec::Files,
PuppetlabsSpec::Fixtures), you just need to add this to your project's spec\_helper.rb:
require 'puppetlabs_spec_helper/puppetlabs_spec_helper'
NOTE that the above line happens automatically if you've required
'puppetlabs\_spec\_helper/puppet\_spec\_helper', so you don't need to do both.
In either case, you'll have all of the functionality of Puppetlabs::Files,
Puppetlabs::Fixtures, etc., mixed-in to your rspec context.
Using Fixtures
==============
`puppetlabs_spec_helper` has the ability to populate the
`spec/fixtures/modules` directory with dependent modules when `rake spec` or
`rake spec_prep` is run. To do so, all required modules should be listed in a
file named `.fixtures.yml` in the root of the project. You can specify a alternate location for that file in the `FIXTURES_YML` environment variable.
You can use the `MODULE_WORKING_DIR` environment variable to specify a diffent location when installing module fixtures via the forge. By default the
working directory is `<module directory>/spec/fixtures/work-dir`.
When specifying the repo source of the fixture you have a few options as to which revision of the codebase you wish to use, and optionally, the puppet versions where the fixture is needed.
* `repo` - the url to the repo
* `scm` - options include git or hg. This is an optional step as the helper code will figure out which scm is used.
```yaml
scm: git
scm: hg
```
* `target` - the directory name to clone the repo into ie. `target: mymodule` defaults to the repo name (Optional)
* `subdir` - directory to be removed from the cloned repo. Its contents will be moved to the root directory (Optional)
* `ref` - used to specify the tag name (like version) or commit hash to be checked out (Optional). Branch names should use the `branch` option instead.
```yaml
ref: 1.0.0
ref: 880fca52c
```
* `branch` - used to specify the branch name you want to use ie. `branch: development`
* `flags` - additional flags passed to the module installer (both puppet and scm)
```yaml
flags: --verbose
```
* `puppet_version` - versions of puppet for which the fixture should be installed. Ruby version constraints are supported. Only works when the `semantic_puppet` gem is available (shipped with puppet 4.0 and up, by default).
```yaml
puppet_version: '>= 6.0.0'
```
**Notes:**
* `ref` and `branch` can be used together to get a specific revision on a specific branch
* Top level `defaults` option could be used to set global options
Using Forge Authorization
-----------------
In order to perform forge operations which require authorization, such as installing premium modules, you can export your forge api key as an environment variable in your terminal.
```bash
FORGE_API_KEY='your_api_key'
```
puppetlabs_spec_helper will then automatically append this key to all `puppet module install` requests when running `rake spec_prep`.
Fixtures Examples
-----------------
Basic fixtures that will symlink `spec/fixtures/modules/my_modules` to the
project root:
```yaml
fixtures:
symlinks:
my_module: "#{source_dir}"
```
This is the same as specifying no symlinks fixtures at all.
Add `firewall` and `stdlib` as required module fixtures:
```yaml
fixtures:
repositories:
firewall: "https://github.com/puppetlabs/puppetlabs-firewall.git"
stdlib: "https://github.com/puppetlabs/puppetlabs-stdlib.git"
```
Put a supplementary repository at a different location
```yaml
fixtures:
repositories:
firewall: "https://github.com/puppetlabs/puppetlabs-firewall.git"
stdlib: "https://github.com/puppetlabs/puppetlabs-stdlib.git"
control_repo:
repo: "https://github.com/puppetlabs/control-repo"
target: "spec/fixtures/control_repos"
```
Specify that the git tag `2.4.2` of `stdlib` should be checked out:
```yaml
fixtures:
repositories:
firewall: "https://github.com/puppetlabs/puppetlabs-firewall.git"
stdlib:
repo: "https://github.com/puppetlabs/puppetlabs-stdlib.git"
ref: "2.6.0"
```
Only install the `yumrepo_core` module when testing against Puppet 7 or greater:
```yaml
fixtures:
repositories:
yumrepo_core:
repo: "https://github.com/puppetlabs/puppetlabs-yumrepo_core.git"
puppet_version: ">= 6.0.0"
```
Move manifests and siblings to root directory when they are inside a `code` directory:
```yaml
fixtures:
repositories:
stdlib:
repo: "https://github.com/puppetlabs/puppetlabs-extradirectory.git"
subdir: "code"
```
Install modules from Puppet Forge:
```yaml
fixtures:
forge_modules:
firewall: "puppetlabs/firewall"
stdlib:
repo: "puppetlabs/stdlib"
ref: "2.6.0"
```
Pass additional flags to module installation:
```yaml
fixtures:
forge_modules:
stdlib:
repo: "puppetlabs/stdlib"
ref: "2.6.0"
flags: "--module_repository https://my_repo.com"
repositories:
firewall:
repo: "https://github.com/puppetlabs/puppetlabs-firewall.git"
ref: "2.6.0"
flags: "--verbose"
```
Use `defaults` to define global parameters:
```yaml
defaults:
forge_modules:
flags: "--module_repository https://my_repo.com"
fixtures:
forge_modules:
stdlib:
repo: "puppetlabs/stdlib"
ref: "2.6.0"
repositories:
firewall:
repo: "https://github.com/puppetlabs/puppetlabs-firewall.git"
ref: "2.6.0"
```
Fixture Loading
---------------
Any module that has a `spec/lib` directory will be available on the ruby `LOAD_PATH` for tests to consume. This allows modules to provide additional helper code to be supplied. The [augeasprovider_core](https://github.com/hercules-team/augeasproviders_core) module has [some examples](https://github.com/hercules-team/augeasproviders_core/tree/master/spec/lib).
Testing Parser Functions
========================
This whole section is superseded by improved support of accessing the scope in
rspec-puppet.
Modify rspec behavior
================================
#### Running Rspec with additional settings
You can add command line options to rspec using the `CI_SPEC_OPTIONS` environment variable. Any text in the `CI_SPEC_OPTIONS` environment variable is added as an rspec option. For example:
If you wanted to output JUnit test reports for a Jenkins CI server you could use;
Bash
``` bash
export CI_SPEC_OPTIONS = "-r yarjuf -f JUnit -o result.xml"
```
PowerShell
``` bash
$ENV:CI_SPEC_OPTIONS = '-r yarjuf -f JUnit -o result.xml'
```
And then run
```
bundle exec rake spec
```
This would cause rspec to load the `yarjuf` gem and output the results in JUnit format to the file `result.xml`
#### Running specs in parallel
When executing tests in a matrix CI environment, tests can be split up to run
a share of specs per CI node in parallel. Set the ``CI_NODE_TOTAL`` environment
variable to the total number of nodes, and the ``CI_NODE_INDEX`` to a number
between 1 and the ``CI_NODE_TOTAL``.
If using Travis CI, add new lines to the "env" section of .travis.yml per node,
remembering to duplicate any existing environment variables:
env:
- FUTURE_PARSER=yes CI_NODE_TOTAL=2 CI_NODE_INDEX=1
- FUTURE_PARSER=yes CI_NODE_TOTAL=2 CI_NODE_INDEX=2
#### Running tests tagged with test tiers
To run tests tagged with risk levels set the ``TEST_TIERS`` environment variable to a comma-separated list of the appropriate tiers.
For example: to run tests marked ``tier_high => true`` and ``tier_medium => true`` in the same test run set the
environment variable``TEST_TIERS=high,medium``
By default ``TEST_TIERS`` only accepts low, medium and high as valid tiers. If you would like to use your own keywords to set the environment variable ``TEST_TIERS_ALLOWED``.
For example: to use the keywords dev, rnd, staging and production you can set
``TEST_TIERS_ALLOWED=dev,rnd,staging,production``. Then you would be able to run tests marked ``tier_dev => true``, ``tier_production => true`` with ``TEST_TIERS=dev,production``
Note, if the ``TEST_TIERS`` environment variable is set to empty string or nil, all tiers will be executed.
Generating code coverage reports
================================
This section describes how to add code coverage reports for Ruby files (types, providers, ...).
See the documentation of [RSpec-Puppet](https://github.com/rodjek/rspec-puppet)
for Puppet manifest coverage reports.
Starting with Ruby 1.9, the *de facto* standard for Ruby code coverage is
[SimpleCov](https://github.com/colszowka/simplecov). It is implemented as a Rake task in this gem.
To run code coverage:
$ rake spec:simplecov
Reports are written to `/coverage/`, which you should add to `.gitignore`.
The reports can be generated every time you invoke RSpec, e.g. via `rake spec`,
You can enable it, set the following environment variable:s
``SIMPLECOV=yes``
Remember to add the simplecov-console gem to your `Gemfile`. If you run `spec:simplecov` on Travis-CI or any of the other supported CI services, reports get generated which can then be uploaded to [codecov.io](https://codecov.io) with the recommended [codecov uploader](https://docs.codecov.com/docs/codecov-uploader).
Some Notes for Windows Users
============================
A windows users may need to do one of two things to execute `rake spec`.
Although things may appear to work, the init.pp may not transfer to the fixtures folder as needed
or may transfer as an empty file.
This is related to a registry security setting requiring elevated privileges to create symbolic links.
Currently, there are two known approaches to get around this problem.
- run your windows shell (cmd) as an Administrator
or
- modify the registry entry settings to allow symbolic links to be created.
The following links may give you some insight into why...
[Server Fault Post](http://serverfault.com/questions/582944/puppet-file-link-doesnt-create-target-under-windows)
[Stack Overflow Post](http://stackoverflow.com/questions/229643/how-do-i-overcome-the-the-symbolic-link-cannot-be-followed-because-its-type-is)
[Microsoft TechNet](https://technet.microsoft.com/en-us/library/cc754077.aspx)
mock_with
=========
There are two major mocking frameworks in modules test suites today: [mocha](https://rubygems.org/gems/mocha) and [rspec-mocks](https://rubygems.org/gems/rspec-mocks). We recommend that you choose rspec-mocks explicitly by specifying `mock_with`, either in your `spec_helper.rb` like so:
```
RSpec.configure do |c|
c.mock_with :rspec
end
require 'puppetlabs_spec_helper/module_spec_helper'
```
or by using Puppet Development Kit's [`mock_with` option in `.sync.yml`](https://github.com/puppetlabs/pdk-templates#specspec_helperrb) and `pdk update`.
You can also continue to use mocha by explicitly specifying `:mocha`, following the [mocha documentation](http://gofreerange.com/mocha/docs/).
Migration
---------
To migrate from mocha to rspec-mocks, in many simple cases the following two kinds of changes are all you need:
Translate all stubs:
```
context.stubs(:some_method).with(arguments).returns('value')
```
to this:
```
allow(context).to receive(:some_method).with(arguments).and_return('value')
```
Translate all expectations:
```
context.expects(:some_method).with(arguments).returns('value')
```
to this:
```
expect(context).to receive(:some_method).with(arguments).and_return('value')
```
Rationale
---------
* As a part of the RSpec project, rspec-mocks integration is better.
* mocha is extending base ruby objects with its `stubs`, and `expects` methods, polluting the global namespace, with a potential for subtle and gnarly errors.
* Currently both rspec-mocks and mocha get loaded, leading to test suites that use both.
* puppetlabs_spec_helper loads and configures mocha unconditionally, causing friction when a different mocking framework is wanted.
* mocha is an additional dependency to carry.
EOF
## License
This codebase is licensed under Apache 2.0. However, the open source dependencies included in this codebase might be subject to other software licenses such as AGPL, GPL2.0, and MIT.
|