mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
Allow gitlab_runner to function for non-admin users (#1491)
* Allow gitlab_runner to function for non-admin users * Fix errant whitespace * Allow later python-gitlab versions * gitlab_runner: Switch to feature instead of bugfix, add examples * Fix version_added in gitlab_runner * Cleanup documentation for gitlab_runner fix * Update plugins/modules/source_control/gitlab/gitlab_runner.py Co-authored-by: Brendan Batliner <brendan.batliner@gmail.com> * Update plugins/modules/source_control/gitlab/gitlab_runner.py Co-authored-by: Felix Fontein <felix@fontein.de> * Revert docs change * Clarify docs * Clarify docs Co-authored-by: Brendan Batliner <brendan.batliner@gmail.com> Co-authored-by: Felix Fontein <felix@fontein.de>
This commit is contained in:
parent
298e0f60be
commit
11cb136971
5 changed files with 93 additions and 28 deletions
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
minor_changes:
|
||||||
|
- gitlab_runner - add ``owned`` option to allow non-admin use (https://github.com/ansible-collections/community.general/pull/1491).
|
||||||
|
bugfixes:
|
||||||
|
- gitlab_runner - fix compatiblity with some versions of python-gitlab (https://github.com/ansible-collections/community.general/pull/1491).
|
|
@ -57,6 +57,12 @@ options:
|
||||||
- The registration token is used to register new runners.
|
- The registration token is used to register new runners.
|
||||||
required: True
|
required: True
|
||||||
type: str
|
type: str
|
||||||
|
owned:
|
||||||
|
description:
|
||||||
|
- Searches only runners available to the user when searching for existing, when false admin token required.
|
||||||
|
default: no
|
||||||
|
type: bool
|
||||||
|
version_added: 2.0.0
|
||||||
active:
|
active:
|
||||||
description:
|
description:
|
||||||
- Define if the runners is immediately active after creation.
|
- Define if the runners is immediately active after creation.
|
||||||
|
@ -114,6 +120,14 @@ EXAMPLES = '''
|
||||||
api_token: "{{ access_token }}"
|
api_token: "{{ access_token }}"
|
||||||
description: Docker Machine t1
|
description: Docker Machine t1
|
||||||
state: absent
|
state: absent
|
||||||
|
|
||||||
|
- name: Delete an owned runner as a non-admin
|
||||||
|
community.general.gitlab_runner:
|
||||||
|
api_url: https://gitlab.example.com/
|
||||||
|
api_token: "{{ access_token }}"
|
||||||
|
description: Docker Machine t1
|
||||||
|
owned: yes
|
||||||
|
state: absent
|
||||||
'''
|
'''
|
||||||
|
|
||||||
RETURN = '''
|
RETURN = '''
|
||||||
|
@ -246,18 +260,28 @@ class GitLabRunner(object):
|
||||||
'''
|
'''
|
||||||
@param description Description of the runner
|
@param description Description of the runner
|
||||||
'''
|
'''
|
||||||
def findRunner(self, description):
|
def findRunner(self, description, owned=False):
|
||||||
runners = self._gitlab.runners.all(as_list=False)
|
if owned:
|
||||||
|
runners = self._gitlab.runners.list(as_list=False)
|
||||||
|
else:
|
||||||
|
runners = self._gitlab.runners.all(as_list=False)
|
||||||
|
|
||||||
for runner in runners:
|
for runner in runners:
|
||||||
if (runner['description'] == description):
|
# python-gitlab 2.2 through at least 2.5 returns a list of dicts for list() instead of a Runner
|
||||||
return self._gitlab.runners.get(runner['id'])
|
# object, so we need to handle both
|
||||||
|
if hasattr(runner, "description"):
|
||||||
|
if (runner.description == description):
|
||||||
|
return self._gitlab.runners.get(runner.id)
|
||||||
|
else:
|
||||||
|
if (runner['description'] == description):
|
||||||
|
return self._gitlab.runners.get(runner['id'])
|
||||||
|
|
||||||
'''
|
'''
|
||||||
@param description Description of the runner
|
@param description Description of the runner
|
||||||
'''
|
'''
|
||||||
def existsRunner(self, description):
|
def existsRunner(self, description, owned=False):
|
||||||
# When runner exists, object will be stored in self.runnerObject.
|
# When runner exists, object will be stored in self.runnerObject.
|
||||||
runner = self.findRunner(description)
|
runner = self.findRunner(description, owned)
|
||||||
|
|
||||||
if runner:
|
if runner:
|
||||||
self.runnerObject = runner
|
self.runnerObject = runner
|
||||||
|
@ -279,6 +303,7 @@ def main():
|
||||||
api_token=dict(type='str', no_log=True),
|
api_token=dict(type='str', no_log=True),
|
||||||
description=dict(type='str', required=True, aliases=["name"]),
|
description=dict(type='str', required=True, aliases=["name"]),
|
||||||
active=dict(type='bool', default=True),
|
active=dict(type='bool', default=True),
|
||||||
|
owned=dict(type='bool', default=False),
|
||||||
tag_list=dict(type='list', default=[]),
|
tag_list=dict(type='list', default=[]),
|
||||||
run_untagged=dict(type='bool', default=True),
|
run_untagged=dict(type='bool', default=True),
|
||||||
locked=dict(type='bool', default=False),
|
locked=dict(type='bool', default=False),
|
||||||
|
@ -304,6 +329,7 @@ def main():
|
||||||
)
|
)
|
||||||
|
|
||||||
state = module.params['state']
|
state = module.params['state']
|
||||||
|
owned = module.params['owned']
|
||||||
runner_description = module.params['description']
|
runner_description = module.params['description']
|
||||||
runner_active = module.params['active']
|
runner_active = module.params['active']
|
||||||
tag_list = module.params['tag_list']
|
tag_list = module.params['tag_list']
|
||||||
|
@ -319,7 +345,7 @@ def main():
|
||||||
gitlab_instance = gitlabAuthentication(module)
|
gitlab_instance = gitlabAuthentication(module)
|
||||||
|
|
||||||
gitlab_runner = GitLabRunner(module, gitlab_instance)
|
gitlab_runner = GitLabRunner(module, gitlab_instance)
|
||||||
runner_exists = gitlab_runner.existsRunner(runner_description)
|
runner_exists = gitlab_runner.existsRunner(runner_description, owned)
|
||||||
|
|
||||||
if state == 'absent':
|
if state == 'absent':
|
||||||
if runner_exists:
|
if runner_exists:
|
||||||
|
|
|
@ -524,20 +524,8 @@ RUNNER API
|
||||||
'''
|
'''
|
||||||
|
|
||||||
|
|
||||||
@urlmatch(scheme="http", netloc="localhost", path="/api/v4/runners/all", method="get")
|
@urlmatch(scheme="http", netloc="localhost", path=r'/api/v4/runners/all$', method="get")
|
||||||
def resp_find_runners_all(url, request):
|
def resp_find_runners_all(url, request):
|
||||||
headers = {'content-type': 'application/json'}
|
|
||||||
content = ('[{"active": true,"description": "test-1-20150125","id": 1,'
|
|
||||||
'"is_shared": false,"ip_address": "127.0.0.1","name": null,'
|
|
||||||
'"online": true,"status": "online"},{"active": true,'
|
|
||||||
'"description": "test-2-20150125","id": 2,"ip_address": "127.0.0.1",'
|
|
||||||
'"is_shared": false,"name": null,"online": false,"status": "offline"}]')
|
|
||||||
content = content.encode("utf-8")
|
|
||||||
return response(200, content, headers, None, 5, request)
|
|
||||||
|
|
||||||
|
|
||||||
@urlmatch(scheme="http", netloc="localhost", path="/api/v4/runners", method="get")
|
|
||||||
def resp_find_runners_list(url, request):
|
|
||||||
headers = {'content-type': 'application/json',
|
headers = {'content-type': 'application/json',
|
||||||
"X-Page": 1,
|
"X-Page": 1,
|
||||||
"X-Next-Page": 2,
|
"X-Next-Page": 2,
|
||||||
|
@ -553,7 +541,41 @@ def resp_find_runners_list(url, request):
|
||||||
return response(200, content, headers, None, 5, request)
|
return response(200, content, headers, None, 5, request)
|
||||||
|
|
||||||
|
|
||||||
@urlmatch(scheme="http", netloc="localhost", path="/api/v4/runners/1", method="get")
|
@urlmatch(scheme="http", netloc="localhost", path=r'/api/v4/runners$', method="get")
|
||||||
|
def resp_find_runners_list(url, request):
|
||||||
|
headers = {'content-type': 'application/json',
|
||||||
|
"X-Page": 1,
|
||||||
|
"X-Next-Page": 2,
|
||||||
|
"X-Per-Page": 1,
|
||||||
|
"X-Total-Pages": 1,
|
||||||
|
"X-Total": 2}
|
||||||
|
content = ('[{"active": true,"description": "test-1-20201214","id": 1,'
|
||||||
|
'"is_shared": false,"ip_address": "127.0.0.1","name": null,'
|
||||||
|
'"online": true,"status": "online"},{"active": true,'
|
||||||
|
'"description": "test-2-20201214","id": 2,"ip_address": "127.0.0.1",'
|
||||||
|
'"is_shared": false,"name": null,"online": false,"status": "offline"}]')
|
||||||
|
content = content.encode("utf-8")
|
||||||
|
return response(200, content, headers, None, 5, request)
|
||||||
|
|
||||||
|
|
||||||
|
@urlmatch(scheme="http", netloc="localhost", path=r'/api/v4/runners/1$', method="put")
|
||||||
|
def resp_update_runner(url, request):
|
||||||
|
headers = {'content-type': 'application/json',
|
||||||
|
"X-Page": 1,
|
||||||
|
"X-Next-Page": 2,
|
||||||
|
"X-Per-Page": 1,
|
||||||
|
"X-Total-Pages": 1,
|
||||||
|
"X-Total": 2}
|
||||||
|
content = ('[{"active": true,"description": "test-1-20201214","id": 1,'
|
||||||
|
'"is_shared": false,"ip_address": "127.0.0.1","name": null,'
|
||||||
|
'"online": true,"status": "online"},{"active": true,'
|
||||||
|
'"description": "test-2-20201214","id": 2,"ip_address": "127.0.0.1",'
|
||||||
|
'"is_shared": false,"name": null,"online": false,"status": "offline"}]')
|
||||||
|
content = content.encode("utf-8")
|
||||||
|
return response(200, content, headers, None, 5, request)
|
||||||
|
|
||||||
|
|
||||||
|
@urlmatch(scheme="http", netloc="localhost", path=r'/api/v4/runners/1$', method="get")
|
||||||
def resp_get_runner(url, request):
|
def resp_get_runner(url, request):
|
||||||
headers = {'content-type': 'application/json'}
|
headers = {'content-type': 'application/json'}
|
||||||
content = ('{"active": true,"description": "test-1-20150125","id": 1,'
|
content = ('{"active": true,"description": "test-1-20150125","id": 1,'
|
||||||
|
@ -563,7 +585,7 @@ def resp_get_runner(url, request):
|
||||||
return response(200, content, headers, None, 5, request)
|
return response(200, content, headers, None, 5, request)
|
||||||
|
|
||||||
|
|
||||||
@urlmatch(scheme="http", netloc="localhost", path="/api/v4/runners", method="post")
|
@urlmatch(scheme="http", netloc="localhost", path=r'/api/v4/runners$', method="post")
|
||||||
def resp_create_runner(url, request):
|
def resp_create_runner(url, request):
|
||||||
headers = {'content-type': 'application/json'}
|
headers = {'content-type': 'application/json'}
|
||||||
content = ('{"active": true,"description": "test-1-20150125","id": 1,'
|
content = ('{"active": true,"description": "test-1-20150125","id": 1,'
|
||||||
|
@ -573,7 +595,7 @@ def resp_create_runner(url, request):
|
||||||
return response(201, content, headers, None, 5, request)
|
return response(201, content, headers, None, 5, request)
|
||||||
|
|
||||||
|
|
||||||
@urlmatch(scheme="http", netloc="localhost", path="/api/v4/runners/1", method="delete")
|
@urlmatch(scheme="http", netloc="localhost", path=r'/api/v4/runners/1$', method="delete")
|
||||||
def resp_delete_runner(url, request):
|
def resp_delete_runner(url, request):
|
||||||
headers = {'content-type': 'application/json'}
|
headers = {'content-type': 'application/json'}
|
||||||
content = ('{}')
|
content = ('{}')
|
||||||
|
|
|
@ -21,6 +21,7 @@ pytestmark = []
|
||||||
try:
|
try:
|
||||||
from .gitlab import (GitlabModuleTestCase,
|
from .gitlab import (GitlabModuleTestCase,
|
||||||
python_version_match_requirement,
|
python_version_match_requirement,
|
||||||
|
resp_find_runners_all,
|
||||||
resp_find_runners_list, resp_get_runner,
|
resp_find_runners_list, resp_get_runner,
|
||||||
resp_create_runner, resp_delete_runner)
|
resp_create_runner, resp_delete_runner)
|
||||||
|
|
||||||
|
@ -50,9 +51,9 @@ class TestGitlabRunner(GitlabModuleTestCase):
|
||||||
|
|
||||||
self.moduleUtil = GitLabRunner(module=self.mock_module, gitlab_instance=self.gitlab_instance)
|
self.moduleUtil = GitLabRunner(module=self.mock_module, gitlab_instance=self.gitlab_instance)
|
||||||
|
|
||||||
@with_httmock(resp_find_runners_list)
|
@with_httmock(resp_find_runners_all)
|
||||||
@with_httmock(resp_get_runner)
|
@with_httmock(resp_get_runner)
|
||||||
def test_runner_exist(self):
|
def test_runner_exist_all(self):
|
||||||
rvalue = self.moduleUtil.existsRunner("test-1-20150125")
|
rvalue = self.moduleUtil.existsRunner("test-1-20150125")
|
||||||
|
|
||||||
self.assertEqual(rvalue, True)
|
self.assertEqual(rvalue, True)
|
||||||
|
@ -61,6 +62,17 @@ class TestGitlabRunner(GitlabModuleTestCase):
|
||||||
|
|
||||||
self.assertEqual(rvalue, False)
|
self.assertEqual(rvalue, False)
|
||||||
|
|
||||||
|
@with_httmock(resp_find_runners_list)
|
||||||
|
@with_httmock(resp_get_runner)
|
||||||
|
def test_runner_exist_owned(self):
|
||||||
|
rvalue = self.moduleUtil.existsRunner("test-1-20201214", True)
|
||||||
|
|
||||||
|
self.assertEqual(rvalue, True)
|
||||||
|
|
||||||
|
rvalue = self.moduleUtil.existsRunner("test-3-00000000", True)
|
||||||
|
|
||||||
|
self.assertEqual(rvalue, False)
|
||||||
|
|
||||||
@with_httmock(resp_create_runner)
|
@with_httmock(resp_create_runner)
|
||||||
def test_create_runner(self):
|
def test_create_runner(self):
|
||||||
runner = self.moduleUtil.createRunner({"token": "token", "description": "test-1-20150125"})
|
runner = self.moduleUtil.createRunner({"token": "token", "description": "test-1-20150125"})
|
||||||
|
@ -68,7 +80,7 @@ class TestGitlabRunner(GitlabModuleTestCase):
|
||||||
self.assertEqual(type(runner), Runner)
|
self.assertEqual(type(runner), Runner)
|
||||||
self.assertEqual(runner.description, "test-1-20150125")
|
self.assertEqual(runner.description, "test-1-20150125")
|
||||||
|
|
||||||
@with_httmock(resp_find_runners_list)
|
@with_httmock(resp_find_runners_all)
|
||||||
@with_httmock(resp_get_runner)
|
@with_httmock(resp_get_runner)
|
||||||
def test_update_runner(self):
|
def test_update_runner(self):
|
||||||
runner = self.moduleUtil.findRunner("test-1-20150125")
|
runner = self.moduleUtil.findRunner("test-1-20150125")
|
||||||
|
@ -84,7 +96,7 @@ class TestGitlabRunner(GitlabModuleTestCase):
|
||||||
self.assertEqual(changed, False)
|
self.assertEqual(changed, False)
|
||||||
self.assertEqual(newRunner.description, "Runner description")
|
self.assertEqual(newRunner.description, "Runner description")
|
||||||
|
|
||||||
@with_httmock(resp_find_runners_list)
|
@with_httmock(resp_find_runners_all)
|
||||||
@with_httmock(resp_get_runner)
|
@with_httmock(resp_get_runner)
|
||||||
@with_httmock(resp_delete_runner)
|
@with_httmock(resp_delete_runner)
|
||||||
def test_delete_runner(self):
|
def test_delete_runner(self):
|
||||||
|
|
|
@ -12,7 +12,7 @@ linode-python # APIv3
|
||||||
linode_api4 ; python_version > '2.6' # APIv4
|
linode_api4 ; python_version > '2.6' # APIv4
|
||||||
|
|
||||||
# requirement for the gitlab module
|
# requirement for the gitlab module
|
||||||
python-gitlab < 2.3.0 # version 2.3.0 makes gitlab_runner tests fail
|
python-gitlab
|
||||||
httmock
|
httmock
|
||||||
|
|
||||||
# requirement for maven_artifact module
|
# requirement for maven_artifact module
|
||||||
|
|
Loading…
Reference in a new issue