| Matthew Treinish | e8ab5f9 | 2017-03-01 15:25:39 -0500 | [diff] [blame] | 1 | .. _tempest_plugin: | 
|  | 2 |  | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 3 | ============================= | 
|  | 4 | Tempest Test Plugin Interface | 
|  | 5 | ============================= | 
|  | 6 |  | 
|  | 7 | Tempest has an external test plugin interface which enables anyone to integrate | 
|  | 8 | an external test suite as part of a tempest run. This will let any project | 
|  | 9 | leverage being run with the rest of the tempest suite while not requiring the | 
|  | 10 | tests live in the tempest tree. | 
|  | 11 |  | 
|  | 12 | Creating a plugin | 
|  | 13 | ================= | 
|  | 14 |  | 
|  | 15 | Creating a plugin is fairly straightforward and doesn't require much additional | 
| Andrea Frittoli (andreaf) | 1370baf | 2016-04-29 14:26:22 -0500 | [diff] [blame] | 16 | effort on top of creating a test suite using tempest.lib. One thing to note with | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 17 | doing this is that the interfaces exposed by tempest are not considered stable | 
|  | 18 | (with the exception of configuration variables which ever effort goes into | 
|  | 19 | ensuring backwards compatibility). You should not need to import anything from | 
| Kiall Mac Innes | 9e6f974 | 2016-05-23 16:20:55 +0100 | [diff] [blame] | 20 | tempest itself except where explicitly noted. | 
|  | 21 |  | 
|  | 22 | Stable Tempest APIs plugins may use | 
|  | 23 | ----------------------------------- | 
|  | 24 |  | 
|  | 25 | As noted above, several tempest APIs are acceptable to use from plugins, while | 
|  | 26 | others are not. A list of stable APIs available to plugins is provided below: | 
|  | 27 |  | 
|  | 28 | * tempest.lib.* | 
|  | 29 | * tempest.config | 
|  | 30 | * tempest.test_discover.plugins | 
| Andrea Frittoli | 17347f0 | 2017-07-26 16:18:30 +0100 | [diff] [blame] | 31 | * tempest.common.credentials_factory | 
| Andrea Frittoli | bf142fc | 2017-10-23 17:30:18 +0200 | [diff] [blame] | 32 | * tempest.clients | 
|  | 33 | * tempest.test | 
| Kiall Mac Innes | 9e6f974 | 2016-05-23 16:20:55 +0100 | [diff] [blame] | 34 |  | 
|  | 35 | If there is an interface from tempest that you need to rely on in your plugin | 
|  | 36 | which is not listed above, it likely needs to be migrated to tempest.lib. In | 
|  | 37 | that situation, file a bug, push a migration patch, etc. to expedite providing | 
|  | 38 | the interface in a reliable manner. | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 39 |  | 
| Marc Koderer | 66210aa | 2015-10-26 10:52:32 +0100 | [diff] [blame] | 40 | Plugin Cookiecutter | 
|  | 41 | ------------------- | 
|  | 42 |  | 
|  | 43 | In order to create the basic structure with base classes and test directories | 
|  | 44 | you can use the tempest-plugin-cookiecutter project:: | 
|  | 45 |  | 
| Yuiko Takada | ccb2bbf | 2015-11-17 10:09:44 +0900 | [diff] [blame] | 46 | > pip install -U cookiecutter && cookiecutter https://git.openstack.org/openstack/tempest-plugin-cookiecutter | 
| Marc Koderer | 66210aa | 2015-10-26 10:52:32 +0100 | [diff] [blame] | 47 |  | 
|  | 48 | Cloning into 'tempest-plugin-cookiecutter'... | 
|  | 49 | remote: Counting objects: 17, done. | 
|  | 50 | remote: Compressing objects: 100% (13/13), done. | 
|  | 51 | remote: Total 17 (delta 1), reused 14 (delta 1) | 
|  | 52 | Unpacking objects: 100% (17/17), done. | 
|  | 53 | Checking connectivity... done. | 
|  | 54 | project (default is "sample")? foo | 
|  | 55 | testclass (default is "SampleTempestPlugin")? FooTempestPlugin | 
|  | 56 |  | 
|  | 57 | This would create a folder called ``foo_tempest_plugin/`` with all necessary | 
|  | 58 | basic classes. You only need to move/create your test in | 
|  | 59 | ``foo_tempest_plugin/tests``. | 
|  | 60 |  | 
|  | 61 | Entry Point | 
|  | 62 | ----------- | 
|  | 63 |  | 
|  | 64 | Once you've created your plugin class you need to add an entry point to your | 
|  | 65 | project to enable tempest to find the plugin. The entry point must be added | 
|  | 66 | to the "tempest.test_plugins" namespace. | 
|  | 67 |  | 
|  | 68 | If you are using pbr this is fairly straightforward, in the setup.cfg just add | 
| Yushiro FURUKAWA | 836361d | 2016-09-30 23:26:58 +0900 | [diff] [blame] | 69 | something like the following: | 
|  | 70 |  | 
|  | 71 | .. code-block:: ini | 
| Marc Koderer | 66210aa | 2015-10-26 10:52:32 +0100 | [diff] [blame] | 72 |  | 
|  | 73 | [entry_points] | 
|  | 74 | tempest.test_plugins = | 
|  | 75 | plugin_name = module.path:PluginClass | 
|  | 76 |  | 
| Matthew Treinish | 00686f2 | 2016-03-09 15:39:19 -0500 | [diff] [blame] | 77 | Standalone Plugin vs In-repo Plugin | 
|  | 78 | ----------------------------------- | 
|  | 79 |  | 
|  | 80 | Since all that's required for a plugin to be detected by tempest is a valid | 
|  | 81 | setuptools entry point in the proper namespace there is no difference from the | 
|  | 82 | tempest perspective on either creating a separate python package to | 
|  | 83 | house the plugin or adding the code to an existing python project. However, | 
|  | 84 | there are tradeoffs to consider when deciding which approach to take when | 
|  | 85 | creating a new plugin. | 
|  | 86 |  | 
|  | 87 | If you create a separate python project for your plugin this makes a lot of | 
|  | 88 | things much easier. Firstly it makes packaging and versioning much simpler, you | 
|  | 89 | can easily decouple the requirements for the plugin from the requirements for | 
|  | 90 | the other project. It lets you version the plugin independently and maintain a | 
|  | 91 | single version of the test code across project release boundaries (see the | 
|  | 92 | `Branchless Tempest Spec`_ for more details on this). It also greatly | 
|  | 93 | simplifies the install time story for external users. Instead of having to | 
|  | 94 | install the right version of a project in the same python namespace as tempest | 
|  | 95 | they simply need to pip install the plugin in that namespace. It also means | 
|  | 96 | that users don't have to worry about inadvertently installing a tempest plugin | 
|  | 97 | when they install another package. | 
|  | 98 |  | 
|  | 99 | .. _Branchless Tempest Spec: http://specs.openstack.org/openstack/qa-specs/specs/tempest/implemented/branchless-tempest.html | 
|  | 100 |  | 
|  | 101 | The sole advantage to integrating a plugin into an existing python project is | 
|  | 102 | that it enables you to land code changes at the same time you land test changes | 
|  | 103 | in the plugin. This reduces some of the burden on contributors by not having | 
|  | 104 | to land 2 changes to add a new API feature and then test it and doing it as a | 
|  | 105 | single combined commit. | 
|  | 106 |  | 
|  | 107 |  | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 108 | Plugin Class | 
| Marc Koderer | 66210aa | 2015-10-26 10:52:32 +0100 | [diff] [blame] | 109 | ============ | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 110 |  | 
|  | 111 | To provide tempest with all the required information it needs to be able to run | 
|  | 112 | your plugin you need to create a plugin class which tempest will load and call | 
|  | 113 | to get information when it needs. To simplify creating this tempest provides an | 
|  | 114 | abstract class that should be used as the parent for your plugin. To use this | 
| Yushiro FURUKAWA | 836361d | 2016-09-30 23:26:58 +0900 | [diff] [blame] | 115 | you would do something like the following: | 
|  | 116 |  | 
|  | 117 | .. code-block:: python | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 118 |  | 
| YAMAMOTO Takashi | cb2ac6e | 2015-10-19 15:54:42 +0900 | [diff] [blame] | 119 | from tempest.test_discover import plugins | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 120 |  | 
| YAMAMOTO Takashi | cb2ac6e | 2015-10-19 15:54:42 +0900 | [diff] [blame] | 121 | class MyPlugin(plugins.TempestPlugin): | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 122 |  | 
| Andrea Frittoli (andreaf) | e07579c | 2016-08-05 07:27:02 +0100 | [diff] [blame] | 123 | Then you need to ensure you locally define all of the mandatory methods in the | 
|  | 124 | abstract class, you can refer to the api doc below for a reference of what that | 
|  | 125 | entails. | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 126 |  | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 127 | Abstract Plugin Class | 
| Marc Koderer | 66210aa | 2015-10-26 10:52:32 +0100 | [diff] [blame] | 128 | --------------------- | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 129 |  | 
|  | 130 | .. autoclass:: tempest.test_discover.plugins.TempestPlugin | 
|  | 131 | :members: | 
|  | 132 |  | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 133 | Plugin Structure | 
| Marc Koderer | 66210aa | 2015-10-26 10:52:32 +0100 | [diff] [blame] | 134 | ================ | 
| zhufl | e9241b5 | 2017-12-06 15:41:08 +0800 | [diff] [blame] | 135 | While there are no hard and fast rules for the structure of a plugin, there are | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 136 | basically no constraints on what the plugin looks like as long as the 2 steps | 
|  | 137 | above are done. However,  there are some recommended patterns to follow to make | 
|  | 138 | it easy for people to contribute and work with your plugin. For example, if you | 
|  | 139 | create a directory structure with something like:: | 
|  | 140 |  | 
|  | 141 | plugin_dir/ | 
|  | 142 | config.py | 
|  | 143 | plugin.py | 
|  | 144 | tests/ | 
|  | 145 | api/ | 
|  | 146 | scenario/ | 
|  | 147 | services/ | 
|  | 148 | client.py | 
|  | 149 |  | 
|  | 150 | That will mirror what people expect from tempest. The file | 
|  | 151 |  | 
|  | 152 | * **config.py**: contains any plugin specific configuration variables | 
|  | 153 | * **plugin.py**: contains the plugin class used for the entry point | 
|  | 154 | * **tests**: the directory where test discovery will be run, all tests should | 
|  | 155 | be under this dir | 
|  | 156 | * **services**: where the plugin specific service clients are | 
|  | 157 |  | 
|  | 158 | Additionally, when you're creating the plugin you likely want to follow all | 
|  | 159 | of the tempest developer and reviewer documentation to ensure that the tests | 
|  | 160 | being added in the plugin act and behave like the rest of tempest. | 
|  | 161 |  | 
| Matthew Treinish | 9392a83 | 2015-08-24 10:00:49 -0400 | [diff] [blame] | 162 | Dealing with configuration options | 
| Marc Koderer | 66210aa | 2015-10-26 10:52:32 +0100 | [diff] [blame] | 163 | ---------------------------------- | 
| Matthew Treinish | 9392a83 | 2015-08-24 10:00:49 -0400 | [diff] [blame] | 164 |  | 
|  | 165 | Historically Tempest didn't provide external guarantees on its configuration | 
|  | 166 | options. However, with the introduction of the plugin interface this is no | 
|  | 167 | longer the case. An external plugin can rely on using any configuration option | 
|  | 168 | coming from Tempest, there will be at least a full deprecation cycle for any | 
|  | 169 | option before it's removed. However, just the options provided by Tempest | 
|  | 170 | may not be sufficient for the plugin. If you need to add any plugin specific | 
|  | 171 | configuration options you should use the ``register_opts`` and | 
|  | 172 | ``get_opt_lists`` methods to pass them to Tempest when the plugin is loaded. | 
|  | 173 | When adding configuration options the ``register_opts`` method gets passed the | 
|  | 174 | CONF object from tempest. This enables the plugin to add options to both | 
|  | 175 | existing sections and also create new configuration sections for new options. | 
|  | 176 |  | 
| Andrea Frittoli (andreaf) | e07579c | 2016-08-05 07:27:02 +0100 | [diff] [blame] | 177 | Service Clients | 
|  | 178 | --------------- | 
|  | 179 |  | 
|  | 180 | If a plugin defines a service client, it is beneficial for it to implement the | 
|  | 181 | ``get_service_clients`` method in the plugin class. All service clients which | 
|  | 182 | are exposed via this interface will be automatically configured and be | 
|  | 183 | available in any instance of the service clients class, defined in | 
|  | 184 | ``tempest.lib.services.clients.ServiceClients``. In case multiple plugins are | 
|  | 185 | installed, all service clients from all plugins will be registered, making it | 
|  | 186 | easy to write tests which rely on multiple APIs whose service clients are in | 
|  | 187 | different plugins. | 
|  | 188 |  | 
| Yushiro FURUKAWA | 836361d | 2016-09-30 23:26:58 +0900 | [diff] [blame] | 189 | Example implementation of ``get_service_clients``: | 
|  | 190 |  | 
|  | 191 | .. code-block:: python | 
| Andrea Frittoli (andreaf) | e07579c | 2016-08-05 07:27:02 +0100 | [diff] [blame] | 192 |  | 
|  | 193 | def get_service_clients(self): | 
|  | 194 | # Example implementation with two service clients | 
|  | 195 | my_service1_config = config.service_client_config('my_service') | 
|  | 196 | params_my_service1 = { | 
|  | 197 | 'name': 'my_service_v1', | 
|  | 198 | 'service_version': 'my_service.v1', | 
|  | 199 | 'module_path': 'plugin_tempest_tests.services.my_service.v1', | 
|  | 200 | 'client_names': ['API1Client', 'API2Client'], | 
|  | 201 | } | 
|  | 202 | params_my_service1.update(my_service_config) | 
|  | 203 | my_service2_config = config.service_client_config('my_service') | 
|  | 204 | params_my_service2 = { | 
|  | 205 | 'name': 'my_service_v2', | 
|  | 206 | 'service_version': 'my_service.v2', | 
|  | 207 | 'module_path': 'plugin_tempest_tests.services.my_service.v2', | 
|  | 208 | 'client_names': ['API1Client', 'API2Client'], | 
|  | 209 | } | 
|  | 210 | params_my_service2.update(my_service2_config) | 
|  | 211 | return [params_my_service1, params_my_service2] | 
|  | 212 |  | 
|  | 213 | Parameters: | 
|  | 214 |  | 
|  | 215 | * **name**: Name of the attribute used to access the ``ClientsFactory`` from | 
|  | 216 | the ``ServiceClients`` instance. See example below. | 
|  | 217 | * **service_version**: Tempest enforces a single implementation for each | 
|  | 218 | service client. Available service clients are held in a ``ClientsRegistry`` | 
|  | 219 | singleton, and registered with ``service_version``, which means that | 
|  | 220 | ``service_version`` must be unique and it should represent the service API | 
|  | 221 | and version implemented by the service client. | 
|  | 222 | * **module_path**: Relative to the service client module, from the root of the | 
|  | 223 | plugin. | 
|  | 224 | * **client_names**: Name of the classes that implement service clients in the | 
|  | 225 | service clients module. | 
|  | 226 |  | 
| Yushiro FURUKAWA | 836361d | 2016-09-30 23:26:58 +0900 | [diff] [blame] | 227 | Example usage of the service clients in tests: | 
|  | 228 |  | 
|  | 229 | .. code-block:: python | 
| Andrea Frittoli (andreaf) | e07579c | 2016-08-05 07:27:02 +0100 | [diff] [blame] | 230 |  | 
|  | 231 | # my_creds is instance of tempest.lib.auth.Credentials | 
|  | 232 | # identity_uri is v2 or v3 depending on the configuration | 
|  | 233 | from tempest.lib.services import clients | 
|  | 234 |  | 
|  | 235 | my_clients = clients.ServiceClients(my_creds, identity_uri) | 
|  | 236 | my_service1_api1_client = my_clients.my_service_v1.API1Client() | 
|  | 237 | my_service2_api1_client = my_clients.my_service_v2.API1Client(my_args='any') | 
|  | 238 |  | 
|  | 239 | Automatic configuration and registration of service clients imposes some extra | 
|  | 240 | constraints on the structure of the configuration options exposed by the | 
|  | 241 | plugin. | 
|  | 242 |  | 
|  | 243 | First ``service_version`` should be in the format `service_config[.version]`. | 
|  | 244 | The `.version` part is optional, and should only be used if there are multiple | 
|  | 245 | versions of the same API available. The `service_config` must match the name of | 
|  | 246 | a configuration options group defined by the plugin. Different versions of one | 
|  | 247 | API must share the same configuration group. | 
|  | 248 |  | 
|  | 249 | Second the configuration options group `service_config` must contain the | 
|  | 250 | following options: | 
|  | 251 |  | 
|  | 252 | * `catalog_type`: corresponds to `service` in the catalog | 
|  | 253 | * `endpoint_type` | 
|  | 254 |  | 
|  | 255 | The following options will be honoured if defined, but they are not mandatory, | 
|  | 256 | as they do not necessarily apply to all service clients. | 
|  | 257 |  | 
|  | 258 | * `region`: default to identity.region | 
|  | 259 | * `build_timeout` : default to compute.build_timeout | 
|  | 260 | * `build_interval`: default to compute.build_interval | 
|  | 261 |  | 
|  | 262 | Third the service client classes should inherit from ``RestClient``, should | 
|  | 263 | accept generic keyword arguments, and should pass those arguments to the | 
|  | 264 | ``__init__`` method of ``RestClient``. Extra arguments can be added. For | 
| Yushiro FURUKAWA | 836361d | 2016-09-30 23:26:58 +0900 | [diff] [blame] | 265 | instance: | 
|  | 266 |  | 
|  | 267 | .. code-block:: python | 
| Andrea Frittoli (andreaf) | e07579c | 2016-08-05 07:27:02 +0100 | [diff] [blame] | 268 |  | 
|  | 269 | class MyAPIClient(rest_client.RestClient): | 
|  | 270 |  | 
|  | 271 | def __init__(self, auth_provider, service, region, | 
|  | 272 | my_arg, my_arg2=True, **kwargs): | 
|  | 273 | super(MyAPIClient, self).__init__( | 
|  | 274 | auth_provider, service, region, **kwargs) | 
|  | 275 | self.my_arg = my_arg | 
|  | 276 | self.my_args2 = my_arg | 
|  | 277 |  | 
|  | 278 | Finally the service client should be structured in a python module, so that all | 
|  | 279 | service client classes are importable from it. Each major API version should | 
|  | 280 | have its own module. | 
|  | 281 |  | 
|  | 282 | The following folder and module structure is recommended for a single major | 
|  | 283 | API version:: | 
|  | 284 |  | 
|  | 285 | plugin_dir/ | 
|  | 286 | services/ | 
|  | 287 | __init__.py | 
|  | 288 | client_api_1.py | 
|  | 289 | client_api_2.py | 
|  | 290 |  | 
| Yushiro FURUKAWA | 836361d | 2016-09-30 23:26:58 +0900 | [diff] [blame] | 291 | The content of __init__.py module should be: | 
|  | 292 |  | 
|  | 293 | .. code-block:: python | 
| Andrea Frittoli (andreaf) | e07579c | 2016-08-05 07:27:02 +0100 | [diff] [blame] | 294 |  | 
|  | 295 | from client_api_1.py import API1Client | 
|  | 296 | from client_api_2.py import API2Client | 
|  | 297 |  | 
|  | 298 | __all__ = ['API1Client', 'API2Client'] | 
|  | 299 |  | 
|  | 300 | The following folder and module structure is recommended for multiple major | 
|  | 301 | API version:: | 
|  | 302 |  | 
|  | 303 | plugin_dir/ | 
|  | 304 | services/ | 
|  | 305 | v1/ | 
|  | 306 | __init__.py | 
|  | 307 | client_api_1.py | 
|  | 308 | client_api_2.py | 
|  | 309 | v2/ | 
|  | 310 | __init__.py | 
|  | 311 | client_api_1.py | 
|  | 312 | client_api_2.py | 
|  | 313 |  | 
| Yushiro FURUKAWA | 836361d | 2016-09-30 23:26:58 +0900 | [diff] [blame] | 314 | The content each of __init__.py module under vN should be: | 
|  | 315 |  | 
|  | 316 | .. code-block:: python | 
| Andrea Frittoli (andreaf) | e07579c | 2016-08-05 07:27:02 +0100 | [diff] [blame] | 317 |  | 
|  | 318 | from client_api_1.py import API1Client | 
|  | 319 | from client_api_2.py import API2Client | 
|  | 320 |  | 
|  | 321 | __all__ = ['API1Client', 'API2Client'] | 
|  | 322 |  | 
| Matthew Treinish | 3a851dc | 2015-07-30 11:34:03 -0400 | [diff] [blame] | 323 | Using Plugins | 
|  | 324 | ============= | 
|  | 325 |  | 
|  | 326 | Tempest will automatically discover any installed plugins when it is run. So by | 
|  | 327 | just installing the python packages which contain your plugin you'll be using | 
|  | 328 | them with tempest, nothing else is really required. | 
|  | 329 |  | 
|  | 330 | However, you should take care when installing plugins. By their very nature | 
|  | 331 | there are no guarantees when running tempest with plugins enabled about the | 
|  | 332 | quality of the plugin. Additionally, while there is no limitation on running | 
|  | 333 | with multiple plugins it's worth noting that poorly written plugins might not | 
|  | 334 | properly isolate their tests which could cause unexpected cross interactions | 
|  | 335 | between plugins. | 
|  | 336 |  | 
|  | 337 | Notes for using plugins with virtualenvs | 
|  | 338 | ---------------------------------------- | 
|  | 339 |  | 
|  | 340 | When using a tempest inside a virtualenv (like when running under tox) you have | 
|  | 341 | to ensure that the package that contains your plugin is either installed in the | 
|  | 342 | venv too or that you have system site-packages enabled. The virtualenv will | 
|  | 343 | isolate the tempest install from the rest of your system so just installing the | 
|  | 344 | plugin package on your system and then running tempest inside a venv will not | 
|  | 345 | work. | 
|  | 346 |  | 
|  | 347 | Tempest also exposes a tox job, all-plugin, which will setup a tox virtualenv | 
|  | 348 | with system site-packages enabled. This will let you leverage tox without | 
|  | 349 | requiring to manually install plugins in the tox venv before running tests. |