mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 21:44:00 -07:00 
			
		
		
		
	[stable-10] Use antsibull-nox instead of extra sanity test runner and extra workfows (#10042)
Use antsibull-nox instead of extra sanity test runner and extra workflows (#10022)
* Use antsibull-nox instead of extra sanity test runner and extra workflows.
* Avoid sys.argv[0].
(cherry picked from commit 3ee55c6828)
	
	
This commit is contained in:
		
					parent
					
						
							
								11a847a7b5
							
						
					
				
			
			
				commit
				
					
						9dd7be05dc
					
				
			
		
					 28 changed files with 302 additions and 655 deletions
				
			
		|  | @ -70,7 +70,6 @@ stages: | ||||||
|             - test: 2 |             - test: 2 | ||||||
|             - test: 3 |             - test: 3 | ||||||
|             - test: 4 |             - test: 4 | ||||||
|             - test: extra |  | ||||||
|   - stage: Sanity_2_18 |   - stage: Sanity_2_18 | ||||||
|     displayName: Sanity 2.18 |     displayName: Sanity 2.18 | ||||||
|     dependsOn: [] |     dependsOn: [] | ||||||
|  |  | ||||||
							
								
								
									
										20
									
								
								.github/workflows/import-galaxy.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								.github/workflows/import-galaxy.yml
									
										
									
									
										vendored
									
									
								
							|  | @ -1,20 +0,0 @@ | ||||||
| --- |  | ||||||
| # Copyright (c) Ansible Project |  | ||||||
| # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| 
 |  | ||||||
| name: import-galaxy |  | ||||||
| 'on': |  | ||||||
|   # Run CI against all pushes (direct commits, also merged PRs) to main, and all Pull Requests |  | ||||||
|   push: |  | ||||||
|     branches: |  | ||||||
|       - main |  | ||||||
|       - stable-* |  | ||||||
|   pull_request: |  | ||||||
| 
 |  | ||||||
| jobs: |  | ||||||
|   import-galaxy: |  | ||||||
|     permissions: |  | ||||||
|       contents: read |  | ||||||
|     name: Test to import built collection artifact with Galaxy importer |  | ||||||
|     uses: ansible-community/github-action-test-galaxy-import/.github/workflows/test-galaxy-import.yml@main |  | ||||||
							
								
								
									
										28
									
								
								.github/workflows/nox.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								.github/workflows/nox.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,28 @@ | ||||||
|  | --- | ||||||
|  | # Copyright (c) Ansible Project | ||||||
|  | # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||||
|  | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  | 
 | ||||||
|  | name: nox | ||||||
|  | 'on': | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - main | ||||||
|  |       - stable-* | ||||||
|  |   pull_request: | ||||||
|  |   # Run CI once per day (at 08:00 UTC) | ||||||
|  |   schedule: | ||||||
|  |     - cron: '0 8 * * *' | ||||||
|  |   workflow_dispatch: | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   nox: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     name: "Run extra sanity tests" | ||||||
|  |     steps: | ||||||
|  |       - name: Check out collection | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |         with: | ||||||
|  |           persist-credentials: false | ||||||
|  |       - name: Run nox | ||||||
|  |         uses: ansible-community/antsibull-nox@main | ||||||
							
								
								
									
										35
									
								
								.github/workflows/reuse.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										35
									
								
								.github/workflows/reuse.yml
									
										
									
									
										vendored
									
									
								
							|  | @ -1,35 +0,0 @@ | ||||||
| --- |  | ||||||
| # Copyright (c) Ansible Project |  | ||||||
| # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| 
 |  | ||||||
| name: Verify REUSE |  | ||||||
| 
 |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|     branches: |  | ||||||
|       - main |  | ||||||
|       - stable-* |  | ||||||
|   pull_request: |  | ||||||
|     types: [opened, synchronize, reopened] |  | ||||||
|     branches: |  | ||||||
|       - main |  | ||||||
|       - stable-* |  | ||||||
|   # Run CI once per day (at 07:30 UTC) |  | ||||||
|   schedule: |  | ||||||
|     - cron: '30 7 * * *' |  | ||||||
| 
 |  | ||||||
| jobs: |  | ||||||
|   check: |  | ||||||
|     permissions: |  | ||||||
|       contents: read |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
| 
 |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           persist-credentials: false |  | ||||||
|           ref: ${{ github.event.pull_request.head.sha || '' }} |  | ||||||
| 
 |  | ||||||
|       - name: REUSE Compliance Check |  | ||||||
|         uses: fsfe/reuse-action@v5 |  | ||||||
|  | @ -9,6 +9,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | ||||||
| [](https://docs.ansible.com/ansible/latest/collections/community/general/) | [](https://docs.ansible.com/ansible/latest/collections/community/general/) | ||||||
| [](https://dev.azure.com/ansible/community.general/_build?definitionId=31) | [](https://dev.azure.com/ansible/community.general/_build?definitionId=31) | ||||||
| [](https://github.com/ansible-collections/community.general/actions) | [](https://github.com/ansible-collections/community.general/actions) | ||||||
|  | [](https://github.com/ansible-collections/community.general/actions) | ||||||
| [](https://codecov.io/gh/ansible-collections/community.general) | [](https://codecov.io/gh/ansible-collections/community.general) | ||||||
| [](https://api.reuse.software/info/github.com/ansible-collections/community.general) | [](https://api.reuse.software/info/github.com/ansible-collections/community.general) | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										48
									
								
								antsibull-nox.toml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								antsibull-nox.toml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,48 @@ | ||||||
|  | # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||||
|  | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  | # SPDX-FileCopyrightText: 2025 Felix Fontein <felix@fontein.de> | ||||||
|  | 
 | ||||||
|  | [collection_sources] | ||||||
|  | "ansible.posix" = "git+https://github.com/ansible-collections/ansible.posix.git,main" | ||||||
|  | "community.crypto" = "git+https://github.com/ansible-collections/community.crypto.git,main" | ||||||
|  | "community.docker" = "git+https://github.com/ansible-collections/community.docker.git,main" | ||||||
|  | "community.internal_test_tools" = "git+https://github.com/ansible-collections/community.internal_test_tools.git,main" | ||||||
|  | 
 | ||||||
|  | [sessions] | ||||||
|  | 
 | ||||||
|  | [sessions.docs_check] | ||||||
|  | validate_collection_refs="all" | ||||||
|  | 
 | ||||||
|  | [sessions.license_check] | ||||||
|  | 
 | ||||||
|  | [sessions.extra_checks] | ||||||
|  | run_no_unwanted_files = true | ||||||
|  | no_unwanted_files_module_extensions = [".py"] | ||||||
|  | no_unwanted_files_yaml_extensions = [".yml"] | ||||||
|  | run_action_groups = true | ||||||
|  | 
 | ||||||
|  | [[sessions.extra_checks.action_groups_config]] | ||||||
|  | name = "consul" | ||||||
|  | pattern = "^consul_.*$" | ||||||
|  | exclusions = [ | ||||||
|  |     "consul_acl_bootstrap", | ||||||
|  |     "consul_kv", | ||||||
|  | ] | ||||||
|  | doc_fragment = "community.general.consul.actiongroup_consul" | ||||||
|  | 
 | ||||||
|  | [[sessions.extra_checks.action_groups_config]] | ||||||
|  | name = "keycloak" | ||||||
|  | pattern = "^keycloak_.*$" | ||||||
|  | exclusions = [ | ||||||
|  |     "keycloak_realm_info", | ||||||
|  | ] | ||||||
|  | doc_fragment = "community.general.keycloak.actiongroup_keycloak" | ||||||
|  | 
 | ||||||
|  | [[sessions.extra_checks.action_groups_config]] | ||||||
|  | name = "proxmox" | ||||||
|  | pattern = "^proxmox(_.*)?$" | ||||||
|  | exclusions = [] | ||||||
|  | doc_fragment = "community.general.proxmox.actiongroup_proxmox" | ||||||
|  | 
 | ||||||
|  | [sessions.build_import_check] | ||||||
|  | run_galaxy_importer = true | ||||||
							
								
								
									
										38
									
								
								noxfile.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								noxfile.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,38 @@ | ||||||
|  | # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||||
|  | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  | # SPDX-FileCopyrightText: 2025 Felix Fontein <felix@fontein.de> | ||||||
|  | 
 | ||||||
|  | # /// script | ||||||
|  | # dependencies = ["nox>=2025.02.09", "antsibull-nox"] | ||||||
|  | # /// | ||||||
|  | 
 | ||||||
|  | import sys | ||||||
|  | 
 | ||||||
|  | import nox | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | try: | ||||||
|  |     import antsibull_nox | ||||||
|  | except ImportError: | ||||||
|  |     print("You need to install antsibull-nox in the same Python environment as nox.") | ||||||
|  |     sys.exit(1) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | antsibull_nox.load_antsibull_nox_toml() | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @nox.session(name="aliases", python=False, default=True) | ||||||
|  | def aliases(session: nox.Session) -> None: | ||||||
|  |     session.run("python", "tests/sanity/extra/aliases.py") | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @nox.session(name="botmeta", default=True) | ||||||
|  | def botmeta(session: nox.Session) -> None: | ||||||
|  |     session.install("PyYAML", "voluptuous") | ||||||
|  |     session.run("python", "tests/sanity/extra/botmeta.py") | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # Allow to run the noxfile with `python noxfile.py`, `pipx run noxfile.py`, or similar. | ||||||
|  | # Requires nox >= 2025.02.09 | ||||||
|  | if __name__ == "__main__": | ||||||
|  |     nox.main() | ||||||
|  | @ -1,12 +0,0 @@ | ||||||
| { |  | ||||||
|     "include_symlinks": true, |  | ||||||
|     "prefixes": [ |  | ||||||
|         "meta/runtime.yml", |  | ||||||
|         "plugins/modules/", |  | ||||||
|         "tests/sanity/extra/action-group." |  | ||||||
|     ], |  | ||||||
|     "output": "path-message", |  | ||||||
|     "requirements": [ |  | ||||||
|         "pyyaml" |  | ||||||
|     ] |  | ||||||
| } |  | ||||||
|  | @ -1,3 +0,0 @@ | ||||||
| GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| SPDX-FileCopyrightText: Ansible Project |  | ||||||
|  | @ -1,134 +0,0 @@ | ||||||
| #!/usr/bin/env python |  | ||||||
| # Copyright (c) 2024, Felix Fontein <felix@fontein.de> |  | ||||||
| # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| """Make sure all modules that should show up in the action group.""" |  | ||||||
| 
 |  | ||||||
| from __future__ import annotations |  | ||||||
| 
 |  | ||||||
| import os |  | ||||||
| import re |  | ||||||
| import yaml |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| ACTION_GROUPS = { |  | ||||||
|     # The format is as follows: |  | ||||||
|     # * 'pattern': a regular expression matching all module names potentially belonging to the action group; |  | ||||||
|     # * 'exclusions': a list of modules that are not part of the action group; all other modules matching 'pattern' must be part of it; |  | ||||||
|     # * 'doc_fragment': the docs fragment that documents membership of the action group. |  | ||||||
|     'consul': { |  | ||||||
|         'pattern': re.compile('^consul_.*$'), |  | ||||||
|         'exclusions': [ |  | ||||||
|             'consul_acl_bootstrap', |  | ||||||
|             'consul_kv', |  | ||||||
|         ], |  | ||||||
|         'doc_fragment': 'community.general.consul.actiongroup_consul', |  | ||||||
|     }, |  | ||||||
|     'keycloak': { |  | ||||||
|         'pattern': re.compile('^keycloak_.*$'), |  | ||||||
|         'exclusions': [ |  | ||||||
|             'keycloak_realm_info', |  | ||||||
|         ], |  | ||||||
|         'doc_fragment': 'community.general.keycloak.actiongroup_keycloak', |  | ||||||
|     }, |  | ||||||
|     'proxmox': { |  | ||||||
|         'pattern': re.compile('^proxmox(_.*)?$'), |  | ||||||
|         'exclusions': [], |  | ||||||
|         'doc_fragment': 'community.general.proxmox.actiongroup_proxmox', |  | ||||||
|     }, |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| def main(): |  | ||||||
|     """Main entry point.""" |  | ||||||
| 
 |  | ||||||
|     # Load redirects |  | ||||||
|     meta_runtime = 'meta/runtime.yml' |  | ||||||
|     self_path = 'tests/sanity/extra/action-group.py' |  | ||||||
|     try: |  | ||||||
|         with open(meta_runtime, 'rb') as f: |  | ||||||
|             data = yaml.safe_load(f) |  | ||||||
|         action_groups = data['action_groups'] |  | ||||||
|     except Exception as exc: |  | ||||||
|         print(f'{meta_runtime}: cannot load action groups: {exc}') |  | ||||||
|         return |  | ||||||
| 
 |  | ||||||
|     for action_group in action_groups: |  | ||||||
|         if action_group not in ACTION_GROUPS: |  | ||||||
|             print(f'{meta_runtime}: found unknown action group {action_group!r}; likely {self_path} needs updating') |  | ||||||
|     for action_group, action_group_data in list(ACTION_GROUPS.items()): |  | ||||||
|         if action_group not in action_groups: |  | ||||||
|             print(f'{meta_runtime}: cannot find action group {action_group!r}; likely {self_path} needs updating') |  | ||||||
| 
 |  | ||||||
|     modules_directory = 'plugins/modules/' |  | ||||||
|     modules_suffix = '.py' |  | ||||||
| 
 |  | ||||||
|     for file in os.listdir(modules_directory): |  | ||||||
|         if not file.endswith(modules_suffix): |  | ||||||
|             continue |  | ||||||
|         module_name = file[:-len(modules_suffix)] |  | ||||||
| 
 |  | ||||||
|         for action_group, action_group_data in ACTION_GROUPS.items(): |  | ||||||
|             action_group_content = action_groups.get(action_group) or [] |  | ||||||
|             path = os.path.join(modules_directory, file) |  | ||||||
| 
 |  | ||||||
|             if not action_group_data['pattern'].match(module_name): |  | ||||||
|                 if module_name in action_group_content: |  | ||||||
|                     print(f'{path}: module is in action group {action_group!r} despite not matching its pattern as defined in {self_path}') |  | ||||||
|                 continue |  | ||||||
| 
 |  | ||||||
|             should_be_in_action_group = module_name not in action_group_data['exclusions'] |  | ||||||
| 
 |  | ||||||
|             if should_be_in_action_group: |  | ||||||
|                 if module_name not in action_group_content: |  | ||||||
|                     print(f'{meta_runtime}: module {module_name!r} is not part of {action_group!r} action group') |  | ||||||
|                 else: |  | ||||||
|                     action_group_content.remove(module_name) |  | ||||||
| 
 |  | ||||||
|             documentation = [] |  | ||||||
|             in_docs = False |  | ||||||
|             with open(path, 'r', encoding='utf-8') as f: |  | ||||||
|                 for line in f: |  | ||||||
|                     if line.startswith('DOCUMENTATION ='): |  | ||||||
|                         in_docs = True |  | ||||||
|                     elif line.startswith(("'''", '"""')) and in_docs: |  | ||||||
|                         in_docs = False |  | ||||||
|                     elif in_docs: |  | ||||||
|                         documentation.append(line) |  | ||||||
|             if in_docs: |  | ||||||
|                 print(f'{path}: cannot find DOCUMENTATION end') |  | ||||||
|             if not documentation: |  | ||||||
|                 print(f'{path}: cannot find DOCUMENTATION') |  | ||||||
|                 continue |  | ||||||
| 
 |  | ||||||
|             try: |  | ||||||
|                 docs = yaml.safe_load('\n'.join(documentation)) |  | ||||||
|                 if not isinstance(docs, dict): |  | ||||||
|                     raise Exception('is not a top-level dictionary') |  | ||||||
|             except Exception as exc: |  | ||||||
|                 print(f'{path}: cannot load DOCUMENTATION as YAML: {exc}') |  | ||||||
|                 continue |  | ||||||
| 
 |  | ||||||
|             docs_fragments = docs.get('extends_documentation_fragment') or [] |  | ||||||
|             is_in_action_group = action_group_data['doc_fragment'] in docs_fragments |  | ||||||
| 
 |  | ||||||
|             if should_be_in_action_group != is_in_action_group: |  | ||||||
|                 if should_be_in_action_group: |  | ||||||
|                     print( |  | ||||||
|                         f'{path}: module does not document itself as part of action group {action_group!r}, but it should;' |  | ||||||
|                         f' you need to add {action_group_data["doc_fragment"]} to "extends_documentation_fragment" in DOCUMENTATION' |  | ||||||
|                     ) |  | ||||||
|                 else: |  | ||||||
|                     print(f'{path}: module documents itself as part of action group {action_group!r}, but it should not be') |  | ||||||
| 
 |  | ||||||
|     for action_group, action_group_data in ACTION_GROUPS.items(): |  | ||||||
|         action_group_content = action_groups.get(action_group) or [] |  | ||||||
|         for module_name in action_group_content: |  | ||||||
|             print( |  | ||||||
|                 f'{meta_runtime}: module {module_name} mentioned in {action_group!r} action group' |  | ||||||
|                 f' does not exist or does not match pattern defined in {self_path}' |  | ||||||
|             ) |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| if __name__ == '__main__': |  | ||||||
|     main() |  | ||||||
|  | @ -1,11 +0,0 @@ | ||||||
| { |  | ||||||
|     "include_symlinks": false, |  | ||||||
|     "prefixes": [ |  | ||||||
|         ".azure-pipelines/azure-pipelines.yml", |  | ||||||
|         "tests/integration/targets/" |  | ||||||
|     ], |  | ||||||
|     "output": "path-message", |  | ||||||
|     "requirements": [ |  | ||||||
|         "PyYAML" |  | ||||||
|     ] |  | ||||||
| } |  | ||||||
|  | @ -1,3 +0,0 @@ | ||||||
| GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| SPDX-FileCopyrightText: Ansible Project |  | ||||||
							
								
								
									
										13
									
								
								tests/sanity/extra/aliases.py
									
										
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						
									
										13
									
								
								tests/sanity/extra/aliases.py
									
										
									
									
									
										
										
										Executable file → Normal file
									
								
							|  | @ -6,6 +6,7 @@ | ||||||
| from __future__ import (absolute_import, division, print_function) | from __future__ import (absolute_import, division, print_function) | ||||||
| __metaclass__ = type | __metaclass__ = type | ||||||
| 
 | 
 | ||||||
|  | import glob | ||||||
| import sys | import sys | ||||||
| 
 | 
 | ||||||
| import yaml | import yaml | ||||||
|  | @ -13,9 +14,6 @@ import yaml | ||||||
| 
 | 
 | ||||||
| def main(): | def main(): | ||||||
|     """Main entry point.""" |     """Main entry point.""" | ||||||
|     paths = sys.argv[1:] or sys.stdin.read().splitlines() |  | ||||||
|     paths = [path for path in paths if path.endswith('/aliases')] |  | ||||||
| 
 |  | ||||||
|     with open('.azure-pipelines/azure-pipelines.yml', 'rb') as f: |     with open('.azure-pipelines/azure-pipelines.yml', 'rb') as f: | ||||||
|         azp = yaml.safe_load(f) |         azp = yaml.safe_load(f) | ||||||
| 
 | 
 | ||||||
|  | @ -27,6 +25,9 @@ def main(): | ||||||
|             for group in job['parameters']['groups']: |             for group in job['parameters']['groups']: | ||||||
|                 allowed_targets.add('azp/posix/{0}'.format(group)) |                 allowed_targets.add('azp/posix/{0}'.format(group)) | ||||||
| 
 | 
 | ||||||
|  |     paths = glob.glob("tests/integration/targets/*/aliases") | ||||||
|  | 
 | ||||||
|  |     has_errors = False | ||||||
|     for path in paths: |     for path in paths: | ||||||
|         targets = [] |         targets = [] | ||||||
|         skip = False |         skip = False | ||||||
|  | @ -56,10 +57,14 @@ def main(): | ||||||
|             if 'targets/setup_' in path: |             if 'targets/setup_' in path: | ||||||
|                 continue |                 continue | ||||||
|             print('%s: %s' % (path, 'found no targets')) |             print('%s: %s' % (path, 'found no targets')) | ||||||
|  |             has_errors = True | ||||||
|         for target in targets: |         for target in targets: | ||||||
|             if target not in allowed_targets: |             if target not in allowed_targets: | ||||||
|                 print('%s: %s' % (path, 'found invalid target "{0}"'.format(target))) |                 print('%s: %s' % (path, 'found invalid target "{0}"'.format(target))) | ||||||
|  |                 has_errors = True | ||||||
|  | 
 | ||||||
|  |     return 1 if has_errors else 0 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     main() |     sys.exit(main()) | ||||||
|  |  | ||||||
|  | @ -1,8 +0,0 @@ | ||||||
| { |  | ||||||
|     "include_symlinks": false, |  | ||||||
|     "output": "path-line-column-message", |  | ||||||
|     "requirements": [ |  | ||||||
|         "PyYAML", |  | ||||||
|         "voluptuous==0.12.1" |  | ||||||
|     ] |  | ||||||
| } |  | ||||||
|  | @ -1,3 +0,0 @@ | ||||||
| GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| SPDX-FileCopyrightText: Ansible Project |  | ||||||
							
								
								
									
										76
									
								
								tests/sanity/extra/botmeta.py
									
										
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						
									
										76
									
								
								tests/sanity/extra/botmeta.py
									
										
									
									
									
										
										
										Executable file → Normal file
									
								
							|  | @ -54,14 +54,18 @@ IGNORE_NO_MAINTAINERS = [ | ||||||
|     'plugins/filter/random_mac.py', |     'plugins/filter/random_mac.py', | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
| FILENAME = '.github/BOTMETA.yml' |  | ||||||
| 
 | 
 | ||||||
| LIST_ENTRIES = frozenset(('supershipit', 'maintainers', 'labels', 'keywords', 'notify', 'ignore')) | class BotmetaCheck: | ||||||
|  |     def __init__(self): | ||||||
|  |         self.errors: list[str] = [] | ||||||
|  |         self.botmeta_filename = '.github/BOTMETA.yml' | ||||||
|  |         self.list_entries = frozenset(('supershipit', 'maintainers', 'labels', 'keywords', 'notify', 'ignore')) | ||||||
|  |         self.author_regex = re.compile(r'^\w.*\(@([\w-]+)\)(?![\w.])') | ||||||
| 
 | 
 | ||||||
| AUTHOR_REGEX = re.compile(r'^\w.*\(@([\w-]+)\)(?![\w.])') |     def report_error(self, error: str) -> None: | ||||||
|  |         self.errors.append(error) | ||||||
| 
 | 
 | ||||||
| 
 |     def read_authors(self, filename: str) -> list[str]: | ||||||
| def read_authors(filename): |  | ||||||
|         data = {} |         data = {} | ||||||
|         try: |         try: | ||||||
|             documentation = [] |             documentation = [] | ||||||
|  | @ -75,16 +79,16 @@ def read_authors(filename): | ||||||
|                     elif in_docs: |                     elif in_docs: | ||||||
|                         documentation.append(line) |                         documentation.append(line) | ||||||
|             if in_docs: |             if in_docs: | ||||||
|             print(f'{filename}: cannot find DOCUMENTATION end') |                 self.report_error(f'{filename}: cannot find DOCUMENTATION end') | ||||||
|                 return [] |                 return [] | ||||||
|             if not documentation: |             if not documentation: | ||||||
|             print(f'{filename}: cannot find DOCUMENTATION') |                 self.report_error(f'{filename}: cannot find DOCUMENTATION') | ||||||
|                 return [] |                 return [] | ||||||
| 
 | 
 | ||||||
|             data = yaml.safe_load('\n'.join(documentation)) |             data = yaml.safe_load('\n'.join(documentation)) | ||||||
| 
 | 
 | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|         print(f'{filename}:0:0: Cannot load DOCUMENTATION: {e}') |             self.report_error(f'{filename}:0:0: Cannot load DOCUMENTATION: {e}') | ||||||
|             return [] |             return [] | ||||||
| 
 | 
 | ||||||
|         author = data.get('author') or [] |         author = data.get('author') or [] | ||||||
|  | @ -92,17 +96,15 @@ def read_authors(filename): | ||||||
|             author = [author] |             author = [author] | ||||||
|         return author |         return author | ||||||
| 
 | 
 | ||||||
| 
 |     def extract_author_name(self, author: str) -> str | None: | ||||||
| def extract_author_name(author): |         m = self.author_regex.match(author) | ||||||
|     m = AUTHOR_REGEX.match(author) |  | ||||||
|         if m: |         if m: | ||||||
|             return m.group(1) |             return m.group(1) | ||||||
|         if author == 'Ansible Core Team': |         if author == 'Ansible Core Team': | ||||||
|             return '$team_ansible_core' |             return '$team_ansible_core' | ||||||
|         return None |         return None | ||||||
| 
 | 
 | ||||||
| 
 |     def validate(self, filename: str, filedata: dict) -> None: | ||||||
| def validate(filename, filedata): |  | ||||||
|         if not filename.startswith('plugins/'): |         if not filename.startswith('plugins/'): | ||||||
|             return |             return | ||||||
|         if filename.startswith(('plugins/doc_fragments/', 'plugins/module_utils/')): |         if filename.startswith(('plugins/doc_fragments/', 'plugins/module_utils/')): | ||||||
|  | @ -110,32 +112,31 @@ def validate(filename, filedata): | ||||||
|         # Compile list of all active and inactive maintainers |         # Compile list of all active and inactive maintainers | ||||||
|         all_maintainers = filedata['maintainers'] + filedata['ignore'] |         all_maintainers = filedata['maintainers'] + filedata['ignore'] | ||||||
|         if not filename.startswith(('plugins/action/', 'plugins/doc_fragments/', 'plugins/filter/', 'plugins/module_utils/', 'plugins/plugin_utils/')): |         if not filename.startswith(('plugins/action/', 'plugins/doc_fragments/', 'plugins/filter/', 'plugins/module_utils/', 'plugins/plugin_utils/')): | ||||||
|         maintainers = read_authors(filename) |             maintainers = self.read_authors(filename) | ||||||
|             for maintainer in maintainers: |             for maintainer in maintainers: | ||||||
|             maintainer = extract_author_name(maintainer) |                 maintainer = self.extract_author_name(maintainer) | ||||||
|                 if maintainer is not None and maintainer not in all_maintainers: |                 if maintainer is not None and maintainer not in all_maintainers: | ||||||
|                     others = ', '.join(all_maintainers) |                     others = ', '.join(all_maintainers) | ||||||
|                     msg = f'Author {maintainer} not mentioned as active or inactive maintainer for {filename} (mentioned are: {others})' |                     msg = f'Author {maintainer} not mentioned as active or inactive maintainer for {filename} (mentioned are: {others})' | ||||||
|                 print(f'{FILENAME}:0:0: {msg}') |                     self.report_error(f'{self.botmeta_filename}:0:0: {msg}') | ||||||
|         should_have_no_maintainer = filename in IGNORE_NO_MAINTAINERS |         should_have_no_maintainer = filename in IGNORE_NO_MAINTAINERS | ||||||
|         if not all_maintainers and not should_have_no_maintainer: |         if not all_maintainers and not should_have_no_maintainer: | ||||||
|         print(f'{FILENAME}:0:0: No (active or inactive) maintainer mentioned for {filename}') |             self.report_error(f'{self.botmeta_filename}:0:0: No (active or inactive) maintainer mentioned for {filename}') | ||||||
|         if all_maintainers and should_have_no_maintainer: |         if all_maintainers and should_have_no_maintainer: | ||||||
|         print(f'{FILENAME}:0:0: Please remove {filename} from the ignore list of {sys.argv[0]}') |             own_path = os.path.relpath(__file__, os.getcwd()) | ||||||
|  |             self.report_error(f'{self.botmeta_filename}:0:0: Please remove {filename} from the ignore list of {own_path}') | ||||||
| 
 | 
 | ||||||
| 
 |     def run(self) -> None: | ||||||
| def main(): |  | ||||||
|     """Main entry point.""" |  | ||||||
|         try: |         try: | ||||||
|         with open(FILENAME, 'rb') as f: |             with open(self.botmeta_filename, 'rb') as f: | ||||||
|                 botmeta = yaml.safe_load(f) |                 botmeta = yaml.safe_load(f) | ||||||
|         except yaml.error.MarkedYAMLError as ex: |         except yaml.error.MarkedYAMLError as ex: | ||||||
|             msg = re.sub(r'\s+', ' ', str(ex)) |             msg = re.sub(r'\s+', ' ', str(ex)) | ||||||
|         print('f{FILENAME}:{ex.context_mark.line + 1}:{ex.context_mark.column + 1}: YAML load failed: {msg}') |             self.report_error('f{self.botmeta_filename}:{ex.context_mark.line + 1}:{ex.context_mark.column + 1}: YAML load failed: {msg}') | ||||||
|             return |             return | ||||||
|         except Exception as ex:  # pylint: disable=broad-except |         except Exception as ex:  # pylint: disable=broad-except | ||||||
|             msg = re.sub(r'\s+', ' ', str(ex)) |             msg = re.sub(r'\s+', ' ', str(ex)) | ||||||
|         print(f'{FILENAME}:0:0: YAML load failed: {msg}') |             self.report_error(f'{self.botmeta_filename}:0:0: YAML load failed: {msg}') | ||||||
|             return |             return | ||||||
| 
 | 
 | ||||||
|         # Validate schema |         # Validate schema | ||||||
|  | @ -168,7 +169,7 @@ def main(): | ||||||
|         except MultipleInvalid as ex: |         except MultipleInvalid as ex: | ||||||
|             for error in ex.errors: |             for error in ex.errors: | ||||||
|                 # No way to get line/column numbers |                 # No way to get line/column numbers | ||||||
|             print(f'{FILENAME}:0:0: {humanize_error(botmeta, error)}') |                 self.report_error(f'{self.botmeta_filename}:0:0: {humanize_error(botmeta, error)}') | ||||||
|             return |             return | ||||||
| 
 | 
 | ||||||
|         # Preprocess (substitute macros, convert to lists) |         # Preprocess (substitute macros, convert to lists) | ||||||
|  | @ -192,10 +193,10 @@ def main(): | ||||||
|                 filedata = {k: convert_macros(v, macros) for k, v in filedata.items()} |                 filedata = {k: convert_macros(v, macros) for k, v in filedata.items()} | ||||||
|                 files[file] = filedata |                 files[file] = filedata | ||||||
|                 for k, v in filedata.items(): |                 for k, v in filedata.items(): | ||||||
|                 if k in LIST_ENTRIES: |                     if k in self.list_entries: | ||||||
|                         filedata[k] = v.split() |                         filedata[k] = v.split() | ||||||
|         except KeyError as e: |         except KeyError as e: | ||||||
|         print(f'{FILENAME}:0:0: Found unknown macro {e}') |             self.report_error(f'{self.botmeta_filename}:0:0: Found unknown macro {e}') | ||||||
|             return |             return | ||||||
| 
 | 
 | ||||||
|         # Scan all files |         # Scan all files | ||||||
|  | @ -216,22 +217,31 @@ def main(): | ||||||
|                                 if file in unmatched: |                                 if file in unmatched: | ||||||
|                                     unmatched.remove(file) |                                     unmatched.remove(file) | ||||||
|                         if not matching_files: |                         if not matching_files: | ||||||
|                         print(f'{FILENAME}:0:0: Did not find any entry for {filename}') |                             self.report_error(f'{self.botmeta_filename}:0:0: Did not find any entry for {filename}') | ||||||
| 
 | 
 | ||||||
|                         matching_files.sort(key=lambda kv: kv[0]) |                         matching_files.sort(key=lambda kv: kv[0]) | ||||||
|                         filedata = {} |                         filedata = {} | ||||||
|                     for k in LIST_ENTRIES: |                         for k in self.list_entries: | ||||||
|                             filedata[k] = [] |                             filedata[k] = [] | ||||||
|                         for dummy, data in matching_files: |                         for dummy, data in matching_files: | ||||||
|                             for k, v in data.items(): |                             for k, v in data.items(): | ||||||
|                             if k in LIST_ENTRIES: |                                 if k in self.list_entries: | ||||||
|                                     v = filedata[k] + v |                                     v = filedata[k] + v | ||||||
|                                 filedata[k] = v |                                 filedata[k] = v | ||||||
|                     validate(filename, filedata) |                         self.validate(filename, filedata) | ||||||
| 
 | 
 | ||||||
|         for file in unmatched: |         for file in unmatched: | ||||||
|         print(f'{FILENAME}:0:0: Entry {file} was not used') |             self.report_error(f'{self.botmeta_filename}:0:0: Entry {file} was not used') | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def main() -> int: | ||||||
|  |     """Main entry point.""" | ||||||
|  |     check = BotmetaCheck() | ||||||
|  |     check.run() | ||||||
|  |     for error in sorted(check.errors): | ||||||
|  |         print(error) | ||||||
|  |     return 1 if check.errors else 0 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     main() |     sys.exit(main()) | ||||||
|  |  | ||||||
|  | @ -1,13 +0,0 @@ | ||||||
| { |  | ||||||
|     "include_symlinks": false, |  | ||||||
|     "prefixes": [ |  | ||||||
|         "docs/docsite/", |  | ||||||
|         "plugins/", |  | ||||||
|         "roles/" |  | ||||||
|     ], |  | ||||||
|     "output": "path-line-column-message", |  | ||||||
|     "requirements": [ |  | ||||||
|         "ansible-core", |  | ||||||
|         "antsibull-docs" |  | ||||||
|     ] |  | ||||||
| } |  | ||||||
|  | @ -1,3 +0,0 @@ | ||||||
| GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| SPDX-FileCopyrightText: Ansible Project |  | ||||||
|  | @ -1,29 +0,0 @@ | ||||||
| #!/usr/bin/env python |  | ||||||
| # Copyright (c) Ansible Project |  | ||||||
| # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| """Check extra collection docs with antsibull-docs.""" |  | ||||||
| from __future__ import (absolute_import, division, print_function) |  | ||||||
| __metaclass__ = type |  | ||||||
| 
 |  | ||||||
| import os |  | ||||||
| import sys |  | ||||||
| import subprocess |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| def main(): |  | ||||||
|     """Main entry point.""" |  | ||||||
|     env = os.environ.copy() |  | ||||||
|     suffix = ':{env}'.format(env=env["ANSIBLE_COLLECTIONS_PATH"]) if 'ANSIBLE_COLLECTIONS_PATH' in env else '' |  | ||||||
|     env['ANSIBLE_COLLECTIONS_PATH'] = '{root}{suffix}'.format(root=os.path.dirname(os.path.dirname(os.path.dirname(os.getcwd()))), suffix=suffix) |  | ||||||
|     p = subprocess.run( |  | ||||||
|         ['antsibull-docs', 'lint-collection-docs', '--plugin-docs', '--skip-rstcheck', '.'], |  | ||||||
|         env=env, |  | ||||||
|         check=False, |  | ||||||
|     ) |  | ||||||
|     if p.returncode not in (0, 3): |  | ||||||
|         print('{0}:0:0: unexpected return code {1}'.format(sys.argv[0], p.returncode)) |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| if __name__ == '__main__': |  | ||||||
|     main() |  | ||||||
|  | @ -1,4 +0,0 @@ | ||||||
| { |  | ||||||
|     "include_symlinks": false, |  | ||||||
|     "output": "path-message" |  | ||||||
| } |  | ||||||
|  | @ -1,3 +0,0 @@ | ||||||
| GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| SPDX-FileCopyrightText: Ansible Project |  | ||||||
|  | @ -1,110 +0,0 @@ | ||||||
| #!/usr/bin/env python |  | ||||||
| # Copyright (c) 2022, Felix Fontein <felix@fontein.de> |  | ||||||
| # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| """Prevent files without a correct license identifier from being added to the source tree.""" |  | ||||||
| from __future__ import (absolute_import, division, print_function) |  | ||||||
| __metaclass__ = type |  | ||||||
| 
 |  | ||||||
| import os |  | ||||||
| import glob |  | ||||||
| import sys |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| def format_license_list(licenses): |  | ||||||
|     if not licenses: |  | ||||||
|         return '(empty)' |  | ||||||
|     return ', '.join(['"%s"' % license for license in licenses]) |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| def find_licenses(filename, relax=False): |  | ||||||
|     spdx_license_identifiers = [] |  | ||||||
|     other_license_identifiers = [] |  | ||||||
|     has_copyright = False |  | ||||||
|     try: |  | ||||||
|         with open(filename, 'r', encoding='utf-8') as f: |  | ||||||
|             for line in f: |  | ||||||
|                 line = line.rstrip() |  | ||||||
|                 if 'Copyright ' in line: |  | ||||||
|                     has_copyright = True |  | ||||||
|                 if 'Copyright: ' in line: |  | ||||||
|                     print('%s: found copyright line with "Copyright:". Please remove the colon.' % (filename, )) |  | ||||||
|                 if 'SPDX-FileCopyrightText: ' in line: |  | ||||||
|                     has_copyright = True |  | ||||||
|                 idx = line.find('SPDX-License-Identifier: ') |  | ||||||
|                 if idx >= 0: |  | ||||||
|                     lic_id = line[idx + len('SPDX-License-Identifier: '):] |  | ||||||
|                     spdx_license_identifiers.extend(lic_id.split(' OR ')) |  | ||||||
|                 if 'GNU General Public License' in line: |  | ||||||
|                     if 'v3.0+' in line: |  | ||||||
|                         other_license_identifiers.append('GPL-3.0-or-later') |  | ||||||
|                     if 'version 3 or later' in line: |  | ||||||
|                         other_license_identifiers.append('GPL-3.0-or-later') |  | ||||||
|                 if 'Simplified BSD License' in line: |  | ||||||
|                     other_license_identifiers.append('BSD-2-Clause') |  | ||||||
|                 if 'Apache License 2.0' in line: |  | ||||||
|                     other_license_identifiers.append('Apache-2.0') |  | ||||||
|                 if 'PSF License' in line or 'Python-2.0' in line: |  | ||||||
|                     other_license_identifiers.append('PSF-2.0') |  | ||||||
|                 if 'MIT License' in line: |  | ||||||
|                     other_license_identifiers.append('MIT') |  | ||||||
|     except Exception as exc: |  | ||||||
|         print('%s: error while processing file: %s' % (filename, exc)) |  | ||||||
|     if len(set(spdx_license_identifiers)) < len(spdx_license_identifiers): |  | ||||||
|         print('%s: found identical SPDX-License-Identifier values' % (filename, )) |  | ||||||
|     if other_license_identifiers and set(other_license_identifiers) != set(spdx_license_identifiers): |  | ||||||
|         print('%s: SPDX-License-Identifier yielded the license list %s, while manual guessing yielded the license list %s' % ( |  | ||||||
|             filename, format_license_list(spdx_license_identifiers), format_license_list(other_license_identifiers))) |  | ||||||
|     if not has_copyright and not relax: |  | ||||||
|         print('%s: found no copyright notice' % (filename, )) |  | ||||||
|     return sorted(spdx_license_identifiers) |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| def main(): |  | ||||||
|     """Main entry point.""" |  | ||||||
|     paths = sys.argv[1:] or sys.stdin.read().splitlines() |  | ||||||
| 
 |  | ||||||
|     # The following paths are allowed to have no license identifier |  | ||||||
|     no_comments_allowed = [ |  | ||||||
|         'changelogs/fragments/*.yml', |  | ||||||
|         'changelogs/fragments/*.yaml', |  | ||||||
|     ] |  | ||||||
| 
 |  | ||||||
|     # These files are completely ignored |  | ||||||
|     ignore_paths = [ |  | ||||||
|         '.ansible-test-timeout.json', |  | ||||||
|         '.reuse/dep5', |  | ||||||
|         'LICENSES/*.txt', |  | ||||||
|         'COPYING', |  | ||||||
|     ] |  | ||||||
| 
 |  | ||||||
|     no_comments_allowed = [fn for pattern in no_comments_allowed for fn in glob.glob(pattern)] |  | ||||||
|     ignore_paths = [fn for pattern in ignore_paths for fn in glob.glob(pattern)] |  | ||||||
| 
 |  | ||||||
|     valid_licenses = [license_file[len('LICENSES/'):-len('.txt')] for license_file in glob.glob('LICENSES/*.txt')] |  | ||||||
| 
 |  | ||||||
|     for path in paths: |  | ||||||
|         if path.startswith('./'): |  | ||||||
|             path = path[2:] |  | ||||||
|         if path in ignore_paths or path.startswith('tests/output/'): |  | ||||||
|             continue |  | ||||||
|         if os.stat(path).st_size == 0: |  | ||||||
|             continue |  | ||||||
|         if not path.endswith('.license') and os.path.exists(path + '.license'): |  | ||||||
|             path = path + '.license' |  | ||||||
|         valid_licenses_for_path = valid_licenses |  | ||||||
|         if path.startswith('plugins/') and not path.startswith(('plugins/modules/', 'plugins/module_utils/', 'plugins/doc_fragments/')): |  | ||||||
|             valid_licenses_for_path = [license for license in valid_licenses if license == 'GPL-3.0-or-later'] |  | ||||||
|         licenses = find_licenses(path, relax=path in no_comments_allowed) |  | ||||||
|         if not licenses: |  | ||||||
|             if path not in no_comments_allowed: |  | ||||||
|                 print('%s: must have at least one license' % (path, )) |  | ||||||
|         else: |  | ||||||
|             for license in licenses: |  | ||||||
|                 if license not in valid_licenses_for_path: |  | ||||||
|                     print('%s: found not allowed license "%s", must be one of %s' % ( |  | ||||||
|                         path, license, format_license_list(valid_licenses_for_path))) |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| if __name__ == '__main__': |  | ||||||
|     main() |  | ||||||
|  | @ -1,3 +0,0 @@ | ||||||
| GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| SPDX-FileCopyrightText: 2022, Felix Fontein <felix@fontein.de> |  | ||||||
|  | @ -1,7 +0,0 @@ | ||||||
| { |  | ||||||
|     "include_symlinks": true, |  | ||||||
|     "prefixes": [ |  | ||||||
|         "plugins/" |  | ||||||
|     ], |  | ||||||
|     "output": "path-message" |  | ||||||
| } |  | ||||||
|  | @ -1,3 +0,0 @@ | ||||||
| GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| SPDX-FileCopyrightText: Ansible Project |  | ||||||
|  | @ -1,58 +0,0 @@ | ||||||
| #!/usr/bin/env python |  | ||||||
| # Copyright (c) Ansible Project |  | ||||||
| # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| """Prevent unwanted files from being added to the source tree.""" |  | ||||||
| from __future__ import (absolute_import, division, print_function) |  | ||||||
| __metaclass__ = type |  | ||||||
| 
 |  | ||||||
| import os |  | ||||||
| import os.path |  | ||||||
| import sys |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| def main(): |  | ||||||
|     """Main entry point.""" |  | ||||||
|     paths = sys.argv[1:] or sys.stdin.read().splitlines() |  | ||||||
| 
 |  | ||||||
|     allowed_extensions = ( |  | ||||||
|         '.cs', |  | ||||||
|         '.ps1', |  | ||||||
|         '.psm1', |  | ||||||
|         '.py', |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     skip_paths = set([ |  | ||||||
|     ]) |  | ||||||
| 
 |  | ||||||
|     skip_directories = ( |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     yaml_directories = ( |  | ||||||
|         'plugins/test/', |  | ||||||
|         'plugins/filter/', |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     for path in paths: |  | ||||||
|         if path in skip_paths: |  | ||||||
|             continue |  | ||||||
| 
 |  | ||||||
|         if any(path.startswith(skip_directory) for skip_directory in skip_directories): |  | ||||||
|             continue |  | ||||||
| 
 |  | ||||||
|         if os.path.islink(path): |  | ||||||
|             print('%s: is a symbolic link' % (path, )) |  | ||||||
|         elif not os.path.isfile(path): |  | ||||||
|             print('%s: is not a regular file' % (path, )) |  | ||||||
| 
 |  | ||||||
|         ext = os.path.splitext(path)[1] |  | ||||||
| 
 |  | ||||||
|         if ext in ('.yml', ) and any(path.startswith(yaml_directory) for yaml_directory in yaml_directories): |  | ||||||
|             continue |  | ||||||
| 
 |  | ||||||
|         if ext not in allowed_extensions: |  | ||||||
|             print('%s: extension must be one of: %s' % (path, ', '.join(allowed_extensions))) |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| if __name__ == '__main__': |  | ||||||
|     main() |  | ||||||
|  | @ -16,11 +16,6 @@ else | ||||||
|     base_branch="" |     base_branch="" | ||||||
| fi | fi | ||||||
| 
 | 
 | ||||||
| if [ "${group}" == "extra" ]; then |  | ||||||
|     ../internal_test_tools/tools/run.py --color --bot --junit |  | ||||||
|     exit |  | ||||||
| fi |  | ||||||
| 
 |  | ||||||
| case "${group}" in | case "${group}" in | ||||||
|     1) options=(--skip-test pylint --skip-test ansible-doc --skip-test validate-modules) ;; |     1) options=(--skip-test pylint --skip-test ansible-doc --skip-test validate-modules) ;; | ||||||
|     2) options=(                   --test      ansible-doc      --test validate-modules) ;; |     2) options=(                   --test      ansible-doc      --test validate-modules) ;; | ||||||
|  | @ -28,17 +23,6 @@ case "${group}" in | ||||||
|     4) options=(--test pylint --exclude plugins/modules/) ;; |     4) options=(--test pylint --exclude plugins/modules/) ;; | ||||||
| esac | esac | ||||||
| 
 | 
 | ||||||
| # allow collection migration sanity tests for groups 3 and 4 to pass without updating this script during migration |  | ||||||
| network_path="lib/ansible/modules/network/" |  | ||||||
| 
 |  | ||||||
| if [ -d "${network_path}" ]; then |  | ||||||
|     if [ "${group}" -eq 3 ]; then |  | ||||||
|         options+=(--exclude "${network_path}") |  | ||||||
|     elif [ "${group}" -eq 4 ]; then |  | ||||||
|         options+=("${network_path}") |  | ||||||
|     fi |  | ||||||
| fi |  | ||||||
| 
 |  | ||||||
| # shellcheck disable=SC2086 | # shellcheck disable=SC2086 | ||||||
| ansible-test sanity --color -v --junit ${COVERAGE:+"$COVERAGE"} ${CHANGED:+"$CHANGED"} \ | ansible-test sanity --color -v --junit ${COVERAGE:+"$COVERAGE"} ${CHANGED:+"$CHANGED"} \ | ||||||
|     --docker --base-branch "${base_branch}" \ |     --docker --base-branch "${base_branch}" \ | ||||||
|  |  | ||||||
|  | @ -67,10 +67,6 @@ fi | ||||||
| 
 | 
 | ||||||
| export ANSIBLE_COLLECTIONS_PATHS="${PWD}/../../../" | export ANSIBLE_COLLECTIONS_PATHS="${PWD}/../../../" | ||||||
| 
 | 
 | ||||||
| if [ "${test}" == "sanity/extra" ]; then |  | ||||||
|     retry pip install junit-xml --disable-pip-version-check |  | ||||||
| fi |  | ||||||
| 
 |  | ||||||
| # START: HACK install dependencies | # START: HACK install dependencies | ||||||
| 
 | 
 | ||||||
| # Nothing further should be added to this list. | # Nothing further should be added to this list. | ||||||
|  | @ -79,7 +75,7 @@ retry git clone --depth=1 --single-branch https://github.com/ansible-collections | ||||||
| # NOTE: we're installing with git to work around Galaxy being a huge PITA (https://github.com/ansible/galaxy/issues/2429) | # NOTE: we're installing with git to work around Galaxy being a huge PITA (https://github.com/ansible/galaxy/issues/2429) | ||||||
| # retry ansible-galaxy -vvv collection install community.internal_test_tools | # retry ansible-galaxy -vvv collection install community.internal_test_tools | ||||||
| 
 | 
 | ||||||
| if [ "${script}" != "sanity" ] && [ "${script}" != "units" ] && [ "${test}" != "sanity/extra" ]; then | if [ "${script}" != "sanity" ] && [ "${script}" != "units" ]; then | ||||||
|     # To prevent Python dependencies on other collections only install other collections for integration tests |     # To prevent Python dependencies on other collections only install other collections for integration tests | ||||||
|     retry git clone --depth=1 --single-branch https://github.com/ansible-collections/ansible.posix.git "${ANSIBLE_COLLECTIONS_PATHS}/ansible_collections/ansible/posix" |     retry git clone --depth=1 --single-branch https://github.com/ansible-collections/ansible.posix.git "${ANSIBLE_COLLECTIONS_PATHS}/ansible_collections/ansible/posix" | ||||||
|     retry git clone --depth=1 --single-branch https://github.com/ansible-collections/community.crypto.git "${ANSIBLE_COLLECTIONS_PATHS}/ansible_collections/community/crypto" |     retry git clone --depth=1 --single-branch https://github.com/ansible-collections/community.crypto.git "${ANSIBLE_COLLECTIONS_PATHS}/ansible_collections/community/crypto" | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue