"""
Module for interacting with the GitHub v3 API.
.. versionadded:: 2016.3.0
:depends: PyGithub python module
Configuration
-------------
Configure this module by specifying the name of a configuration
profile in the minion config, minion pillar, or master config. The module
will use the 'github' key by default, if defined.
For example:
.. code-block:: yaml
github:
token: abc1234
org_name: my_organization
# optional: some functions require a repo_name, which
# can be set in the config file, or passed in at the CLI.
repo_name: my_repo
# optional: it can be dangerous to change the privacy of a repository
# in an automated way. set this to True to allow privacy modifications
allow_repo_privacy_changes: False
"""
import logging
import salt.utils.http
import salt.utils.json
from salt.exceptions import CommandExecutionError
HAS_LIBS = False
try:
# pylint: disable=no-name-in-module
import github
import github.NamedUser
import github.PaginatedList
from github.GithubException import UnknownObjectException
# pylint: enable=no-name-in-module
HAS_LIBS = True
except ImportError:
pass
log = logging.getLogger(__name__)
__virtualname__ = "github"
[docs]
def __virtual__():
"""
Only load this module if PyGithub is installed on this minion.
"""
if HAS_LIBS:
return __virtualname__
return (
False,
"The github execution module cannot be loaded: PyGithub library is not installed.",
)
def _get_config_value(profile, config_name):
"""
Helper function that returns a profile's configuration value based on
the supplied configuration name.
profile
The profile name that contains configuration information.
config_name
The configuration item's name to use to return configuration values.
"""
config = __salt__["config.option"](profile)
if not config:
raise CommandExecutionError(
f"Authentication information could not be found for the '{profile}' profile."
)
config_value = config.get(config_name)
if config_value is None:
raise CommandExecutionError(
f"The '{config_name}' parameter was not found in the '{profile}' profile."
)
return config_value
def _get_client(profile):
"""
Return the GitHub client, cached into __context__ for performance
"""
token = _get_config_value(profile, "token")
key = "github.{}:{}".format( # pylint: disable=consider-using-f-string
token, _get_config_value(profile, "org_name")
)
if key not in __context__:
__context__[key] = github.Github(token, per_page=100)
return __context__[key]
def _get_members(organization, params=None):
return github.PaginatedList.PaginatedList(
github.NamedUser.NamedUser,
organization._requester,
organization.url + "/members",
params,
)
def _get_repos(profile, params=None, ignore_cache=False):
# Use cache when no params are given
org_name = _get_config_value(profile, "org_name")
key = f"github.{org_name}:repos"
if key not in __context__ or ignore_cache or params is not None:
org_name = _get_config_value(profile, "org_name")
client = _get_client(profile)
organization = client.get_organization(org_name)
result = github.PaginatedList.PaginatedList(
github.Repository.Repository,
organization._requester,
organization.url + "/repos",
params,
)
# Only cache results if no params were given (full scan)
if params is not None:
return result
next_result = []
for repo in result:
next_result.append(repo)
# Cache a copy of each repo for single lookups
repo_key = f"github.{org_name}:{repo.name.lower()}:repo_info"
__context__[repo_key] = _repo_to_dict(repo)
__context__[key] = next_result
return __context__[key]
[docs]
def list_users(profile="github", ignore_cache=False):
"""
List all users within the organization.
profile
The name of the profile configuration to use. Defaults to ``github``.
ignore_cache
Bypasses the use of cached users.
.. versionadded:: 2016.11.0
CLI Example:
.. code-block:: bash
salt myminion github.list_users
salt myminion github.list_users profile='my-github-profile'
"""
org_name = _get_config_value(profile, "org_name")
key = f"github.{org_name}:users"
if key not in __context__ or ignore_cache:
client = _get_client(profile)
organization = client.get_organization(org_name)
__context__[key] = [member.login for member in _get_members(organization, None)]
return __context__[key]
[docs]
def get_user(name, profile="github", user_details=False):
"""
Get a GitHub user by name.
name
The user for which to obtain information.
profile
The name of the profile configuration to use. Defaults to ``github``.
user_details
Prints user information details. Defaults to ``False``. If the user is
already in the organization and user_details is set to False, the
get_user function returns ``True``. If the user is not already present
in the organization, user details will be printed by default.
CLI Example:
.. code-block:: bash
salt myminion github.get_user github-handle
salt myminion github.get_user github-handle user_details=true
"""
if not user_details and name in list_users(profile):
# User is in the org, no need for additional Data
return True
response = {}
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
try:
user = client.get_user(name)
except UnknownObjectException:
log.exception("Resource not found")
return False
response["company"] = user.company
response["created_at"] = user.created_at
response["email"] = user.email
response["html_url"] = user.html_url
response["id"] = user.id
response["login"] = user.login
response["name"] = user.name
response["type"] = user.type
response["url"] = user.url
try:
_, data = organization._requester.requestJsonAndCheck(
"GET", organization.url + "/memberships/" + user._identity
)
except UnknownObjectException:
response["membership_state"] = "nonexistent"
response["in_org"] = False
return response
response["in_org"] = organization.has_in_members(user)
response["membership_state"] = data.get("state")
return response
[docs]
def add_user(name, profile="github"):
"""
Add a GitHub user.
name
The user for which to obtain information.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.add_user github-handle
"""
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
try:
github_named_user = client.get_user(name)
except UnknownObjectException:
log.exception("Resource not found")
return False
_, data = organization._requester.requestJsonAndCheck(
"PUT", organization.url + "/memberships/" + github_named_user._identity
)
return data.get("state") == "pending"
[docs]
def remove_user(name, profile="github"):
"""
Remove a Github user by name.
name
The user for which to obtain information.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.remove_user github-handle
"""
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
try:
git_user = client.get_user(name)
except UnknownObjectException:
log.exception("Resource not found")
return False
if organization.has_in_members(git_user):
organization.remove_from_members(git_user)
return not organization.has_in_members(git_user)
[docs]
def get_issue(issue_number, repo_name=None, profile="github", output="min"):
"""
Return information about a single issue in a named repository.
.. versionadded:: 2016.11.0
issue_number
The number of the issue to retrieve.
repo_name
The name of the repository from which to get the issue. This argument is
required, either passed via the CLI, or defined in the configured
profile. A ``repo_name`` passed as a CLI argument will override the
repo_name defined in the configured profile, if provided.
profile
The name of the profile configuration to use. Defaults to ``github``.
output
The amount of data returned by each issue. Defaults to ``min``. Change
to ``full`` to see all issue output.
CLI Example:
.. code-block:: bash
salt myminion github.get_issue 514
salt myminion github.get_issue 514 repo_name=salt
"""
org_name = _get_config_value(profile, "org_name")
if repo_name is None:
repo_name = _get_config_value(profile, "repo_name")
action = "/".join(["repos", org_name, repo_name])
command = "issues/" + str(issue_number)
ret = {}
issue_data = _query(profile, action=action, command=command)
issue_id = issue_data.get("dict", {}).get("id")
if issue_id:
if output.lower() == "full":
ret[issue_id] = issue_data["dict"]
else:
ret[issue_id] = _format_issue(issue_data["dict"])
return ret
[docs]
def get_issues(
repo_name=None,
profile="github",
milestone=None,
state="open",
assignee=None,
creator=None,
mentioned=None,
labels=None,
sort="created",
direction="desc",
since=None,
output="min",
per_page=None,
):
"""
Returns information for all issues in a given repository, based on the search options.
.. versionadded:: 2016.11.0
repo_name
The name of the repository for which to list issues. This argument is
required, either passed via the CLI, or defined in the configured
profile. A ``repo_name`` passed as a CLI argument will override the
repo_name defined in the configured profile, if provided.
profile
The name of the profile configuration to use. Defaults to ``github``.
milestone
The number of a GitHub milestone, or a string of either ``*`` or
``none``.
If a number is passed, it should refer to a milestone by its number
field. Use the ``github.get_milestone`` function to obtain a milestone's
number.
If the string ``*`` is passed, issues with any milestone are
accepted. If the string ``none`` is passed, issues without milestones
are returned.
state
Indicates the state of the issues to return. Can be either ``open``,
``closed``, or ``all``. Default is ``open``.
assignee
Can be the name of a user. Pass in ``none`` (as a string) for issues
with no assigned user or ``*`` for issues assigned to any user.
creator
The user that created the issue.
mentioned
A user that's mentioned in the issue.
labels
A string of comma separated label names. For example, ``bug,ui,@high``.
sort
What to sort results by. Can be either ``created``, ``updated``, or
``comments``. Default is ``created``.
direction
The direction of the sort. Can be either ``asc`` or ``desc``. Default
is ``desc``.
since
Only issues updated at or after this time are returned. This is a
timestamp in ISO 8601 format: ``YYYY-MM-DDTHH:MM:SSZ``.
output
The amount of data returned by each issue. Defaults to ``min``. Change
to ``full`` to see all issue output.
per_page
GitHub paginates data in their API calls. Use this value to increase or
decrease the number of issues gathered from GitHub, per page. If not set,
GitHub defaults are used. Maximum is 100.
CLI Example:
.. code-block:: bash
salt myminion github.get_issues my-github-repo
"""
org_name = _get_config_value(profile, "org_name")
if repo_name is None:
repo_name = _get_config_value(profile, "repo_name")
action = "/".join(["repos", org_name, repo_name])
args = {}
# Build API arguments, as necessary.
if milestone:
args["milestone"] = milestone
if assignee:
args["assignee"] = assignee
if creator:
args["creator"] = creator
if mentioned:
args["mentioned"] = mentioned
if labels:
args["labels"] = labels
if since:
args["since"] = since
if per_page:
args["per_page"] = per_page
# Only pass the following API args if they're not the defaults listed.
if state and state != "open":
args["state"] = state
if sort and sort != "created":
args["sort"] = sort
if direction and direction != "desc":
args["direction"] = direction
ret = {}
issues = _query(profile, action=action, command="issues", args=args)
for issue in issues.get("dict", {}):
# Pull requests are included in the issue list from GitHub
# Let's not include those in the return.
if issue.get("pull_request"):
continue
issue_id = issue.get("id")
if output == "full":
ret[issue_id] = issue
else:
ret[issue_id] = _format_issue(issue)
return ret
[docs]
def get_milestones(
repo_name=None,
profile="github",
state="open",
sort="due_on",
direction="asc",
output="min",
per_page=None,
):
"""
Return information about milestones for a given repository.
.. versionadded:: 2016.11.0
repo_name
The name of the repository for which to list issues. This argument is
required, either passed via the CLI, or defined in the configured
profile. A ``repo_name`` passed as a CLI argument will override the
repo_name defined in the configured profile, if provided.
profile
The name of the profile configuration to use. Defaults to ``github``.
state
The state of the milestone. Either ``open``, ``closed``, or ``all``.
Default is ``open``.
sort
What to sort results by. Either ``due_on`` or ``completeness``. Default
is ``due_on``.
direction
The direction of the sort. Either ``asc`` or ``desc``. Default is ``asc``.
output
The amount of data returned by each issue. Defaults to ``min``. Change
to ``full`` to see all issue output.
per_page
GitHub paginates data in their API calls. Use this value to increase or
decrease the number of issues gathered from GitHub, per page. If not set,
GitHub defaults are used.
CLI Example:
.. code-block:: bash
salt myminion github.get_milestones
"""
org_name = _get_config_value(profile, "org_name")
if repo_name is None:
repo_name = _get_config_value(profile, "repo_name")
action = "/".join(["repos", org_name, repo_name])
args = {}
if per_page:
args["per_page"] = per_page
# Only pass the following API args if they're not the defaults listed.
if state and state != "open":
args["state"] = state
if sort and sort != "due_on":
args["sort"] = sort
if direction and direction != "asc":
args["direction"] = direction
ret = {}
milestones = _query(profile, action=action, command="milestones", args=args)
for milestone in milestones.get("dict", {}):
milestone_id = milestone.get("id")
if output == "full":
ret[milestone_id] = milestone
else:
milestone.pop("creator")
milestone.pop("html_url")
milestone.pop("labels_url")
ret[milestone_id] = milestone
return ret
[docs]
def get_milestone(number=None, name=None, repo_name=None, profile="github", output="min"):
"""
Return information about a single milestone in a named repository.
.. versionadded:: 2016.11.0
number
The number of the milestone to retrieve. If provided, this option
will be favored over ``name``.
name
The name of the milestone to retrieve.
repo_name
The name of the repository for which to list issues. This argument is
required, either passed via the CLI, or defined in the configured
profile. A ``repo_name`` passed as a CLI argument will override the
repo_name defined in the configured profile, if provided.
profile
The name of the profile configuration to use. Defaults to ``github``.
output
The amount of data returned by each issue. Defaults to ``min``. Change
to ``full`` to see all issue output.
CLI Example:
.. code-block:: bash
salt myminion github.get_milestone 72
salt myminion github.get_milestone name=my_milestone
"""
ret = {}
if not any([number, name]):
raise CommandExecutionError("Either a milestone 'name' or 'number' must be provided.")
org_name = _get_config_value(profile, "org_name")
if repo_name is None:
repo_name = _get_config_value(profile, "repo_name")
action = "/".join(["repos", org_name, repo_name])
if number:
command = "milestones/" + str(number)
milestone_data = _query(profile, action=action, command=command)
milestone_data = milestone_data.get("dict", {})
milestone_id = milestone_data.get("id")
if milestone_id:
if output.lower() == "full":
ret[milestone_id] = milestone_data
else:
milestone_data.pop("creator")
milestone_data.pop("html_url")
milestone_data.pop("labels_url")
ret[milestone_id] = milestone_data
return ret
else:
milestones = get_milestones(repo_name=repo_name, profile=profile, output=output)
for key, val in milestones.items():
if val.get("title") == name:
ret[key] = val
return ret
return ret
def _repo_to_dict(repo):
ret = {}
ret["id"] = repo.id
ret["name"] = repo.name
ret["full_name"] = repo.full_name
ret["owner"] = repo.owner.login
ret["private"] = repo.private
ret["html_url"] = repo.html_url
ret["description"] = repo.description
ret["fork"] = repo.fork
ret["homepage"] = repo.homepage
ret["size"] = repo.size
ret["stargazers_count"] = repo.stargazers_count
ret["watchers_count"] = repo.watchers_count
ret["language"] = repo.language
ret["open_issues_count"] = repo.open_issues_count
ret["forks"] = repo.forks
ret["open_issues"] = repo.open_issues
ret["watchers"] = repo.watchers
ret["default_branch"] = repo.default_branch
ret["has_issues"] = repo.has_issues
ret["has_wiki"] = repo.has_wiki
ret["has_downloads"] = repo.has_downloads
return ret
[docs]
def get_repo_info(repo_name, profile="github", ignore_cache=False):
"""
Return information for a given repo.
.. versionadded:: 2016.11.0
repo_name
The name of the repository.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.get_repo_info salt
salt myminion github.get_repo_info salt profile='my-github-profile'
"""
org_name = _get_config_value(profile, "org_name")
key = "github.{}:{}:repo_info".format( # pylint: disable=consider-using-f-string
_get_config_value(profile, "org_name"), repo_name.lower()
)
if key not in __context__ or ignore_cache:
client = _get_client(profile)
try:
repo = client.get_repo("/".join([org_name, repo_name]))
if not repo:
return {}
# client.get_repo can return a github.Repository.Repository object,
# even if the repo is invalid. We need to catch the exception when
# we try to perform actions on the repo object, rather than above
# the if statement.
ret = _repo_to_dict(repo)
__context__[key] = ret
except github.UnknownObjectException as exc:
raise CommandExecutionError(
f"The '{repo_name}' repository under the '{org_name}' organization could not be found."
) from exc
return __context__[key]
[docs]
def get_repo_teams(repo_name, profile="github"):
"""
Return teams belonging to a repository.
.. versionadded:: 2017.7.0
repo_name
The name of the repository from which to retrieve teams.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.get_repo_teams salt
salt myminion github.get_repo_teams salt profile='my-github-profile'
"""
ret = []
org_name = _get_config_value(profile, "org_name")
client = _get_client(profile)
try:
repo = client.get_repo("/".join([org_name, repo_name]))
except github.UnknownObjectException as exc:
raise CommandExecutionError(
f"The '{repo_name}' repository under the '{org_name}' organization could not be found."
) from exc
try:
teams = repo.get_teams()
for team in teams:
ret.append({"id": team.id, "name": team.name, "permission": team.permission})
except github.UnknownObjectException as exc:
raise CommandExecutionError(
f"Unable to retrieve teams for repository '{repo_name}' under the '{org_name}' organization."
) from exc
return ret
[docs]
def list_repos(profile="github"):
"""
List all repositories within the organization. Includes public and private
repositories within the organization Dependent upon the access rights of
the profile token.
.. versionadded:: 2016.11.0
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.list_repos
salt myminion github.list_repos profile='my-github-profile'
"""
return [repo.name for repo in _get_repos(profile)]
[docs]
def list_private_repos(profile="github"):
"""
List private repositories within the organization. Dependent upon the access
rights of the profile token.
.. versionadded:: 2016.11.0
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.list_private_repos
salt myminion github.list_private_repos profile='my-github-profile'
"""
repos = []
for repo in _get_repos(profile):
if repo.private is True:
repos.append(repo.name)
return repos
[docs]
def list_public_repos(profile="github"):
"""
List public repositories within the organization.
.. versionadded:: 2016.11.0
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.list_public_repos
salt myminion github.list_public_repos profile='my-github-profile'
"""
repos = []
for repo in _get_repos(profile):
if repo.private is False:
repos.append(repo.name)
return repos
[docs]
def add_repo(
name,
description=None,
homepage=None,
private=None,
has_issues=None,
has_wiki=None,
has_downloads=None,
auto_init=None,
gitignore_template=None,
license_template=None,
profile="github",
):
"""
Create a new github repository.
name
The name of the team to be created.
description
The description of the repository.
homepage
The URL with more information about the repository.
private
The visiblity of the repository. Note that private repositories require
a paid GitHub account.
has_issues
Whether to enable issues for this repository.
has_wiki
Whether to enable the wiki for this repository.
has_downloads
Whether to enable downloads for this repository.
auto_init
Whether to create an initial commit with an empty README.
gitignore_template
The desired language or platform for a .gitignore, e.g "Haskell".
license_template
The desired LICENSE template to apply, e.g "mit" or "mozilla".
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.add_repo 'repo_name'
.. versionadded:: 2016.11.0
"""
try:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
given_params = {
"description": description,
"homepage": homepage,
"private": private,
"has_issues": has_issues,
"has_wiki": has_wiki,
"has_downloads": has_downloads,
"auto_init": auto_init,
"gitignore_template": gitignore_template,
"license_template": license_template,
}
parameters = {"name": name}
for param_name, param_value in given_params.items():
if param_value is not None:
parameters[param_name] = param_value
organization._requester.requestJsonAndCheck(
"POST", organization.url + "/repos", input=parameters
)
return True
except github.GithubException:
log.exception("Error creating a repo")
return False
[docs]
def edit_repo(
name,
description=None,
homepage=None,
private=None,
has_issues=None,
has_wiki=None,
has_downloads=None,
profile="github",
):
"""
Updates an existing Github repository.
name
The name of the team to be created.
description
The description of the repository.
homepage
The URL with more information about the repository.
private
The visiblity of the repository. Note that private repositories require
a paid GitHub account.
has_issues
Whether to enable issues for this repository.
has_wiki
Whether to enable the wiki for this repository.
has_downloads
Whether to enable downloads for this repository.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.add_repo 'repo_name'
.. versionadded:: 2016.11.0
"""
try:
allow_private_change = _get_config_value(profile, "allow_repo_privacy_changes")
except CommandExecutionError:
allow_private_change = False
if private is not None and not allow_private_change:
raise CommandExecutionError(
f"The private field is set to be changed for repo {name} but allow_repo_privacy_changes disallows this."
)
try:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
repo = organization.get_repo(name)
given_params = {
"description": description,
"homepage": homepage,
"private": private,
"has_issues": has_issues,
"has_wiki": has_wiki,
"has_downloads": has_downloads,
}
parameters = {"name": name}
for param_name, param_value in given_params.items():
if param_value is not None:
parameters[param_name] = param_value
organization._requester.requestJsonAndCheck("PATCH", repo.url, input=parameters)
get_repo_info(name, profile=profile, ignore_cache=True) # Refresh cache
return True
except github.GithubException:
log.exception("Error editing a repo")
return False
[docs]
def remove_repo(name, profile="github"):
"""
Remove a Github repository.
name
The name of the repository to be removed.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.remove_repo 'my-repo'
.. versionadded:: 2016.11.0
"""
repo_info = get_repo_info(name, profile=profile)
if not repo_info:
log.error("Repo %s to be removed does not exist.", name)
return False
try:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
repo = organization.get_repo(name)
repo.delete()
_get_repos(profile=profile, ignore_cache=True) # refresh cache
return True
except github.GithubException:
log.exception("Error deleting a repo")
return False
[docs]
def get_team(name, profile="github"):
"""
Returns the team details if a team with the given name exists, or None
otherwise.
name
The team name for which to obtain information.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.get_team 'team_name'
"""
return list_teams(profile).get(name)
[docs]
def add_team(
name,
description=None,
repo_names=None,
privacy=None,
permission=None,
profile="github",
):
"""
Create a new Github team within an organization.
name
The name of the team to be created.
description
The description of the team.
repo_names
The names of repositories to add the team to.
privacy
The level of privacy for the team, can be 'secret' or 'closed'.
permission
The default permission for new repositories added to the team, can be
'pull', 'push' or 'admin'.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.add_team 'team_name'
.. versionadded:: 2016.11.0
"""
try:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
parameters = {}
parameters["name"] = name
if description is not None:
parameters["description"] = description
if repo_names is not None:
parameters["repo_names"] = repo_names
if permission is not None:
parameters["permission"] = permission
if privacy is not None:
parameters["privacy"] = privacy
organization._requester.requestJsonAndCheck(
"POST", organization.url + "/teams", input=parameters
)
list_teams(ignore_cache=True) # Refresh cache
return True
except github.GithubException:
log.exception("Error creating a team")
return False
[docs]
def edit_team(name, description=None, privacy=None, permission=None, profile="github"):
"""
Updates an existing Github team.
name
The name of the team to be edited.
description
The description of the team.
privacy
The level of privacy for the team, can be 'secret' or 'closed'.
permission
The default permission for new repositories added to the team, can be
'pull', 'push' or 'admin'.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.edit_team 'team_name' description='Team description'
.. versionadded:: 2016.11.0
"""
team = get_team(name, profile=profile)
if not team:
log.error("Team %s does not exist", name)
return False
try:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
team = organization.get_team(team["id"])
parameters = {}
if name is not None:
parameters["name"] = name
if description is not None:
parameters["description"] = description
if privacy is not None:
parameters["privacy"] = privacy
if permission is not None:
parameters["permission"] = permission
team._requester.requestJsonAndCheck("PATCH", team.url, input=parameters)
return True
except UnknownObjectException:
log.exception("Resource not found")
return False
[docs]
def remove_team(name, profile="github"):
"""
Remove a github team.
name
The name of the team to be removed.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.remove_team 'team_name'
.. versionadded:: 2016.11.0
"""
team_info = get_team(name, profile=profile)
if not team_info:
log.error("Team %s to be removed does not exist.", name)
return False
try:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
team = organization.get_team(team_info["id"])
team.delete()
return list_teams(ignore_cache=True, profile=profile).get(name) is None
except github.GithubException:
log.exception("Error deleting a team")
return False
[docs]
def list_team_repos(team_name, profile="github", ignore_cache=False):
"""
Gets the repo details for a given team as a dict from repo_name to repo details.
Note that repo names are always in lower case.
team_name
The name of the team from which to list repos.
profile
The name of the profile configuration to use. Defaults to ``github``.
ignore_cache
Bypasses the use of cached team repos.
CLI Example:
.. code-block:: bash
salt myminion github.list_team_repos 'team_name'
.. versionadded:: 2016.11.0
"""
cached_team = get_team(team_name, profile=profile)
if not cached_team:
log.error("Team %s does not exist.", team_name)
return False
# Return from cache if available
if cached_team.get("repos") and not ignore_cache:
return cached_team.get("repos")
try:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
team = organization.get_team(cached_team["id"])
except UnknownObjectException:
log.exception("Resource not found: %s", cached_team["id"])
try:
repos = {}
for repo in team.get_repos():
permission = "pull"
if repo.permissions.admin:
permission = "admin"
elif repo.permissions.push:
permission = "push"
repos[repo.name.lower()] = {"permission": permission}
cached_team["repos"] = repos
return repos
except UnknownObjectException:
log.exception("Resource not found: %s", cached_team["id"])
return []
[docs]
def add_team_repo(repo_name, team_name, profile="github", permission=None):
"""
Adds a repository to a team with team_name.
repo_name
The name of the repository to add.
team_name
The name of the team of which to add the repository.
profile
The name of the profile configuration to use. Defaults to ``github``.
permission
The permission for team members within the repository, can be 'pull',
'push' or 'admin'. If not specified, the default permission specified on
the team will be used.
.. versionadded:: 2017.7.0
CLI Example:
.. code-block:: bash
salt myminion github.add_team_repo 'my_repo' 'team_name'
.. versionadded:: 2016.11.0
"""
team = get_team(team_name, profile=profile)
if not team:
log.error("Team %s does not exist", team_name)
return False
try:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
team = organization.get_team(team["id"])
repo = organization.get_repo(repo_name)
except UnknownObjectException:
log.exception("Resource not found: %s", team["id"])
return False
params = None
if permission is not None:
params = {"permission": permission}
team._requester.requestJsonAndCheck("PUT", team.url + "/repos/" + repo._identity, input=params)
# Try to refresh cache
list_team_repos(team_name, profile=profile, ignore_cache=True)
return True
[docs]
def remove_team_repo(repo_name, team_name, profile="github"):
"""
Removes a repository from a team with team_name.
repo_name
The name of the repository to remove.
team_name
The name of the team of which to remove the repository.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.remove_team_repo 'my_repo' 'team_name'
.. versionadded:: 2016.11.0
"""
team = get_team(team_name, profile=profile)
if not team:
log.error("Team %s does not exist", team_name)
return False
try:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
team = organization.get_team(team["id"])
repo = organization.get_repo(repo_name)
except UnknownObjectException:
log.exception("Resource not found: %s", team["id"])
return False
team.remove_from_repos(repo)
return repo_name not in list_team_repos(team_name, profile=profile, ignore_cache=True)
[docs]
def list_team_members(team_name, profile="github", ignore_cache=False):
"""
Gets the names of team members in lower case.
team_name
The name of the team from which to list members.
profile
The name of the profile configuration to use. Defaults to ``github``.
ignore_cache
Bypasses the use of cached team members.
CLI Example:
.. code-block:: bash
salt myminion github.list_team_members 'team_name'
.. versionadded:: 2016.11.0
"""
cached_team = get_team(team_name, profile=profile)
if not cached_team:
log.error("Team %s does not exist.", team_name)
return False
# Return from cache if available
if cached_team.get("members") and not ignore_cache:
return cached_team.get("members")
try:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
team = organization.get_team(cached_team["id"])
except UnknownObjectException:
log.exception("Resource not found: %s", cached_team["id"])
try:
cached_team["members"] = [member.login.lower() for member in team.get_members()]
return cached_team["members"]
except UnknownObjectException:
log.exception("Resource not found: %s", cached_team["id"])
return []
[docs]
def list_members_without_mfa(profile="github", ignore_cache=False):
"""
List all members (in lower case) without MFA turned on.
profile
The name of the profile configuration to use. Defaults to ``github``.
ignore_cache
Bypasses the use of cached team repos.
CLI Example:
.. code-block:: bash
salt myminion github.list_members_without_mfa
.. versionadded:: 2016.11.0
"""
key = "github.{}:non_mfa_users".format( # pylint: disable=consider-using-f-string
_get_config_value(profile, "org_name")
)
if key not in __context__ or ignore_cache:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
filter_key = "filter"
# Silly hack to see if we're past PyGithub 1.26.0, where the name of
# the filter kwarg changed
if hasattr(github.Team.Team, "membership"):
filter_key = "filter_"
__context__[key] = [
m.login.lower() for m in _get_members(organization, {filter_key: "2fa_disabled"})
]
return __context__[key]
[docs]
def is_team_member(name, team_name, profile="github"):
"""
Returns True if the github user is in the team with team_name, or False
otherwise.
name
The name of the user whose membership to check.
team_name
The name of the team to check membership in.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.is_team_member 'user_name' 'team_name'
.. versionadded:: 2016.11.0
"""
return name.lower() in list_team_members(team_name, profile=profile)
[docs]
def add_team_member(name, team_name, profile="github"):
"""
Adds a team member to a team with team_name.
name
The name of the team member to add.
team_name
The name of the team of which to add the user.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.add_team_member 'user_name' 'team_name'
.. versionadded:: 2016.11.0
"""
team = get_team(team_name, profile=profile)
if not team:
log.error("Team %s does not exist", team_name)
return False
try:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
team = organization.get_team(team["id"])
member = client.get_user(name)
except UnknownObjectException:
log.exception("Resource not found: %s", team["id"])
return False
try:
# Can't use team.add_membership due to this bug that hasn't made it into
# a PyGithub release yet https://github.com/PyGithub/PyGithub/issues/363
team._requester.requestJsonAndCheck(
"PUT",
team.url + "/memberships/" + member._identity,
input={"role": "member"},
parameters={"role": "member"},
)
except github.GithubException:
log.exception("Error in adding a member to a team")
return False
return True
[docs]
def remove_team_member(name, team_name, profile="github"):
"""
Removes a team member from a team with team_name.
name
The name of the team member to remove.
team_name
The name of the team from which to remove the user.
profile
The name of the profile configuration to use. Defaults to ``github``.
CLI Example:
.. code-block:: bash
salt myminion github.remove_team_member 'user_name' 'team_name'
.. versionadded:: 2016.11.0
"""
team = get_team(team_name, profile=profile)
if not team:
log.error("Team %s does not exist", team_name)
return False
try:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
team = organization.get_team(team["id"])
member = client.get_user(name)
except UnknownObjectException:
log.exception("Resource not found: %s", team["id"])
return False
if not hasattr(team, "remove_from_members"):
return (
False,
"PyGithub 1.26.0 or greater is required for team management, please upgrade.",
)
team.remove_from_members(member)
return not team.has_in_members(member)
[docs]
def list_teams(profile="github", ignore_cache=False):
"""
Lists all teams with the organization.
profile
The name of the profile configuration to use. Defaults to ``github``.
ignore_cache
Bypasses the use of cached teams.
CLI Example:
.. code-block:: bash
salt myminion github.list_teams
.. versionadded:: 2016.11.0
"""
key = "github.{}:teams".format( # pylint: disable=consider-using-f-string
_get_config_value(profile, "org_name")
)
if key not in __context__ or ignore_cache:
client = _get_client(profile)
organization = client.get_organization(_get_config_value(profile, "org_name"))
teams_data = organization.get_teams()
teams = {}
for team in teams_data:
# Note that _rawData is used to access some properties here as they
# are not exposed in older versions of PyGithub. It's VERY important
# to use team._rawData instead of team.raw_data, as the latter forces
# an API call to retrieve team details again.
teams[team.name] = {
"id": team.id,
"slug": team.slug,
"description": team._rawData["description"],
"permission": team.permission,
"privacy": team._rawData["privacy"],
}
__context__[key] = teams
return __context__[key]
[docs]
def get_prs(
repo_name=None,
profile="github",
state="open",
head=None,
base=None,
sort="created",
direction="desc",
output="min",
per_page=None,
):
"""
Returns information for all pull requests in a given repository, based on
the search options provided.
.. versionadded:: 2017.7.0
repo_name
The name of the repository for which to list pull requests. This
argument is required, either passed via the CLI, or defined in the
configured profile. A ``repo_name`` passed as a CLI argument will
override the ``repo_name`` defined in the configured profile, if
provided.
profile
The name of the profile configuration to use. Defaults to ``github``.
state
Indicates the state of the pull requests to return. Can be either
``open``, ``closed``, or ``all``. Default is ``open``.
head
Filter pull requests by head user and branch name in the format of
``user:ref-name``. Example: ``'github:new-script-format'``. Default
is ``None``.
base
Filter pulls by base branch name. Example: ``gh-pages``. Default is
``None``.
sort
What to sort results by. Can be either ``created``, ``updated``,
``popularity`` (comment count), or ``long-running`` (age, filtering
by pull requests updated within the last month). Default is ``created``.
direction
The direction of the sort. Can be either ``asc`` or ``desc``. Default
is ``desc``.
output
The amount of data returned by each pull request. Defaults to ``min``.
Change to ``full`` to see all pull request output.
per_page
GitHub paginates data in their API calls. Use this value to increase or
decrease the number of pull requests gathered from GitHub, per page. If
not set, GitHub defaults are used. Maximum is 100.
CLI Example:
.. code-block:: bash
salt myminion github.get_prs
salt myminion github.get_prs base=2016.11
"""
org_name = _get_config_value(profile, "org_name")
if repo_name is None:
repo_name = _get_config_value(profile, "repo_name")
action = "/".join(["repos", org_name, repo_name])
args = {}
# Build API arguments, as necessary.
if head:
args["head"] = head
if base:
args["base"] = base
if per_page:
args["per_page"] = per_page
# Only pass the following API args if they're not the defaults listed.
if state and state != "open":
args["state"] = state
if sort and sort != "created":
args["sort"] = sort
if direction and direction != "desc":
args["direction"] = direction
ret = {}
prs = _query(profile, action=action, command="pulls", args=args)
for pr_ in prs.get("dict", {}):
pr_id = pr_.get("id")
if output == "full":
ret[pr_id] = pr_
else:
ret[pr_id] = _format_pr(pr_)
return ret
def _format_pr(pr_):
"""
Helper function to format API return information into a more manageable
and useful dictionary for pull request information.
pr_
The pull request to format.
"""
ret = {
"id": pr_.get("id"),
"pr_number": pr_.get("number"),
"state": pr_.get("state"),
"title": pr_.get("title"),
"user": pr_.get("user").get("login"),
"html_url": pr_.get("html_url"),
"base_branch": pr_.get("base").get("ref"),
}
return ret
def _format_issue(issue):
"""
Helper function to format API return information into a more manageable
and useful dictionary for issue information.
issue
The issue to format.
"""
ret = {
"id": issue.get("id"),
"issue_number": issue.get("number"),
"state": issue.get("state"),
"title": issue.get("title"),
"user": issue.get("user").get("login"),
"html_url": issue.get("html_url"),
}
assignee = issue.get("assignee")
if assignee:
assignee = assignee.get("login")
labels = issue.get("labels")
label_names = []
for label in labels:
label_names.append(label.get("name"))
milestone = issue.get("milestone")
if milestone:
milestone = milestone.get("title")
ret["assignee"] = assignee
ret["labels"] = label_names
ret["milestone"] = milestone
return ret
def _query(
profile,
action=None,
command=None,
args=None,
method="GET",
header_dict=None,
data=None,
url="https://api.github.com/",
per_page=None,
):
"""
Make a web call to the GitHub API and deal with paginated results.
"""
if not isinstance(args, dict):
args = {}
if action:
url += action
if command:
url += f"/{command}"
log.debug("GitHub URL: %s", url)
if header_dict is None:
header_dict = {}
if not header_dict.get("Authorization"):
header_dict["Authorization"] = f"Bearer {_get_config_value(profile, 'token')}"
if per_page and "per_page" not in args.keys():
args["per_page"] = per_page
header_dict["Accept"] = "application/vnd.github+json"
decode = True
if method == "DELETE":
decode = False
# GitHub paginates all queries when returning many items.
# Gather all data using multiple queries and handle pagination.
ret = {}
complete_result = []
next_page = True
page_number = ""
while next_page is True:
if page_number:
args["page"] = page_number
result = salt.utils.http.query(
url,
method,
params=args,
data=data,
header_dict=header_dict,
decode=decode,
decode_type="json",
headers=True,
status=True,
text=True,
hide_fields=None,
opts=__opts__,
)
log.debug("GitHub Response Status Code: %s", result.get("status"))
if result.get("status") in [200, 201]:
if isinstance(result["dict"], dict):
# If only querying for one item, such as a single issue
# The GitHub API returns a single dictionary, instead of
# A list of dictionaries. In that case, we can return.
return result
ret = result
complete_result = complete_result + result["dict"]
ret["dict"] = complete_result
else:
if result:
return result
raise CommandExecutionError
try:
link_info = result.get("headers").get("Link").split(",")[0]
except AttributeError:
# Only one page of data was returned; exit the loop.
next_page = False
continue
if "next" in link_info:
# Get the 'next' page number from the Link header.
page_number = link_info.split(">")[0].split("&page=")[1]
else:
# Last page already processed; break the loop.
next_page = False
return ret
def _check_ruleset_params(profile, rule_id=False, rule_params=False, **kwargs):
"""
Helper function to check for param values on command line and in config file
profile
The name of the profile configuration to use.
ruleset_id
Defautls to false. If ruleset_id is required, set to True.
ruleset_params
Defautls to false. If ruleset_params are required, set to True.
"""
if not kwargs.get("ruleset_type"):
kwargs["ruleset_type"] = _get_config_value(profile, "ruleset_type")
if kwargs["ruleset_type"] not in ["repo", "org"]:
raise CommandExecutionError
if kwargs["ruleset_type"] == "repo":
if not kwargs.get("owner"):
kwargs["owner"] = _get_config_value(profile, "owner")
if not kwargs.get("repo_name"):
kwargs["repo_name"] = _get_config_value(profile, "repo_name")
action = "/".join(["repos", kwargs["owner"], kwargs["repo_name"], "rulesets"])
else:
if not kwargs.get("org_name"):
kwargs["org_name"] = _get_config_value(profile, "org_name")
action = "/".join(["orgs", kwargs["org_name"], "rulesets"])
# check for optional header_dict
if not kwargs.get("header_dict"):
try:
kwargs["header_dict"] = _get_config_value(profile, "header_dict")
except CommandExecutionError:
kwargs["header_dict"] = {}
# verify header_dict is a dict
if not isinstance(kwargs["header_dict"], dict):
kwargs["header_dict"] = {}
# if ruleset is required, check for ruleset_id
if rule_id:
if not kwargs.get("ruleset_id"):
kwargs["ruleset_id"] = _get_config_value(profile, "ruleset_id")
action = action + "/" + str(kwargs["ruleset_id"])
# if ruleset_params are required, check for ruleset_params
if rule_params:
if not kwargs.get("ruleset_params"):
kwargs["ruleset_params"] = _get_config_value(profile, "ruleset_params")
return kwargs, action
[docs]
def get_ruleset(profile="github", **kwargs):
"""
Get information of specific ruleset.
profile
The name of the profile configuration to use. Defaults to ``github``.
ruleset_type
The type of ruleset. 'repo' or 'org'.
ruleset_id
Id of the ruleset.
owner
The account owner of the repository.
repo_name
The name of the repo.
org_name
The name of the organization.
header_dict
Headers to pass to Github API.
CLI Example:
.. code-block:: bash
salt myminion github.get_ruleset
salt myminion github.get_ruleset ruleset_id=1
"""
params, action = _check_ruleset_params(profile, rule_id=True, **kwargs)
try:
ret = _query(profile, action, header_dict=params["header_dict"])
if not ret.get("error"):
return ret["dict"]
ret["comment"] = f"GitHub Response Status Code: {ret.get('error')}"
ret["result"] = False
except CommandExecutionError:
ret = {}
ret["result"] = False
ret["comment"] = "Error getting ruleset"
return ret
[docs]
def delete_ruleset(profile="github", **kwargs):
"""
Delete ruleset
profile
The name of the profile configuration to use. Defaults to ``github``.
ruleset_type
The type of ruleset. 'repo' or 'org'.
owner
The account owner of the repository.
repo_name
The name of the repo.
ruleset_id
Id of the ruleset.
org_name
The name of the organization.
header_dict
Headers to pass to Github API.
CLI Example:
.. code-block:: bash
salt myminion github.delete_ruleset
salt myminion github.delete_ruleset ruleset_id=1
"""
params, action = _check_ruleset_params(profile, rule_id=True, **kwargs)
try:
ret = _query(profile, action, method="DELETE", header_dict=params["header_dict"])
if ret.get("error"):
ret["comment"] = f"GitHub Response Status Code: {ret.get('error')}"
ret["result"] = False
else:
ret["comment"] = f"ruleset {params['ruleset_id']} successfully deleted"
except CommandExecutionError:
ret = {}
ret["result"] = False
ret["comment"] = "Error deleting ruleset"
return ret
[docs]
def update_ruleset(profile="github", **kwargs):
"""
Update ruleset
profile
The name of the profile configuration to use. Defaults to ``github``.
ruleset_type
The type of ruleset. 'repo' or 'org'.
ruleset_id
Id of the ruleset.
owner
The account owner of the repository.
repo_name
The name of the repo.
org_name
The name of the organization.
header_dict
headers to pass to Github API.
ruleset_params
number of returned results (max 100).
CLI Example:
.. code-block:: bash
salt myminion github.update_ruleset
salt myminion github.update_ruleset ruleset_id=1
"""
params, action = _check_ruleset_params(profile, rule_id=True, rule_params=True, **kwargs)
if not isinstance(params["ruleset_params"], dict):
raise CommandExecutionError("params need to be a dict")
ruleset_params = salt.utils.json.dumps(params["ruleset_params"])
try:
ret = _query(
profile, action, method="PUT", data=ruleset_params, header_dict=params["header_dict"]
)
if not ret.get("error"):
return ret["dict"]
else:
ret["comment"] = f"GitHub Response Status Code: {ret.get('error')}"
ret["result"] = False
except CommandExecutionError:
ret = {}
ret["result"] = False
ret["comment"] = "Error updating ruleset"
return ret
[docs]
def add_ruleset(profile="github", **kwargs):
"""
Add ruleset
profile
The name of the profile configuration to use. Defaults to ``github``.
ruleset_type
The type of ruleset. 'repo' or 'org'.
owner
The account owner of the repository.
repo_name
The name of the repo.
org_name
The name of the organization.
header_dict
headers to pass to Github API.
ruleset_params
number of returned results (max 100).
CLI Example:
.. code-block:: bash
salt myminion github.add_ruleset
"""
params, action = _check_ruleset_params(profile, rule_params=True, **kwargs)
if not isinstance(params["ruleset_params"], dict):
raise CommandExecutionError("params need to be a dict")
if "name" not in params["ruleset_params"]:
raise CommandExecutionError("name is required")
if "enforcement" not in params["ruleset_params"]:
raise CommandExecutionError("enforcement is required")
ruleset_params = salt.utils.json.dumps(params["ruleset_params"])
try:
ret = _query(
profile, action, method="POST", data=ruleset_params, header_dict=params["header_dict"]
)
if not ret.get("error"):
return ret["dict"]
else:
ret["comment"] = f"GitHub Response Status Code: {ret.get('error')}"
ret["result"] = False
except CommandExecutionError:
ret = {}
ret["result"] = False
ret["comment"] = "Error adding ruleset"
return ret
[docs]
def list_rulesets(profile="github", **kwargs):
"""
List rulesets
profile
The name of the profile configuration to use. Defaults to ``github``.
ruleset_type
The type of ruleset. 'repo' or 'org'.
owner
The account owner of the repository.
repo_name
The name of the repo.
org_name
The name of the organization.
header_dict
headers to pass to Github API.
per_page
number of returned results (max 100).
CLI Example:
.. code-block:: bash
salt myminion github.list_rulesets
"""
params, action = _check_ruleset_params(profile, **kwargs)
if not kwargs.get("per_page"):
try:
kwargs["per_page"] = _get_config_value(profile, "per_page")
except CommandExecutionError:
kwargs["per_page"] = None
try:
ret = _query(
profile, action, header_dict=params["header_dict"], per_page=kwargs["per_page"]
)
if not ret.get("error"):
if ret["dict"]:
return {"rulesets": ret["dict"]}
return {"rulesets": None}
else:
ret["comment"] = f"GitHub Response Status Code: {ret.get('error')}"
ret["result"] = False
except CommandExecutionError:
ret = {}
ret["result"] = False
ret["comment"] = "Error getting rulesets"
return ret