// Copyright 2013 The go-github AUTHORS. All rights reserved.
//
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.

package github

import (
	
	
	
	
	
	
	
	
	
)

// RepositoryRelease represents a GitHub release in a repository.
type RepositoryRelease struct {
	TagName         *string `json:"tag_name,omitempty"`
	TargetCommitish *string `json:"target_commitish,omitempty"`
	Name            *string `json:"name,omitempty"`
	Body            *string `json:"body,omitempty"`
	Draft           *bool   `json:"draft,omitempty"`
	Prerelease      *bool   `json:"prerelease,omitempty"`
	// MakeLatest can be one of: "true", "false", or "legacy".
	MakeLatest             *string `json:"make_latest,omitempty"`
	DiscussionCategoryName *string `json:"discussion_category_name,omitempty"`

	// The following fields are not used in EditRelease:
	GenerateReleaseNotes *bool `json:"generate_release_notes,omitempty"`

	// The following fields are not used in CreateRelease or EditRelease:
	ID          *int64          `json:"id,omitempty"`
	CreatedAt   *Timestamp      `json:"created_at,omitempty"`
	PublishedAt *Timestamp      `json:"published_at,omitempty"`
	URL         *string         `json:"url,omitempty"`
	HTMLURL     *string         `json:"html_url,omitempty"`
	AssetsURL   *string         `json:"assets_url,omitempty"`
	Assets      []*ReleaseAsset `json:"assets,omitempty"`
	UploadURL   *string         `json:"upload_url,omitempty"`
	ZipballURL  *string         `json:"zipball_url,omitempty"`
	TarballURL  *string         `json:"tarball_url,omitempty"`
	Author      *User           `json:"author,omitempty"`
	NodeID      *string         `json:"node_id,omitempty"`
}

func ( RepositoryRelease) () string {
	return Stringify()
}

// RepositoryReleaseNotes represents a GitHub-generated release notes.
type RepositoryReleaseNotes struct {
	Name string `json:"name"`
	Body string `json:"body"`
}

// GenerateNotesOptions represents the options to generate release notes.
type GenerateNotesOptions struct {
	TagName         string  `json:"tag_name"`
	PreviousTagName *string `json:"previous_tag_name,omitempty"`
	TargetCommitish *string `json:"target_commitish,omitempty"`
}

// ReleaseAsset represents a GitHub release asset in a repository.
type ReleaseAsset struct {
	ID                 *int64     `json:"id,omitempty"`
	URL                *string    `json:"url,omitempty"`
	Name               *string    `json:"name,omitempty"`
	Label              *string    `json:"label,omitempty"`
	State              *string    `json:"state,omitempty"`
	ContentType        *string    `json:"content_type,omitempty"`
	Size               *int       `json:"size,omitempty"`
	DownloadCount      *int       `json:"download_count,omitempty"`
	CreatedAt          *Timestamp `json:"created_at,omitempty"`
	UpdatedAt          *Timestamp `json:"updated_at,omitempty"`
	BrowserDownloadURL *string    `json:"browser_download_url,omitempty"`
	Uploader           *User      `json:"uploader,omitempty"`
	NodeID             *string    `json:"node_id,omitempty"`
}

func ( ReleaseAsset) () string {
	return Stringify()
}

// ListReleases lists the releases for a repository.
//
// GitHub API docs: https://docs.github.com/rest/releases/releases#list-releases
//
//meta:operation GET /repos/{owner}/{repo}/releases
func ( *RepositoriesService) ( context.Context, ,  string,  *ListOptions) ([]*RepositoryRelease, *Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases", , )
	,  := addOptions(, )
	if  != nil {
		return nil, nil, 
	}

	,  := .client.NewRequest("GET", , nil)
	if  != nil {
		return nil, nil, 
	}

	var  []*RepositoryRelease
	,  := .client.Do(, , &)
	if  != nil {
		return nil, , 
	}
	return , , nil
}

