defmodule GitHub.SecurityAdvisories do
@moduledoc """
Provides API endpoints related to security advisories
"""
@default_client GitHub.Client
@doc """
Privately report a security vulnerability
Report a security vulnerability to the maintainers of the repository.
See "[Privately reporting a security vulnerability](https://docs.github.com/code-security/security-advisories/guidance-on-reporting-and-writing/privately-reporting-a-security-vulnerability)" for more information about private vulnerability reporting.
## Resources
* [API method documentation](https://docs.github.com/rest/security-advisories/repository-advisories#privately-report-a-security-vulnerability)
"""
@spec create_private_vulnerability_report(
String.t(),
String.t(),
GitHub.PrivateVulnerabilityReport.Create.t(),
keyword
) :: {:ok, GitHub.Repository.Advisory.t()} | {:error, GitHub.Error.t()}
def create_private_vulnerability_report(owner, repo, body, opts \\ []) do
client = opts[:client] || @default_client
client.request(%{
args: [owner: owner, repo: repo, body: body],
call: {GitHub.SecurityAdvisories, :create_private_vulnerability_report},
url: "/repos/#{owner}/#{repo}/security-advisories/reports",
body: body,
method: :post,
request: [{"application/json", {GitHub.PrivateVulnerabilityReport.Create, :t}}],
response: [
{201, {GitHub.Repository.Advisory, :t}},
{403, {GitHub.BasicError, :t}},
{404, {GitHub.BasicError, :t}},
{422, {GitHub.ValidationError, :t}}
],
opts: opts
})
end
@doc """
Create a repository security advisory
Creates a new repository security advisory.
You must authenticate using an access token with the `repo` scope or `repository_advisories:write` permission to use this endpoint.
In order to create a draft repository security advisory, you must be a security manager or administrator of that repository.
## Resources
* [API method documentation](https://docs.github.com/rest/security-advisories/repository-advisories#create-a-repository-security-advisory)
"""
@spec create_repository_advisory(
String.t(),
String.t(),
GitHub.Repository.Advisory.Create.t(),
keyword
) :: {:ok, GitHub.Repository.Advisory.t()} | {:error, GitHub.Error.t()}
def create_repository_advisory(owner, repo, body, opts \\ []) do
client = opts[:client] || @default_client
client.request(%{
args: [owner: owner, repo: repo, body: body],
call: {GitHub.SecurityAdvisories, :create_repository_advisory},
url: "/repos/#{owner}/#{repo}/security-advisories",
body: body,
method: :post,
request: [{"application/json", {GitHub.Repository.Advisory.Create, :t}}],
response: [
{201, {GitHub.Repository.Advisory, :t}},
{403, {GitHub.BasicError, :t}},
{404, {GitHub.BasicError, :t}},
{422, {GitHub.ValidationError, :t}}
],
opts: opts
})
end
@doc """
Request a CVE for a repository security advisory
If you want a CVE identification number for the security vulnerability in your project, and don't already have one, you can request a CVE identification number from GitHub. For more information see "[Requesting a CVE identification number](https://docs.github.com/code-security/security-advisories/repository-security-advisories/publishing-a-repository-security-advisory#requesting-a-cve-identification-number-optional)."
You may request a CVE for public repositories, but cannot do so for private repositories.
You must authenticate using an access token with the `repo` scope or `repository_advisories:write` permission to use this endpoint.
In order to request a CVE for a repository security advisory, you must be a security manager or administrator of that repository.
## Resources
* [API method documentation](https://docs.github.com/rest/security-advisories/repository-advisories#request-a-cve-for-a-repository-security-advisory)
"""
@spec create_repository_advisory_cve_request(String.t(), String.t(), String.t(), keyword) ::
{:ok, map} | {:error, GitHub.Error.t()}
def create_repository_advisory_cve_request(owner, repo, ghsa_id, opts \\ []) do
client = opts[:client] || @default_client
client.request(%{
args: [owner: owner, repo: repo, ghsa_id: ghsa_id],
call: {GitHub.SecurityAdvisories, :create_repository_advisory_cve_request},
url: "/repos/#{owner}/#{repo}/security-advisories/#{ghsa_id}/cve",
method: :post,
response: [
{202, :map},
{400, {:union, [{GitHub.BasicError, :t}, {GitHub.SCIM.Error, :t}]}},
{403, {GitHub.BasicError, :t}},
{404, {GitHub.BasicError, :t}},
{422, {GitHub.ValidationError, :t}}
],
opts: opts
})
end
@doc """
Get a global security advisory
Gets a global security advisory using its GitHub Security Advisory (GHSA) identifier.
## Resources
* [API method documentation](https://docs.github.com/rest/security-advisories/global-advisories#get-a-global-security-advisory)
"""
@spec get_global_advisory(String.t(), keyword) ::
{:ok, GitHub.GlobalAdvisory.t()} | {:error, GitHub.Error.t()}
def get_global_advisory(ghsa_id, opts \\ []) do
client = opts[:client] || @default_client
client.request(%{
args: [ghsa_id: ghsa_id],
call: {GitHub.SecurityAdvisories, :get_global_advisory},
url: "/advisories/#{ghsa_id}",
method: :get,
response: [{200, {GitHub.GlobalAdvisory, :t}}, {404, {GitHub.BasicError, :t}}],
opts: opts
})
end
@doc """
Get a repository security advisory
Get a repository security advisory using its GitHub Security Advisory (GHSA) identifier.
You can access any published security advisory on a public repository.
You must authenticate using an access token with the `repo` scope or `repository_advisories:read` permission
in order to get a published security advisory in a private repository, or any unpublished security advisory that you have access to.
You can access an unpublished security advisory from a repository if you are a security manager or administrator of that repository, or if you are a
collaborator on the security advisory.
## Resources
* [API method documentation](https://docs.github.com/rest/security-advisories/repository-advisories#get-a-repository-security-advisory)
"""
@spec get_repository_advisory(String.t(), String.t(), String.t(), keyword) ::
{:ok, GitHub.Repository.Advisory.t()} | {:error, GitHub.Error.t()}
def get_repository_advisory(owner, repo, ghsa_id, opts \\ []) do
client = opts[:client] || @default_client
client.request(%{
args: [owner: owner, repo: repo, ghsa_id: ghsa_id],
call: {GitHub.SecurityAdvisories, :get_repository_advisory},
url: "/repos/#{owner}/#{repo}/security-advisories/#{ghsa_id}",
method: :get,
response: [
{200, {GitHub.Repository.Advisory, :t}},
{403, {GitHub.BasicError, :t}},
{404, {GitHub.BasicError, :t}}
],
opts: opts
})
end
@doc """
List global security advisories
Lists all global security advisories that match the specified parameters. If no other parameters are defined, the request will return only GitHub-reviewed advisories that are not malware.
By default, all responses will exclude advisories for malware, because malware are not standard vulnerabilities. To list advisories for malware, you must include the `type` parameter in your request, with the value `malware`. For more information about the different types of security advisories, see "[About the GitHub Advisory database](https://docs.github.com/code-security/security-advisories/global-security-advisories/about-the-github-advisory-database#about-types-of-security-advisories)."
## Options
* `ghsa_id`: If specified, only advisories with this GHSA (GitHub Security Advisory) identifier will be returned.
* `type`: If specified, only advisories of this type will be returned. By default, a request with no other parameters defined will only return reviewed advisories that are not malware.
* `cve_id`: If specified, only advisories with this CVE (Common Vulnerabilities and Exposures) identifier will be returned.
* `ecosystem`: If specified, only advisories for these ecosystems will be returned.
* `severity`: If specified, only advisories with these severities will be returned.
* `cwes`: If specified, only advisories with these Common Weakness Enumerations (CWEs) will be returned.
Example: `cwes=79,284,22` or `cwes[]=79&cwes[]=284&cwes[]=22`
* `is_withdrawn`: Whether to only return advisories that have been withdrawn.
* `affects`: If specified, only return advisories that affect any of `package` or `package@version`. A maximum of 1000 packages can be specified.
If the query parameter causes the URL to exceed the maximum URL length supported by your client, you must specify fewer packages.
Example: `affects=package1,package2@1.0.0,package3@^2.0.0` or `affects[]=package1&affects[]=package2@1.0.0`
* `published`: If specified, only return advisories that were published on a date or date range.
For more information on the syntax of the date range, see "[Understanding the search syntax](https://docs.github.com/search-github/getting-started-with-searching-on-github/understanding-the-search-syntax#query-for-dates)."
* `updated`: If specified, only return advisories that were updated on a date or date range.
For more information on the syntax of the date range, see "[Understanding the search syntax](https://docs.github.com/search-github/getting-started-with-searching-on-github/understanding-the-search-syntax#query-for-dates)."
* `modified`: If specified, only show advisories that were updated or published on a date or date range.
For more information on the syntax of the date range, see "[Understanding the search syntax](https://docs.github.com/search-github/getting-started-with-searching-on-github/understanding-the-search-syntax#query-for-dates)."
* `before`: A cursor, as given in the [Link header](https://docs.github.com/rest/guides/using-pagination-in-the-rest-api#using-link-headers). If specified, the query only searches for results before this cursor.
* `after`: A cursor, as given in the [Link header](https://docs.github.com/rest/guides/using-pagination-in-the-rest-api#using-link-headers). If specified, the query only searches for results after this cursor.
* `direction`: The direction to sort the results by.
* `per_page`: The number of results per page (max 100).
* `sort`: The property to sort the results by.
## Resources
* [API method documentation](https://docs.github.com/rest/security-advisories/global-advisories#list-global-security-advisories)
"""
@spec list_global_advisories(keyword) ::
{:ok, [GitHub.GlobalAdvisory.t()]} | {:error, GitHub.Error.t()}
def list_global_advisories(opts \\ []) do
client = opts[:client] || @default_client
query =
Keyword.take(opts, [
:affects,
:after,
:before,
:cve_id,
:cwes,
:direction,
:ecosystem,
:ghsa_id,
:is_withdrawn,
:modified,
:per_page,
:published,
:severity,
:sort,
:type,
:updated
])
client.request(%{
args: [],
call: {GitHub.SecurityAdvisories, :list_global_advisories},
url: "/advisories",
method: :get,
query: query,
response: [
{200, [{GitHub.GlobalAdvisory, :t}]},
{422, {GitHub.ValidationError, :simple}},
{429, {GitHub.BasicError, :t}}
],
opts: opts
})
end
@doc """
List repository security advisories for an organization
Lists repository security advisories for an organization.
To use this endpoint, you must be an owner or security manager for the organization, and you must use an access token with the `repo` scope or `repository_advisories:write` permission.
## Options
* `direction`: The direction to sort the results by.
* `sort`: The property to sort the results by.
* `before`: A cursor, as given in the [Link header](https://docs.github.com/rest/guides/using-pagination-in-the-rest-api#using-link-headers). If specified, the query only searches for results before this cursor.
* `after`: A cursor, as given in the [Link header](https://docs.github.com/rest/guides/using-pagination-in-the-rest-api#using-link-headers). If specified, the query only searches for results after this cursor.
* `per_page`: The number of advisories to return per page.
* `state`: Filter by the state of the repository advisories. Only advisories of this state will be returned.
## Resources
* [API method documentation](https://docs.github.com/rest/security-advisories/repository-advisories#list-repository-security-advisories-for-an-organization)
"""
@spec list_org_repository_advisories(String.t(), keyword) ::
{:ok, [GitHub.Repository.Advisory.t()]} | {:error, GitHub.Error.t()}
def list_org_repository_advisories(org, opts \\ []) do
client = opts[:client] || @default_client
query = Keyword.take(opts, [:after, :before, :direction, :per_page, :sort, :state])
client.request(%{
args: [org: org],
call: {GitHub.SecurityAdvisories, :list_org_repository_advisories},
url: "/orgs/#{org}/security-advisories",
method: :get,
query: query,
response: [
{200, [{GitHub.Repository.Advisory, :t}]},
{400, {:union, [{GitHub.BasicError, :t}, {GitHub.SCIM.Error, :t}]}},
{404, {GitHub.BasicError, :t}}
],
opts: opts
})
end
@doc """
List repository security advisories
Lists security advisories in a repository.
You must authenticate using an access token with the `repo` scope or `repository_advisories:read` permission
in order to get published security advisories in a private repository, or any unpublished security advisories that you have access to.
You can access unpublished security advisories from a repository if you are a security manager or administrator of that repository, or if you are a collaborator on any security advisory.
## Options
* `direction`: The direction to sort the results by.
* `sort`: The property to sort the results by.
* `before`: A cursor, as given in the [Link header](https://docs.github.com/rest/guides/using-pagination-in-the-rest-api#using-link-headers). If specified, the query only searches for results before this cursor.
* `after`: A cursor, as given in the [Link header](https://docs.github.com/rest/guides/using-pagination-in-the-rest-api#using-link-headers). If specified, the query only searches for results after this cursor.
* `per_page`: Number of advisories to return per page.
* `state`: Filter by state of the repository advisories. Only advisories of this state will be returned.
## Resources
* [API method documentation](https://docs.github.com/rest/security-advisories/repository-advisories#list-repository-security-advisories)
"""
@spec list_repository_advisories(String.t(), String.t(), keyword) ::
{:ok, [GitHub.Repository.Advisory.t()]} | {:error, GitHub.Error.t()}
def list_repository_advisories(owner, repo, opts \\ []) do
client = opts[:client] || @default_client
query = Keyword.take(opts, [:after, :before, :direction, :per_page, :sort, :state])
client.request(%{
args: [owner: owner, repo: repo],
call: {GitHub.SecurityAdvisories, :list_repository_advisories},
url: "/repos/#{owner}/#{repo}/security-advisories",
method: :get,
query: query,
response: [
{200, [{GitHub.Repository.Advisory, :t}]},
{400, {:union, [{GitHub.BasicError, :t}, {GitHub.SCIM.Error, :t}]}},
{404, {GitHub.BasicError, :t}}
],
opts: opts
})
end
@doc """
Update a repository security advisory
Update a repository security advisory using its GitHub Security Advisory (GHSA) identifier.
You must authenticate using an access token with the `repo` scope or `repository_advisories:write` permission to use this endpoint.
In order to update any security advisory, you must be a security manager or administrator of that repository,
or a collaborator on the repository security advisory.
## Resources
* [API method documentation](https://docs.github.com/rest/security-advisories/repository-advisories#update-a-repository-security-advisory)
"""
@spec update_repository_advisory(
String.t(),
String.t(),
String.t(),
GitHub.Repository.Advisory.Update.t(),
keyword
) :: {:ok, GitHub.Repository.Advisory.t()} | {:error, GitHub.Error.t()}
def update_repository_advisory(owner, repo, ghsa_id, body, opts \\ []) do
client = opts[:client] || @default_client
client.request(%{
args: [owner: owner, repo: repo, ghsa_id: ghsa_id, body: body],
call: {GitHub.SecurityAdvisories, :update_repository_advisory},
url: "/repos/#{owner}/#{repo}/security-advisories/#{ghsa_id}",
body: body,
method: :patch,
request: [{"application/json", {GitHub.Repository.Advisory.Update, :t}}],
response: [
{200, {GitHub.Repository.Advisory, :t}},
{403, {GitHub.BasicError, :t}},
{404, {GitHub.BasicError, :t}},
{422, {GitHub.ValidationError, :t}}
],
opts: opts
})
end
end