github.com/google/go-github/v65@v65.0.0/github/repos_releases.go (about)

     1  // Copyright 2013 The go-github AUTHORS. All rights reserved.
     2  //
     3  // Use of this source code is governed by a BSD-style
     4  // license that can be found in the LICENSE file.
     5  
     6  package github
     7  
     8  import (
     9  	"context"
    10  	"errors"
    11  	"fmt"
    12  	"io"
    13  	"mime"
    14  	"net/http"
    15  	"os"
    16  	"path/filepath"
    17  	"strings"
    18  )
    19  
    20  // RepositoryRelease represents a GitHub release in a repository.
    21  type RepositoryRelease struct {
    22  	TagName         *string `json:"tag_name,omitempty"`
    23  	TargetCommitish *string `json:"target_commitish,omitempty"`
    24  	Name            *string `json:"name,omitempty"`
    25  	Body            *string `json:"body,omitempty"`
    26  	Draft           *bool   `json:"draft,omitempty"`
    27  	Prerelease      *bool   `json:"prerelease,omitempty"`
    28  	// MakeLatest can be one of: "true", "false", or "legacy".
    29  	MakeLatest             *string `json:"make_latest,omitempty"`
    30  	DiscussionCategoryName *string `json:"discussion_category_name,omitempty"`
    31  
    32  	// The following fields are not used in EditRelease:
    33  	GenerateReleaseNotes *bool `json:"generate_release_notes,omitempty"`
    34  
    35  	// The following fields are not used in CreateRelease or EditRelease:
    36  	ID          *int64          `json:"id,omitempty"`
    37  	CreatedAt   *Timestamp      `json:"created_at,omitempty"`
    38  	PublishedAt *Timestamp      `json:"published_at,omitempty"`
    39  	URL         *string         `json:"url,omitempty"`
    40  	HTMLURL     *string         `json:"html_url,omitempty"`
    41  	AssetsURL   *string         `json:"assets_url,omitempty"`
    42  	Assets      []*ReleaseAsset `json:"assets,omitempty"`
    43  	UploadURL   *string         `json:"upload_url,omitempty"`
    44  	ZipballURL  *string         `json:"zipball_url,omitempty"`
    45  	TarballURL  *string         `json:"tarball_url,omitempty"`
    46  	Author      *User           `json:"author,omitempty"`
    47  	NodeID      *string         `json:"node_id,omitempty"`
    48  }
    49  
    50  func (r RepositoryRelease) String() string {
    51  	return Stringify(r)
    52  }
    53  
    54  // RepositoryReleaseNotes represents a GitHub-generated release notes.
    55  type RepositoryReleaseNotes struct {
    56  	Name string `json:"name"`
    57  	Body string `json:"body"`
    58  }
    59  
    60  // GenerateNotesOptions represents the options to generate release notes.
    61  type GenerateNotesOptions struct {
    62  	TagName         string  `json:"tag_name"`
    63  	PreviousTagName *string `json:"previous_tag_name,omitempty"`
    64  	TargetCommitish *string `json:"target_commitish,omitempty"`
    65  }
    66  
    67  // ReleaseAsset represents a GitHub release asset in a repository.
    68  type ReleaseAsset struct {
    69  	ID                 *int64     `json:"id,omitempty"`
    70  	URL                *string    `json:"url,omitempty"`
    71  	Name               *string    `json:"name,omitempty"`
    72  	Label              *string    `json:"label,omitempty"`
    73  	State              *string    `json:"state,omitempty"`
    74  	ContentType        *string    `json:"content_type,omitempty"`
    75  	Size               *int       `json:"size,omitempty"`
    76  	DownloadCount      *int       `json:"download_count,omitempty"`
    77  	CreatedAt          *Timestamp `json:"created_at,omitempty"`
    78  	UpdatedAt          *Timestamp `json:"updated_at,omitempty"`
    79  	BrowserDownloadURL *string    `json:"browser_download_url,omitempty"`
    80  	Uploader           *User      `json:"uploader,omitempty"`
    81  	NodeID             *string    `json:"node_id,omitempty"`
    82  }
    83  
    84  func (r ReleaseAsset) String() string {
    85  	return Stringify(r)
    86  }
    87  
    88  // ListReleases lists the releases for a repository.
    89  //
    90  // GitHub API docs: https://docs.github.com/rest/releases/releases#list-releases
    91  //
    92  //meta:operation GET /repos/{owner}/{repo}/releases
    93  func (s *RepositoriesService) ListReleases(ctx context.Context, owner, repo string, opts *ListOptions) ([]*RepositoryRelease, *Response, error) {
    94  	u := fmt.Sprintf("repos/%s/%s/releases", owner, repo)
    95  	u, err := addOptions(u, opts)
    96  	if err != nil {
    97  		return nil, nil, err
    98  	}
    99  
   100  	req, err := s.client.NewRequest("GET", u, nil)
   101  	if err != nil {
   102  		return nil, nil, err
   103  	}
   104  
   105  	var releases []*RepositoryRelease
   106  	resp, err := s.client.Do(ctx, req, &releases)
   107  	if err != nil {
   108  		return nil, resp, err
   109  	}
   110  	return releases, resp, nil
   111  }
   112  
   113  // GetRelease fetches a single release.
   114  //
   115  // GitHub API docs: https://docs.github.com/rest/releases/releases#get-a-release
   116  //
   117  //meta:operation GET /repos/{owner}/{repo}/releases/{release_id}
   118  func (s *RepositoriesService) GetRelease(ctx context.Context, owner, repo string, id int64) (*RepositoryRelease, *Response, error) {
   119  	u := fmt.Sprintf("repos/%s/%s/releases/%d", owner, repo, id)
   120  	return s.getSingleRelease(ctx, u)
   121  }
   122  
   123  // GetLatestRelease fetches the latest published release for the repository.
   124  //
   125  // GitHub API docs: https://docs.github.com/rest/releases/releases#get-the-latest-release
   126  //
   127  //meta:operation GET /repos/{owner}/{repo}/releases/latest
   128  func (s *RepositoriesService) GetLatestRelease(ctx context.Context, owner, repo string) (*RepositoryRelease, *Response, error) {
   129  	u := fmt.Sprintf("repos/%s/%s/releases/latest", owner, repo)
   130  	return s.getSingleRelease(ctx, u)
   131  }
   132  
   133  // GetReleaseByTag fetches a release with the specified tag.
   134  //
   135  // GitHub API docs: https://docs.github.com/rest/releases/releases#get-a-release-by-tag-name
   136  //
   137  //meta:operation GET /repos/{owner}/{repo}/releases/tags/{tag}
   138  func (s *RepositoriesService) GetReleaseByTag(ctx context.Context, owner, repo, tag string) (*RepositoryRelease, *Response, error) {
   139  	u := fmt.Sprintf("repos/%s/%s/releases/tags/%s", owner, repo, tag)
   140  	return s.getSingleRelease(ctx, u)
   141  }
   142  
   143  // GenerateReleaseNotes generates the release notes for the given tag.
   144  //
   145  // GitHub API docs: https://docs.github.com/rest/releases/releases#generate-release-notes-content-for-a-release
   146  //
   147  //meta:operation POST /repos/{owner}/{repo}/releases/generate-notes
   148  func (s *RepositoriesService) GenerateReleaseNotes(ctx context.Context, owner, repo string, opts *GenerateNotesOptions) (*RepositoryReleaseNotes, *Response, error) {
   149  	u := fmt.Sprintf("repos/%s/%s/releases/generate-notes", owner, repo)
   150  	req, err := s.client.NewRequest("POST", u, opts)
   151  	if err != nil {
   152  		return nil, nil, err
   153  	}
   154  
   155  	r := new(RepositoryReleaseNotes)
   156  	resp, err := s.client.Do(ctx, req, r)
   157  	if err != nil {
   158  		return nil, resp, err
   159  	}
   160  
   161  	return r, resp, nil
   162  }
   163  
   164  func (s *RepositoriesService) getSingleRelease(ctx context.Context, url string) (*RepositoryRelease, *Response, error) {
   165  	req, err := s.client.NewRequest("GET", url, nil)
   166  	if err != nil {
   167  		return nil, nil, err
   168  	}
   169  
   170  	release := new(RepositoryRelease)
   171  	resp, err := s.client.Do(ctx, req, release)
   172  	if err != nil {
   173  		return nil, resp, err
   174  	}
   175  	return release, resp, nil
   176  }
   177  
   178  // repositoryReleaseRequest is a subset of RepositoryRelease and
   179  // is used internally by CreateRelease and EditRelease to pass
   180  // only the known fields for these endpoints.
   181  //
   182  // See https://github.com/google/go-github/issues/992 for more
   183  // information.
   184  type repositoryReleaseRequest struct {
   185  	TagName                *string `json:"tag_name,omitempty"`
   186  	TargetCommitish        *string `json:"target_commitish,omitempty"`
   187  	Name                   *string `json:"name,omitempty"`
   188  	Body                   *string `json:"body,omitempty"`
   189  	Draft                  *bool   `json:"draft,omitempty"`
   190  	Prerelease             *bool   `json:"prerelease,omitempty"`
   191  	MakeLatest             *string `json:"make_latest,omitempty"`
   192  	GenerateReleaseNotes   *bool   `json:"generate_release_notes,omitempty"`
   193  	DiscussionCategoryName *string `json:"discussion_category_name,omitempty"`
   194  }
   195  
   196  // CreateRelease adds a new release for a repository.
   197  //
   198  // Note that only a subset of the release fields are used.
   199  // See RepositoryRelease for more information.
   200  //
   201  // GitHub API docs: https://docs.github.com/rest/releases/releases#create-a-release
   202  //
   203  //meta:operation POST /repos/{owner}/{repo}/releases
   204  func (s *RepositoriesService) CreateRelease(ctx context.Context, owner, repo string, release *RepositoryRelease) (*RepositoryRelease, *Response, error) {
   205  	u := fmt.Sprintf("repos/%s/%s/releases", owner, repo)
   206  
   207  	releaseReq := &repositoryReleaseRequest{
   208  		TagName:                release.TagName,
   209  		TargetCommitish:        release.TargetCommitish,
   210  		Name:                   release.Name,
   211  		Body:                   release.Body,
   212  		Draft:                  release.Draft,
   213  		Prerelease:             release.Prerelease,
   214  		MakeLatest:             release.MakeLatest,
   215  		DiscussionCategoryName: release.DiscussionCategoryName,
   216  		GenerateReleaseNotes:   release.GenerateReleaseNotes,
   217  	}
   218  
   219  	req, err := s.client.NewRequest("POST", u, releaseReq)
   220  	if err != nil {
   221  		return nil, nil, err
   222  	}
   223  
   224  	r := new(RepositoryRelease)
   225  	resp, err := s.client.Do(ctx, req, r)
   226  	if err != nil {
   227  		return nil, resp, err
   228  	}
   229  	return r, resp, nil
   230  }
   231  
   232  // EditRelease edits a repository release.
   233  //
   234  // Note that only a subset of the release fields are used.
   235  // See RepositoryRelease for more information.
   236  //
   237  // GitHub API docs: https://docs.github.com/rest/releases/releases#update-a-release
   238  //
   239  //meta:operation PATCH /repos/{owner}/{repo}/releases/{release_id}
   240  func (s *RepositoriesService) EditRelease(ctx context.Context, owner, repo string, id int64, release *RepositoryRelease) (*RepositoryRelease, *Response, error) {
   241  	u := fmt.Sprintf("repos/%s/%s/releases/%d", owner, repo, id)
   242  
   243  	releaseReq := &repositoryReleaseRequest{
   244  		TagName:                release.TagName,
   245  		TargetCommitish:        release.TargetCommitish,
   246  		Name:                   release.Name,
   247  		Body:                   release.Body,
   248  		Draft:                  release.Draft,
   249  		Prerelease:             release.Prerelease,
   250  		MakeLatest:             release.MakeLatest,
   251  		DiscussionCategoryName: release.DiscussionCategoryName,
   252  	}
   253  
   254  	req, err := s.client.NewRequest("PATCH", u, releaseReq)
   255  	if err != nil {
   256  		return nil, nil, err
   257  	}
   258  
   259  	r := new(RepositoryRelease)
   260  	resp, err := s.client.Do(ctx, req, r)
   261  	if err != nil {
   262  		return nil, resp, err
   263  	}
   264  	return r, resp, nil
   265  }
   266  
   267  // DeleteRelease delete a single release from a repository.
   268  //
   269  // GitHub API docs: https://docs.github.com/rest/releases/releases#delete-a-release
   270  //
   271  //meta:operation DELETE /repos/{owner}/{repo}/releases/{release_id}
   272  func (s *RepositoriesService) DeleteRelease(ctx context.Context, owner, repo string, id int64) (*Response, error) {
   273  	u := fmt.Sprintf("repos/%s/%s/releases/%d", owner, repo, id)
   274  
   275  	req, err := s.client.NewRequest("DELETE", u, nil)
   276  	if err != nil {
   277  		return nil, err
   278  	}
   279  	return s.client.Do(ctx, req, nil)
   280  }
   281  
   282  // ListReleaseAssets lists the release's assets.
   283  //
   284  // GitHub API docs: https://docs.github.com/rest/releases/assets#list-release-assets
   285  //
   286  //meta:operation GET /repos/{owner}/{repo}/releases/{release_id}/assets
   287  func (s *RepositoriesService) ListReleaseAssets(ctx context.Context, owner, repo string, id int64, opts *ListOptions) ([]*ReleaseAsset, *Response, error) {
   288  	u := fmt.Sprintf("repos/%s/%s/releases/%d/assets", owner, repo, id)
   289  	u, err := addOptions(u, opts)
   290  	if err != nil {
   291  		return nil, nil, err
   292  	}
   293  
   294  	req, err := s.client.NewRequest("GET", u, nil)
   295  	if err != nil {
   296  		return nil, nil, err
   297  	}
   298  
   299  	var assets []*ReleaseAsset
   300  	resp, err := s.client.Do(ctx, req, &assets)
   301  	if err != nil {
   302  		return nil, resp, err
   303  	}
   304  	return assets, resp, nil
   305  }
   306  
   307  // GetReleaseAsset fetches a single release asset.
   308  //
   309  // GitHub API docs: https://docs.github.com/rest/releases/assets#get-a-release-asset
   310  //
   311  //meta:operation GET /repos/{owner}/{repo}/releases/assets/{asset_id}
   312  func (s *RepositoriesService) GetReleaseAsset(ctx context.Context, owner, repo string, id int64) (*ReleaseAsset, *Response, error) {
   313  	u := fmt.Sprintf("repos/%s/%s/releases/assets/%d", owner, repo, id)
   314  
   315  	req, err := s.client.NewRequest("GET", u, nil)
   316  	if err != nil {
   317  		return nil, nil, err
   318  	}
   319  
   320  	asset := new(ReleaseAsset)
   321  	resp, err := s.client.Do(ctx, req, asset)
   322  	if err != nil {
   323  		return nil, resp, err
   324  	}
   325  	return asset, resp, nil
   326  }
   327  
   328  // DownloadReleaseAsset downloads a release asset or returns a redirect URL.
   329  //
   330  // DownloadReleaseAsset returns an io.ReadCloser that reads the contents of the
   331  // specified release asset. It is the caller's responsibility to close the ReadCloser.
   332  // If a redirect is returned, the redirect URL will be returned as a string instead
   333  // of the io.ReadCloser. Exactly one of rc and redirectURL will be zero.
   334  //
   335  // followRedirectsClient can be passed to download the asset from a redirected
   336  // location. Passing http.DefaultClient is recommended unless special circumstances
   337  // exist, but it's possible to pass any http.Client. If nil is passed the
   338  // redirectURL will be returned instead.
   339  //
   340  // GitHub API docs: https://docs.github.com/rest/releases/assets#get-a-release-asset
   341  //
   342  //meta:operation GET /repos/{owner}/{repo}/releases/assets/{asset_id}
   343  func (s *RepositoriesService) DownloadReleaseAsset(ctx context.Context, owner, repo string, id int64, followRedirectsClient *http.Client) (rc io.ReadCloser, redirectURL string, err error) {
   344  	u := fmt.Sprintf("repos/%s/%s/releases/assets/%d", owner, repo, id)
   345  
   346  	req, err := s.client.NewRequest("GET", u, nil)
   347  	if err != nil {
   348  		return nil, "", err
   349  	}
   350  	req.Header.Set("Accept", defaultMediaType)
   351  
   352  	s.client.clientMu.Lock()
   353  	defer s.client.clientMu.Unlock()
   354  
   355  	var loc string
   356  	saveRedirect := s.client.client.CheckRedirect
   357  	s.client.client.CheckRedirect = func(req *http.Request, via []*http.Request) error {
   358  		loc = req.URL.String()
   359  		return errors.New("disable redirect")
   360  	}
   361  	defer func() { s.client.client.CheckRedirect = saveRedirect }()
   362  
   363  	req = withContext(ctx, req)
   364  	resp, err := s.client.client.Do(req)
   365  	if err != nil {
   366  		if !strings.Contains(err.Error(), "disable redirect") {
   367  			return nil, "", err
   368  		}
   369  		if followRedirectsClient != nil {
   370  			rc, err := s.downloadReleaseAssetFromURL(ctx, followRedirectsClient, loc)
   371  			return rc, "", err
   372  		}
   373  		return nil, loc, nil // Intentionally return no error with valid redirect URL.
   374  	}
   375  
   376  	if err := CheckResponse(resp); err != nil {
   377  		_ = resp.Body.Close()
   378  		return nil, "", err
   379  	}
   380  
   381  	return resp.Body, "", nil
   382  }
   383  
   384  func (s *RepositoriesService) downloadReleaseAssetFromURL(ctx context.Context, followRedirectsClient *http.Client, url string) (rc io.ReadCloser, err error) {
   385  	req, err := http.NewRequest("GET", url, nil)
   386  	if err != nil {
   387  		return nil, err
   388  	}
   389  	req = withContext(ctx, req)
   390  	req.Header.Set("Accept", "*/*")
   391  	resp, err := followRedirectsClient.Do(req)
   392  	if err != nil {
   393  		return nil, err
   394  	}
   395  	if err := CheckResponse(resp); err != nil {
   396  		_ = resp.Body.Close()
   397  		return nil, err
   398  	}
   399  	return resp.Body, nil
   400  }
   401  
   402  // EditReleaseAsset edits a repository release asset.
   403  //
   404  // GitHub API docs: https://docs.github.com/rest/releases/assets#update-a-release-asset
   405  //
   406  //meta:operation PATCH /repos/{owner}/{repo}/releases/assets/{asset_id}
   407  func (s *RepositoriesService) EditReleaseAsset(ctx context.Context, owner, repo string, id int64, release *ReleaseAsset) (*ReleaseAsset, *Response, error) {
   408  	u := fmt.Sprintf("repos/%s/%s/releases/assets/%d", owner, repo, id)
   409  
   410  	req, err := s.client.NewRequest("PATCH", u, release)
   411  	if err != nil {
   412  		return nil, nil, err
   413  	}
   414  
   415  	asset := new(ReleaseAsset)
   416  	resp, err := s.client.Do(ctx, req, asset)
   417  	if err != nil {
   418  		return nil, resp, err
   419  	}
   420  	return asset, resp, nil
   421  }
   422  
   423  // DeleteReleaseAsset delete a single release asset from a repository.
   424  //
   425  // GitHub API docs: https://docs.github.com/rest/releases/assets#delete-a-release-asset
   426  //
   427  //meta:operation DELETE /repos/{owner}/{repo}/releases/assets/{asset_id}
   428  func (s *RepositoriesService) DeleteReleaseAsset(ctx context.Context, owner, repo string, id int64) (*Response, error) {
   429  	u := fmt.Sprintf("repos/%s/%s/releases/assets/%d", owner, repo, id)
   430  
   431  	req, err := s.client.NewRequest("DELETE", u, nil)
   432  	if err != nil {
   433  		return nil, err
   434  	}
   435  	return s.client.Do(ctx, req, nil)
   436  }
   437  
   438  // UploadReleaseAsset creates an asset by uploading a file into a release repository.
   439  // To upload assets that cannot be represented by an os.File, call NewUploadRequest directly.
   440  //
   441  // GitHub API docs: https://docs.github.com/rest/releases/assets#upload-a-release-asset
   442  //
   443  //meta:operation POST /repos/{owner}/{repo}/releases/{release_id}/assets
   444  func (s *RepositoriesService) UploadReleaseAsset(ctx context.Context, owner, repo string, id int64, opts *UploadOptions, file *os.File) (*ReleaseAsset, *Response, error) {
   445  	u := fmt.Sprintf("repos/%s/%s/releases/%d/assets", owner, repo, id)
   446  	u, err := addOptions(u, opts)
   447  	if err != nil {
   448  		return nil, nil, err
   449  	}
   450  
   451  	stat, err := file.Stat()
   452  	if err != nil {
   453  		return nil, nil, err
   454  	}
   455  	if stat.IsDir() {
   456  		return nil, nil, errors.New("the asset to upload can't be a directory")
   457  	}
   458  
   459  	mediaType := mime.TypeByExtension(filepath.Ext(file.Name()))
   460  	if opts.MediaType != "" {
   461  		mediaType = opts.MediaType
   462  	}
   463  
   464  	req, err := s.client.NewUploadRequest(u, file, stat.Size(), mediaType)
   465  	if err != nil {
   466  		return nil, nil, err
   467  	}
   468  
   469  	asset := new(ReleaseAsset)
   470  	resp, err := s.client.Do(ctx, req, asset)
   471  	if err != nil {
   472  		return nil, resp, err
   473  	}
   474  	return asset, resp, nil
   475  }