// GetRelease fetches a single release.
//
// GitHub API docs: https://docs.github.com/rest/releases/releases#get-a-release
//
//meta:operation GET /repos/{owner}/{repo}/releases/{release_id}
func ( *RepositoriesService) ( context.Context, ,  string,  int64) (*RepositoryRelease, *Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases/%d", , , )
	return .getSingleRelease(, )
}

// GetLatestRelease fetches the latest published release for the repository.
//
// GitHub API docs: https://docs.github.com/rest/releases/releases#get-the-latest-release
//
//meta:operation GET /repos/{owner}/{repo}/releases/latest
func ( *RepositoriesService) ( context.Context, ,  string) (*RepositoryRelease, *Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases/latest", , )
	return .getSingleRelease(, )
}

// GetReleaseByTag fetches a release with the specified tag.
//
// GitHub API docs: https://docs.github.com/rest/releases/releases#get-a-release-by-tag-name
//
//meta:operation GET /repos/{owner}/{repo}/releases/tags/{tag}
func ( *RepositoriesService) ( context.Context, , ,  string) (*RepositoryRelease, *Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases/tags/%s", , , )
	return .getSingleRelease(, )
}

// GenerateReleaseNotes generates the release notes for the given tag.
//
// GitHub API docs: https://docs.github.com/rest/releases/releases#generate-release-notes-content-for-a-release
//
//meta:operation POST /repos/{owner}/{repo}/releases/generate-notes
func ( *RepositoriesService) ( context.Context, ,  string,  *GenerateNotesOptions) (*RepositoryReleaseNotes, *Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases/generate-notes", , )
	,  := .client.NewRequest("POST", , )
	if  != nil {
		return nil, nil, 
	}

	 := new(RepositoryReleaseNotes)
	,  := .client.Do(, , )
	if  != nil {
		return nil, , 
	}

	return , , nil
}

func ( *RepositoriesService) ( context.Context,  string) (*RepositoryRelease, *Response, error) {
	,  := .client.NewRequest("GET", , nil)
	if  != nil {
		return nil, nil, 
	}

	 := new(RepositoryRelease)
	,  := .client.Do(, , )
	if  != nil {
		return nil, , 
	}
	return , , nil
}

// repositoryReleaseRequest is a subset of RepositoryRelease and
// is used internally by CreateRelease and EditRelease to pass
// only the known fields for these endpoints.
//
// See https://github.com/google/go-github/issues/992 for more
// information.
type repositoryReleaseRequest struct {
	TagName                *string `json:"tag_name,omitempty"`
	TargetCommitish        *string `json:"target_commitish,omitempty"`
	Name                   *string `json:"name,omitempty"`
	Body                   *string `json:"body,omitempty"`
	Draft                  *bool   `json:"draft,omitempty"`
	Prerelease             *bool   `json:"prerelease,omitempty"`
	MakeLatest             *string `json:"make_latest,omitempty"`
	GenerateReleaseNotes   *bool   `json:"generate_release_notes,omitempty"`
	DiscussionCategoryName *string `json:"discussion_category_name,omitempty"`
}

// CreateRelease adds a new release for a repository.
//
// Note that only a subset of the release fields are used.
// See RepositoryRelease for more information.
//
// GitHub API docs: https://docs.github.com/rest/releases/releases#create-a-release
//
//meta:operation POST /repos/{owner}/{repo}/releases
func ( *RepositoriesService) ( context.Context, ,  string,  *RepositoryRelease) (*RepositoryRelease, *Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases", , )

	 := &repositoryReleaseRequest{
		TagName:                .TagName,
		TargetCommitish:        .TargetCommitish,
		Name:                   .Name,
		Body:                   .Body,
		Draft:                  .Draft,
		Prerelease:             .Prerelease,
		MakeLatest:             .MakeLatest,
		DiscussionCategoryName: .DiscussionCategoryName,
		GenerateReleaseNotes:   .GenerateReleaseNotes,
	}

	,  := .client.NewRequest("POST", , )
	if  != nil {
		return nil, nil, 
	}

	 := new(RepositoryRelease)
	,  := .client.Do(, , )
	if  != nil {
		return nil, , 
	}
	return , , nil
}

