mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
Bugfix: github_repo does not apply defaults on existing repos (#2386)
* github_repo do not apply defaults on currently existing repos * Fixed sanity * Fixed doc defaults * Added changelog * Fix "or" statement and some formatting * Improve description change check * Added api_url parameter for unit tests and default values for private and description parameters * Added force_defaults parameter * Improved docs * Fixed doc anchors for force_defaults parameter * Update plugins/modules/source_control/github/github_repo.py Co-authored-by: Felix Fontein <felix@fontein.de> Co-authored-by: Felix Fontein <felix@fontein.de>
This commit is contained in:
parent
08067f08df
commit
17c3708f31
3 changed files with 119 additions and 28 deletions
|
@ -0,0 +1,2 @@
|
||||||
|
bugfixes:
|
||||||
|
- github_repo - ``private`` and ``description`` attributes should not be set to default values when the repo already exists (https://github.com/ansible-collections/community.general/pull/2386).
|
|
@ -42,16 +42,18 @@ options:
|
||||||
description:
|
description:
|
||||||
description:
|
description:
|
||||||
- Description for the repository.
|
- Description for the repository.
|
||||||
|
- Defaults to empty if I(force_defaults=true), which is the default in this module.
|
||||||
|
- Defaults to empty if I(force_defaults=false) when creating a new repository.
|
||||||
- This is only used when I(state) is C(present).
|
- This is only used when I(state) is C(present).
|
||||||
type: str
|
type: str
|
||||||
default: ''
|
|
||||||
required: false
|
required: false
|
||||||
private:
|
private:
|
||||||
description:
|
description:
|
||||||
- Whether the new repository should be private or not.
|
- Whether the repository should be private or not.
|
||||||
|
- Defaults to C(false) if I(force_defaults=true), which is the default in this module.
|
||||||
|
- Defaults to C(false) if I(force_defaults=false) when creating a new repository.
|
||||||
- This is only used when I(state) is C(present).
|
- This is only used when I(state) is C(present).
|
||||||
type: bool
|
type: bool
|
||||||
default: no
|
|
||||||
required: false
|
required: false
|
||||||
state:
|
state:
|
||||||
description:
|
description:
|
||||||
|
@ -72,6 +74,14 @@ options:
|
||||||
type: str
|
type: str
|
||||||
default: 'https://api.github.com'
|
default: 'https://api.github.com'
|
||||||
version_added: "3.5.0"
|
version_added: "3.5.0"
|
||||||
|
force_defaults:
|
||||||
|
description:
|
||||||
|
- Overwrite current I(description) and I(private) attributes with defaults if set to C(true), which currently is the default.
|
||||||
|
- The default for this option will be deprecated in a future version of this collection, and eventually change to C(false).
|
||||||
|
type: bool
|
||||||
|
default: true
|
||||||
|
required: false
|
||||||
|
version_added: 4.1.0
|
||||||
requirements:
|
requirements:
|
||||||
- PyGithub>=1.54
|
- PyGithub>=1.54
|
||||||
notes:
|
notes:
|
||||||
|
@ -92,6 +102,7 @@ EXAMPLES = '''
|
||||||
description: "Just for fun"
|
description: "Just for fun"
|
||||||
private: yes
|
private: yes
|
||||||
state: present
|
state: present
|
||||||
|
force_defaults: no
|
||||||
register: result
|
register: result
|
||||||
|
|
||||||
- name: Delete the repository
|
- name: Delete the repository
|
||||||
|
@ -117,7 +128,7 @@ import sys
|
||||||
|
|
||||||
GITHUB_IMP_ERR = None
|
GITHUB_IMP_ERR = None
|
||||||
try:
|
try:
|
||||||
from github import Github, GithubException
|
from github import Github, GithubException, GithubObject
|
||||||
from github.GithubException import UnknownObjectException
|
from github.GithubException import UnknownObjectException
|
||||||
HAS_GITHUB_PACKAGE = True
|
HAS_GITHUB_PACKAGE = True
|
||||||
except Exception:
|
except Exception:
|
||||||
|
@ -135,7 +146,7 @@ def authenticate(username=None, password=None, access_token=None, api_url=None):
|
||||||
return Github(base_url=api_url, login_or_token=username, password=password)
|
return Github(base_url=api_url, login_or_token=username, password=password)
|
||||||
|
|
||||||
|
|
||||||
def create_repo(gh, name, organization=None, private=False, description='', check_mode=False):
|
def create_repo(gh, name, organization=None, private=None, description=None, check_mode=False):
|
||||||
result = dict(
|
result = dict(
|
||||||
changed=False,
|
changed=False,
|
||||||
repo=dict())
|
repo=dict())
|
||||||
|
@ -151,15 +162,20 @@ def create_repo(gh, name, organization=None, private=False, description='', chec
|
||||||
except UnknownObjectException:
|
except UnknownObjectException:
|
||||||
if not check_mode:
|
if not check_mode:
|
||||||
repo = target.create_repo(
|
repo = target.create_repo(
|
||||||
name=name, private=private, description=description)
|
name=name,
|
||||||
|
private=GithubObject.NotSet if private is None else private,
|
||||||
|
description=GithubObject.NotSet if description is None else description,
|
||||||
|
)
|
||||||
result['repo'] = repo.raw_data
|
result['repo'] = repo.raw_data
|
||||||
|
|
||||||
result['changed'] = True
|
result['changed'] = True
|
||||||
|
|
||||||
changes = {}
|
changes = {}
|
||||||
|
if private is not None:
|
||||||
if repo is None or repo.raw_data['private'] != private:
|
if repo is None or repo.raw_data['private'] != private:
|
||||||
changes['private'] = private
|
changes['private'] = private
|
||||||
if repo is None or repo.raw_data['description'] != description:
|
if description is not None:
|
||||||
|
if repo is None or repo.raw_data['description'] not in (description, description or None):
|
||||||
changes['description'] = description
|
changes['description'] = description
|
||||||
|
|
||||||
if changes:
|
if changes:
|
||||||
|
@ -193,6 +209,10 @@ def delete_repo(gh, name, organization=None, check_mode=False):
|
||||||
|
|
||||||
|
|
||||||
def run_module(params, check_mode=False):
|
def run_module(params, check_mode=False):
|
||||||
|
if params['force_defaults']:
|
||||||
|
params['description'] = params['description'] or ''
|
||||||
|
params['private'] = params['private'] or False
|
||||||
|
|
||||||
gh = authenticate(
|
gh = authenticate(
|
||||||
username=params['username'], password=params['password'], access_token=params['access_token'],
|
username=params['username'], password=params['password'], access_token=params['access_token'],
|
||||||
api_url=params['api_url'])
|
api_url=params['api_url'])
|
||||||
|
@ -216,17 +236,17 @@ def run_module(params, check_mode=False):
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
module_args = dict(
|
module_args = dict(
|
||||||
username=dict(type='str', required=False, default=None),
|
username=dict(type='str'),
|
||||||
password=dict(type='str', required=False, default=None, no_log=True),
|
password=dict(type='str', no_log=True),
|
||||||
access_token=dict(type='str', required=False,
|
access_token=dict(type='str', no_log=True),
|
||||||
default=None, no_log=True),
|
|
||||||
name=dict(type='str', required=True),
|
name=dict(type='str', required=True),
|
||||||
state=dict(type='str', required=False, default="present",
|
state=dict(type='str', required=False, default="present",
|
||||||
choices=["present", "absent"]),
|
choices=["present", "absent"]),
|
||||||
organization=dict(type='str', required=False, default=None),
|
organization=dict(type='str', required=False, default=None),
|
||||||
private=dict(type='bool', required=False, default=False),
|
private=dict(type='bool'),
|
||||||
description=dict(type='str', required=False, default=''),
|
description=dict(type='str'),
|
||||||
api_url=dict(type='str', required=False, default='https://api.github.com'),
|
api_url=dict(type='str', required=False, default='https://api.github.com'),
|
||||||
|
force_defaults=dict(type='bool', default=True),
|
||||||
)
|
)
|
||||||
module = AnsibleModule(
|
module = AnsibleModule(
|
||||||
argument_spec=module_args,
|
argument_spec=module_args,
|
||||||
|
|
|
@ -71,6 +71,28 @@ def get_repo_mock(url, request):
|
||||||
return response(200, content, headers, None, 5, request)
|
return response(200, content, headers, None, 5, request)
|
||||||
|
|
||||||
|
|
||||||
|
@urlmatch(netloc=r'api\.github\.com(:[0-9]+)?$', path=r'/repos/.*/.*', method="get")
|
||||||
|
def get_private_repo_mock(url, request):
|
||||||
|
match = re.search(
|
||||||
|
r"api\.github\.com(:[0-9]+)?/repos/(?P<org>[^/]+)/(?P<repo>[^/]+)", request.url)
|
||||||
|
org = match.group("org")
|
||||||
|
repo = match.group("repo")
|
||||||
|
|
||||||
|
# https://docs.github.com/en/rest/reference/repos#get-a-repository
|
||||||
|
headers = {'content-type': 'application/json'}
|
||||||
|
content = {
|
||||||
|
"name": repo,
|
||||||
|
"full_name": "{0}/{1}".format(org, repo),
|
||||||
|
"url": "https://api.github.com/repos/{0}/{1}".format(org, repo),
|
||||||
|
"private": True,
|
||||||
|
"description": "This your first repo!",
|
||||||
|
"default_branch": "master",
|
||||||
|
"allow_rebase_merge": True
|
||||||
|
}
|
||||||
|
content = json.dumps(content).encode("utf-8")
|
||||||
|
return response(200, content, headers, None, 5, request)
|
||||||
|
|
||||||
|
|
||||||
@urlmatch(netloc=r'api\.github\.com(:[0-9]+)?$', path=r'/orgs/.*/repos', method="post")
|
@urlmatch(netloc=r'api\.github\.com(:[0-9]+)?$', path=r'/orgs/.*/repos', method="post")
|
||||||
def create_new_org_repo_mock(url, request):
|
def create_new_org_repo_mock(url, request):
|
||||||
match = re.search(
|
match = re.search(
|
||||||
|
@ -83,8 +105,8 @@ def create_new_org_repo_mock(url, request):
|
||||||
content = {
|
content = {
|
||||||
"name": repo['name'],
|
"name": repo['name'],
|
||||||
"full_name": "{0}/{1}".format(org, repo['name']),
|
"full_name": "{0}/{1}".format(org, repo['name']),
|
||||||
"private": repo['private'],
|
"private": repo.get('private', False),
|
||||||
"description": repo['description']
|
"description": repo.get('description')
|
||||||
}
|
}
|
||||||
content = json.dumps(content).encode("utf-8")
|
content = json.dumps(content).encode("utf-8")
|
||||||
return response(201, content, headers, None, 5, request)
|
return response(201, content, headers, None, 5, request)
|
||||||
|
@ -99,8 +121,8 @@ def create_new_user_repo_mock(url, request):
|
||||||
content = {
|
content = {
|
||||||
"name": repo['name'],
|
"name": repo['name'],
|
||||||
"full_name": "{0}/{1}".format("octocat", repo['name']),
|
"full_name": "{0}/{1}".format("octocat", repo['name']),
|
||||||
"private": repo['private'],
|
"private": repo.get('private', False),
|
||||||
"description": repo['description']
|
"description": repo.get('description')
|
||||||
}
|
}
|
||||||
content = json.dumps(content).encode("utf-8")
|
content = json.dumps(content).encode("utf-8")
|
||||||
return response(201, content, headers, None, 5, request)
|
return response(201, content, headers, None, 5, request)
|
||||||
|
@ -120,8 +142,8 @@ def patch_repo_mock(url, request):
|
||||||
"name": repo,
|
"name": repo,
|
||||||
"full_name": "{0}/{1}".format(org, repo),
|
"full_name": "{0}/{1}".format(org, repo),
|
||||||
"url": "https://api.github.com/repos/{0}/{1}".format(org, repo),
|
"url": "https://api.github.com/repos/{0}/{1}".format(org, repo),
|
||||||
"private": body['private'],
|
"private": body.get('private', False),
|
||||||
"description": body['description'],
|
"description": body.get('description'),
|
||||||
"default_branch": "master",
|
"default_branch": "master",
|
||||||
"allow_rebase_merge": True
|
"allow_rebase_merge": True
|
||||||
}
|
}
|
||||||
|
@ -160,11 +182,34 @@ class TestGithubRepo(unittest.TestCase):
|
||||||
"description": "Just for fun",
|
"description": "Just for fun",
|
||||||
"private": False,
|
"private": False,
|
||||||
"state": "present",
|
"state": "present",
|
||||||
"api_url": "https://api.github.com"
|
"api_url": "https://api.github.com",
|
||||||
|
"force_defaults": False,
|
||||||
})
|
})
|
||||||
|
|
||||||
self.assertEqual(result['changed'], True)
|
self.assertEqual(result['changed'], True)
|
||||||
self.assertEqual(result['repo']['private'], False)
|
self.assertEqual(result['repo']['private'], False)
|
||||||
|
self.assertEqual(result['repo']['description'], 'Just for fun')
|
||||||
|
|
||||||
|
@with_httmock(get_orgs_mock)
|
||||||
|
@with_httmock(get_repo_notfound_mock)
|
||||||
|
@with_httmock(create_new_org_repo_mock)
|
||||||
|
def test_create_new_org_repo_incomplete(self):
|
||||||
|
result = github_repo.run_module({
|
||||||
|
'username': None,
|
||||||
|
'password': None,
|
||||||
|
"access_token": "mytoken",
|
||||||
|
"organization": "MyOrganization",
|
||||||
|
"name": "myrepo",
|
||||||
|
"description": None,
|
||||||
|
"private": None,
|
||||||
|
"state": "present",
|
||||||
|
"api_url": "https://api.github.com",
|
||||||
|
"force_defaults": False,
|
||||||
|
})
|
||||||
|
|
||||||
|
self.assertEqual(result['changed'], True)
|
||||||
|
self.assertEqual(result['repo']['private'], False)
|
||||||
|
self.assertEqual(result['repo']['description'], None)
|
||||||
|
|
||||||
@with_httmock(get_user_mock)
|
@with_httmock(get_user_mock)
|
||||||
@with_httmock(get_repo_notfound_mock)
|
@with_httmock(get_repo_notfound_mock)
|
||||||
|
@ -179,7 +224,8 @@ class TestGithubRepo(unittest.TestCase):
|
||||||
"description": "Just for fun",
|
"description": "Just for fun",
|
||||||
"private": True,
|
"private": True,
|
||||||
"state": "present",
|
"state": "present",
|
||||||
"api_url": "https://api.github.com"
|
"api_url": "https://api.github.com",
|
||||||
|
"force_defaults": False,
|
||||||
})
|
})
|
||||||
self.assertEqual(result['changed'], True)
|
self.assertEqual(result['changed'], True)
|
||||||
self.assertEqual(result['repo']['private'], True)
|
self.assertEqual(result['repo']['private'], True)
|
||||||
|
@ -197,11 +243,31 @@ class TestGithubRepo(unittest.TestCase):
|
||||||
"description": "Just for fun",
|
"description": "Just for fun",
|
||||||
"private": True,
|
"private": True,
|
||||||
"state": "present",
|
"state": "present",
|
||||||
"api_url": "https://api.github.com"
|
"api_url": "https://api.github.com",
|
||||||
|
"force_defaults": False,
|
||||||
})
|
})
|
||||||
self.assertEqual(result['changed'], True)
|
self.assertEqual(result['changed'], True)
|
||||||
self.assertEqual(result['repo']['private'], True)
|
self.assertEqual(result['repo']['private'], True)
|
||||||
|
|
||||||
|
@with_httmock(get_orgs_mock)
|
||||||
|
@with_httmock(get_private_repo_mock)
|
||||||
|
def test_idempotency_existing_org_private_repo(self):
|
||||||
|
result = github_repo.run_module({
|
||||||
|
'username': None,
|
||||||
|
'password': None,
|
||||||
|
"access_token": "mytoken",
|
||||||
|
"organization": "MyOrganization",
|
||||||
|
"name": "myrepo",
|
||||||
|
"description": None,
|
||||||
|
"private": None,
|
||||||
|
"state": "present",
|
||||||
|
"api_url": "https://api.github.com",
|
||||||
|
"force_defaults": False,
|
||||||
|
})
|
||||||
|
self.assertEqual(result['changed'], False)
|
||||||
|
self.assertEqual(result['repo']['private'], True)
|
||||||
|
self.assertEqual(result['repo']['description'], 'This your first repo!')
|
||||||
|
|
||||||
@with_httmock(get_orgs_mock)
|
@with_httmock(get_orgs_mock)
|
||||||
@with_httmock(get_repo_mock)
|
@with_httmock(get_repo_mock)
|
||||||
@with_httmock(delete_repo_mock)
|
@with_httmock(delete_repo_mock)
|
||||||
|
@ -215,7 +281,8 @@ class TestGithubRepo(unittest.TestCase):
|
||||||
"description": "Just for fun",
|
"description": "Just for fun",
|
||||||
"private": False,
|
"private": False,
|
||||||
"state": "absent",
|
"state": "absent",
|
||||||
"api_url": "https://api.github.com"
|
"api_url": "https://api.github.com",
|
||||||
|
"force_defaults": False,
|
||||||
})
|
})
|
||||||
self.assertEqual(result['changed'], True)
|
self.assertEqual(result['changed'], True)
|
||||||
|
|
||||||
|
@ -232,7 +299,8 @@ class TestGithubRepo(unittest.TestCase):
|
||||||
"description": "Just for fun",
|
"description": "Just for fun",
|
||||||
"private": False,
|
"private": False,
|
||||||
"state": "absent",
|
"state": "absent",
|
||||||
"api_url": "https://api.github.com"
|
"api_url": "https://api.github.com",
|
||||||
|
"force_defaults": False,
|
||||||
})
|
})
|
||||||
self.assertEqual(result['changed'], True)
|
self.assertEqual(result['changed'], True)
|
||||||
|
|
||||||
|
@ -249,7 +317,8 @@ class TestGithubRepo(unittest.TestCase):
|
||||||
"description": "Just for fun",
|
"description": "Just for fun",
|
||||||
"private": True,
|
"private": True,
|
||||||
"state": "absent",
|
"state": "absent",
|
||||||
"api_url": "https://api.github.com"
|
"api_url": "https://api.github.com",
|
||||||
|
"force_defaults": False,
|
||||||
})
|
})
|
||||||
self.assertEqual(result['changed'], False)
|
self.assertEqual(result['changed'], False)
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue