 
    Python program to help test or validate Ansible modules.
validate-modules is one of the ansible-test Sanity Tests, see Sanity Tests for more information.
Originally developed by Matt Martz (@sivel)
cd /path/to/ansible/source
source hacking/env-setup
ansible-test sanity --test validate-modules
usage: validate-modules [-h] [-w] [--exclude EXCLUDE] [--arg-spec]
                        [--base-branch BASE_BRANCH] [--format {json,plain}]
                        [--output OUTPUT]
                        modules [modules ...]
positional arguments:
  modules               Path to module or module directory
optional arguments:
  -h, --help            show this help message and exit
  -w, --warnings        Show warnings
  --exclude EXCLUDE     RegEx exclusion pattern
  --arg-spec            Analyze module argument spec
  --base-branch BASE_BRANCH
                        Used in determining if new options were added
  --format {json,plain}
                        Output format. Default: "plain"
  --output OUTPUT       Output location, use "-" for stdout. Default "-"
The validate-modules tool has a schema.py that is used to validate the YAML blocks, such as DOCUMENTATION and RETURNS.
| code | sample message | 
| 1xx | Locations | 
| 101 | Interpreter line is not #!/usr/bin/python | 
| 102 | Interpreter line is not #!powershell | 
| 103 | Did not find a call to main()(orremoved_module()in the case of deprecated & docs only modules) | 
| 104 | Call to main()not the last line (orremoved_module()in the case of deprecated & docs only modules) | 
| 105 | GPLv3 license header not found | 
| 106 | Import found before documentation variables. All imports must appear below DOCUMENTATION/EXAMPLES/RETURN/ANSIBLE_METADATA | 
| 107 | Imports should be directly below DOCUMENTATION/EXAMPLES/RETURN/ANSIBLE_METADATA | 
| 108 | GPLv3 license header should be the short form for new modules | 
| 109 | Next to last line is not if __name__ == "__main__": | 
| 2xx | Imports | 
| 201 | Did not find a module_utilsimport | 
| 203 | requestsimport found, should useansible.module_utils.urlsinstead | 
| 204 | botoimport found, new modules should useboto3 | 
| 205 | sys.exit()call found. Should beexit_json/fail_json | 
| 206 | WANT_JSONnot found in module | 
| 207 | REPLACER_WINDOWSnot found in module | 
| 208 | module_utilsimports should import specific components, not* | 
| 209 | Only the following from __future__imports are allowed:absolute_import,division, andprint_function. | 
| 3xx | Documentation | 
| 301 | No DOCUMENTATIONprovided | 
| 302 | DOCUMENTATIONis not valid YAML | 
| 303 | DOCUMENTATIONfragment missing | 
| 304 | Unknown DOCUMENTATIONerror | 
| 305 | Invalid DOCUMENTATIONschema | 
| 306 | Module level version_addedis not a valid version number | 
| 307 | Module level version_addedis incorrect | 
| 308 | version_addedfor new option is not a valid version number | 
| 309 | version_addedfor new option is incorrect | 
| 310 | No EXAMPLESprovided | 
| 311 | EXAMPLESis not valid YAML | 
| 312 | No RETURNdocumentation provided | 
| 313 | RETURNis not valid YAML | 
| 314 | No ANSIBLE_METADATAprovided | 
| 315 | ANSIBLE_METADATAwas not provided as a dict, YAML not supported | 
| 316 | Invalid ANSIBLE_METADATAschema | 
| 317 | option is marked as required but specifies a default. Arguments with a default should not be marked as required | 
| 318 | Module deprecated, but DOCUMENTATION.deprecated is missing | 
| 319 | RETURNfragments missing  or invalid | 
| 320 | DOCUMENTATION.optionsmust be a dictionary/hash when used | 
| 321 | Exceptionattempting to import module forargument_specintrospection | 
| 322 | argument is listed in the argument_spec, but not documented in the module | 
| 323 | argument is listed in DOCUMENTATION.options, but not accepted by the module | 
| 324 | Value for “default” from the argument_spec does not match the documentation | 
| 325 | argument_spec defines type=”bool” but documentation does not | 
| 326 | Value for “choices” from the argument_spec does not match the documentation | 
| 327 | Default value from the documentation is not compatible with type defined in the argument_spec | 
| 328 | Choices value from the documentation is not compatible with type defined in the argument_spec | 
| 329 | Default value from the argument_spec is not compatible with type defined in the argument_spec | 
| 330 | Choices value from the argument_spec is not compatible with type defined in the argument_spec | 
| 4xx | Syntax | 
| 401 | Python SyntaxErrorwhile parsing module | 
| 402 | Indentation contains tabs | 
| 403 | Type comparison using type()found. Useisinstance()instead | 
| 5xx | Naming | 
| 501 | Official Ansible modules must have a .pyextension for python
modules or a.ps1for powershell modules | 
| 502 | Ansible module subdirectories must contain an __init__.py | 
| 503 | Missing python documentation file | 
| code | sample message | 
| 1xx | Locations | 
| 107 | Imports should be directly below DOCUMENTATION/EXAMPLES/RETURN/ANSIBLE_METADATAfor legacy modules | 
| 2xx | Imports | 
| 208 | module_utilsimports should import specific components for legacy module, not* | 
| 291 | Try/Except HAS_expression missing | 
| 292 | Did not find ansible.module_utils.basicimport | 
| 3xx | Documentation | 
| 312 | No RETURNdocumentation provided for legacy module | 
| 391 | Unknown pre-existing DOCUMENTATIONerror | 
| 392 | Pre-existing DOCUMENTATIONfragment missing |