// EditRelease edits a repository release.
//
// Note that only a subset of the release fields are used.
// See RepositoryRelease for more information.
//
// GitHub API docs: https://docs.github.com/rest/releases/releases#update-a-release
//
//meta:operation PATCH /repos/{owner}/{repo}/releases/{release_id}
func ( *RepositoriesService) ( context.Context, ,  string,  int64,  *RepositoryRelease) (*RepositoryRelease, *Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases/%d", , , )

	 := &repositoryReleaseRequest{
		TagName:                .TagName,
		TargetCommitish:        .TargetCommitish,
		Name:                   .Name,
		Body:                   .Body,
		Draft:                  .Draft,
		Prerelease:             .Prerelease,
		MakeLatest:             .MakeLatest,
		DiscussionCategoryName: .DiscussionCategoryName,
	}

	,  := .client.NewRequest("PATCH", , )
	if  != nil {
		return nil, nil, 
	}

	 := new(RepositoryRelease)
	,  := .client.Do(, , )
	if  != nil {
		return nil, , 
	}
	return , , nil
}

// DeleteRelease delete a single release from a repository.
//
// GitHub API docs: https://docs.github.com/rest/releases/releases#delete-a-release
//
//meta:operation DELETE /repos/{owner}/{repo}/releases/{release_id}
func ( *RepositoriesService) ( context.Context, ,  string,  int64) (*Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases/%d", , , )

	,  := .client.NewRequest("DELETE", , nil)
	if  != nil {
		return nil, 
	}
	return .client.Do(, , nil)
}

// ListReleaseAssets lists the release's assets.
//
// GitHub API docs: https://docs.github.com/rest/releases/assets#list-release-assets
//
//meta:operation GET /repos/{owner}/{repo}/releases/{release_id}/assets
func ( *RepositoriesService) ( context.Context, ,  string,  int64,  *ListOptions) ([]*ReleaseAsset, *Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases/%d/assets", , , )
	,  := addOptions(, )
	if  != nil {
		return nil, nil, 
	}

	,  := .client.NewRequest("GET", , nil)
	if  != nil {
		return nil, nil, 
	}

	var  []*ReleaseAsset
	,  := .client.Do(, , &)
	if  != nil {
		return nil, , 
	}
	return , , nil
}

// GetReleaseAsset fetches a single release asset.
//
// GitHub API docs: https://docs.github.com/rest/releases/assets#get-a-release-asset
//
//meta:operation GET /repos/{owner}/{repo}/releases/assets/{asset_id}
func ( *RepositoriesService) ( context.Context, ,  string,  int64) (*ReleaseAsset, *Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases/assets/%d", , , )

	,  := .client.NewRequest("GET", , nil)
	if  != nil {
		return nil, nil, 
	}

	 := new(ReleaseAsset)
	,  := .client.Do(, , )
	if  != nil {
		return nil, , 
	}
	return , , nil
}

// DownloadReleaseAsset downloads a release asset or returns a redirect URL.
//
// DownloadReleaseAsset returns an io.ReadCloser that reads the contents of the
// specified release asset. It is the caller's responsibility to close the ReadCloser.
// If a redirect is returned, the redirect URL will be returned as a string instead
// of the io.ReadCloser. Exactly one of rc and redirectURL will be zero.
//
// followRedirectsClient can be passed to download the asset from a redirected
// location. Passing http.DefaultClient is recommended unless special circumstances
// exist, but it's possible to pass any http.Client. If nil is passed the
// redirectURL will be returned instead.
//
// GitHub API docs: https://docs.github.com/rest/releases/assets#get-a-release-asset
//
//meta:operation GET /repos/{owner}/{repo}/releases/assets/{asset_id}
func ( *RepositoriesService) ( context.Context, ,  string,  int64,  *http.Client) ( io.ReadCloser,  string,  error) {
	 := fmt.Sprintf("repos/%s/%s/releases/assets/%d", , , )

	,  := .client.NewRequest("GET", , nil)
	if  != nil {
		return nil, "", 
	}
	.Header.Set("Accept", defaultMediaType)

	.client.clientMu.Lock()
	defer .client.clientMu.Unlock()

	var  string
	 := .client.client.CheckRedirect
	.client.client.CheckRedirect = func( *http.Request,  []*http.Request) error {
		 = .URL.String()
		return errors.New("disable redirect")
	}
	defer func() { .client.client.CheckRedirect =  }()

	 = withContext(, )
	,  := .client.client.Do()
	if  != nil {
		if !strings.Contains(.Error(), "disable redirect") {
			return nil, "", 
		}
		if  != nil {
			,  := .downloadReleaseAssetFromURL(, , )
			return , "", 
		}
		return nil, , nil // Intentionally return no error with valid redirect URL.
	}

	if  := CheckResponse();  != nil {
		_ = .Body.Close()
		return nil, "", 
	}

	return .Body, "", nil
}

func ( *RepositoriesService) ( context.Context,  *http.Client,  string) ( io.ReadCloser,  error) {
	,  := http.NewRequest("GET", , nil)
	if  != nil {
		return nil, 
	}
	 = withContext(, )
	.Header.Set("Accept", "*/*")
	,  := .Do()
	if  != nil {
		return nil, 
	}
	if  := CheckResponse();  != nil {
		_ = .Body.Close()
		return nil, 
	}
	return .Body, nil
}

// EditReleaseAsset edits a repository release asset.
//
// GitHub API docs: https://docs.github.com/rest/releases/assets#update-a-release-asset
//
//meta:operation PATCH /repos/{owner}/{repo}/releases/assets/{asset_id}
func ( *RepositoriesService) ( context.Context, ,  string,  int64,  *ReleaseAsset) (*ReleaseAsset, *Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases/assets/%d", , , )

	,  := .client.NewRequest("PATCH", , )
	if  != nil {
		return nil, nil, 
	}

	 := new(ReleaseAsset)
	,  := .client.Do(, , )
	if  != nil {
		return nil, , 
	}
	return , , nil
}

// DeleteReleaseAsset delete a single release asset from a repository.
//
// GitHub API docs: https://docs.github.com/rest/releases/assets#delete-a-release-asset
//
//meta:operation DELETE /repos/{owner}/{repo}/releases/assets/{asset_id}
func ( *RepositoriesService) ( context.Context, ,  string,  int64) (*Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases/assets/%d", , , )

	,  := .client.NewRequest("DELETE", , nil)
	if  != nil {
		return nil, 
	}
	return .client.Do(, , nil)
}

// UploadReleaseAsset creates an asset by uploading a file into a release repository.
// To upload assets that cannot be represented by an os.File, call NewUploadRequest directly.
//
// GitHub API docs: https://docs.github.com/rest/releases/assets#upload-a-release-asset
//
//meta:operation POST /repos/{owner}/{repo}/releases/{release_id}/assets
func ( *RepositoriesService) ( context.Context, ,  string,  int64,  *UploadOptions,  *os.File) (*ReleaseAsset, *Response, error) {
	 := fmt.Sprintf("repos/%s/%s/releases/%d/assets", , , )
	,  := addOptions(, )
	if  != nil {
		return nil, nil, 
	}

	,  := .Stat()
	if  != nil {
		return nil, nil, 
	}
	if .IsDir() {
		return nil, nil, errors.New("the asset to upload can't be a directory")
	}

	 := mime.TypeByExtension(filepath.Ext(.Name()))
	if .MediaType != "" {
		 = .MediaType
	}

	,  := .client.NewUploadRequest(, , .Size(), )
	if  != nil {
		return nil, nil, 
	}

	 := new(ReleaseAsset)
	,  := .client.Do(, , )
	if  != nil {
		return nil, , 
	}
	return , , nil
}