2019-05-07 09:12:51 +08:00
|
|
|
// 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.
|
|
|
|
|
|
|
|
// Repository contents API methods.
|
2021-08-10 17:49:43 +08:00
|
|
|
// GitHub API docs: https://docs.github.com/en/free-pro-team@latest/rest/reference/repos/contents/
|
2019-05-07 09:12:51 +08:00
|
|
|
|
|
|
|
package github
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/base64"
|
|
|
|
"encoding/json"
|
2020-07-31 22:22:34 +08:00
|
|
|
"errors"
|
2019-05-07 09:12:51 +08:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"path"
|
2021-08-10 17:49:43 +08:00
|
|
|
"strings"
|
2019-05-07 09:12:51 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
// RepositoryContent represents a file or directory in a github repository.
|
|
|
|
type RepositoryContent struct {
|
|
|
|
Type *string `json:"type,omitempty"`
|
|
|
|
// Target is only set if the type is "symlink" and the target is not a normal file.
|
|
|
|
// If Target is set, Path will be the symlink path.
|
|
|
|
Target *string `json:"target,omitempty"`
|
|
|
|
Encoding *string `json:"encoding,omitempty"`
|
|
|
|
Size *int `json:"size,omitempty"`
|
|
|
|
Name *string `json:"name,omitempty"`
|
|
|
|
Path *string `json:"path,omitempty"`
|
|
|
|
// Content contains the actual file content, which may be encoded.
|
|
|
|
// Callers should call GetContent which will decode the content if
|
|
|
|
// necessary.
|
|
|
|
Content *string `json:"content,omitempty"`
|
|
|
|
SHA *string `json:"sha,omitempty"`
|
|
|
|
URL *string `json:"url,omitempty"`
|
|
|
|
GitURL *string `json:"git_url,omitempty"`
|
|
|
|
HTMLURL *string `json:"html_url,omitempty"`
|
|
|
|
DownloadURL *string `json:"download_url,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// RepositoryContentResponse holds the parsed response from CreateFile, UpdateFile, and DeleteFile.
|
|
|
|
type RepositoryContentResponse struct {
|
|
|
|
Content *RepositoryContent `json:"content,omitempty"`
|
|
|
|
Commit `json:"commit,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// RepositoryContentFileOptions specifies optional parameters for CreateFile, UpdateFile, and DeleteFile.
|
|
|
|
type RepositoryContentFileOptions struct {
|
|
|
|
Message *string `json:"message,omitempty"`
|
|
|
|
Content []byte `json:"content,omitempty"` // unencoded
|
|
|
|
SHA *string `json:"sha,omitempty"`
|
|
|
|
Branch *string `json:"branch,omitempty"`
|
|
|
|
Author *CommitAuthor `json:"author,omitempty"`
|
|
|
|
Committer *CommitAuthor `json:"committer,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// RepositoryContentGetOptions represents an optional ref parameter, which can be a SHA,
|
|
|
|
// branch, or tag
|
|
|
|
type RepositoryContentGetOptions struct {
|
|
|
|
Ref string `url:"ref,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// String converts RepositoryContent to a string. It's primarily for testing.
|
|
|
|
func (r RepositoryContent) String() string {
|
|
|
|
return Stringify(r)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetContent returns the content of r, decoding it if necessary.
|
|
|
|
func (r *RepositoryContent) GetContent() (string, error) {
|
|
|
|
var encoding string
|
|
|
|
if r.Encoding != nil {
|
|
|
|
encoding = *r.Encoding
|
|
|
|
}
|
|
|
|
|
|
|
|
switch encoding {
|
|
|
|
case "base64":
|
2020-07-31 22:22:34 +08:00
|
|
|
if r.Content == nil {
|
|
|
|
return "", errors.New("malformed response: base64 encoding of null content")
|
|
|
|
}
|
2019-05-07 09:12:51 +08:00
|
|
|
c, err := base64.StdEncoding.DecodeString(*r.Content)
|
|
|
|
return string(c), err
|
|
|
|
case "":
|
|
|
|
if r.Content == nil {
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
return *r.Content, nil
|
|
|
|
default:
|
|
|
|
return "", fmt.Errorf("unsupported content encoding: %v", encoding)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetReadme gets the Readme file for the repository.
|
|
|
|
//
|
2021-08-10 17:49:43 +08:00
|
|
|
// GitHub API docs: https://docs.github.com/en/free-pro-team@latest/rest/reference/repos/#get-a-repository-readme
|
2020-07-31 22:22:34 +08:00
|
|
|
func (s *RepositoriesService) GetReadme(ctx context.Context, owner, repo string, opts *RepositoryContentGetOptions) (*RepositoryContent, *Response, error) {
|
2019-05-07 09:12:51 +08:00
|
|
|
u := fmt.Sprintf("repos/%v/%v/readme", owner, repo)
|
2020-07-31 22:22:34 +08:00
|
|
|
u, err := addOptions(u, opts)
|
2019-05-07 09:12:51 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
req, err := s.client.NewRequest("GET", u, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
readme := new(RepositoryContent)
|
|
|
|
resp, err := s.client.Do(ctx, req, readme)
|
|
|
|
if err != nil {
|
|
|
|
return nil, resp, err
|
|
|
|
}
|
|
|
|
return readme, resp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DownloadContents returns an io.ReadCloser that reads the contents of the
|
|
|
|
// specified file. This function will work with files of any size, as opposed
|
|
|
|
// to GetContents which is limited to 1 Mb files. It is the caller's
|
|
|
|
// responsibility to close the ReadCloser.
|
2021-08-10 17:49:43 +08:00
|
|
|
//
|
|
|
|
// It is possible for the download to result in a failed response when the
|
|
|
|
// returned error is nil. Callers should check the returned Response status
|
|
|
|
// code to verify the content is from a successful response.
|
|
|
|
func (s *RepositoriesService) DownloadContents(ctx context.Context, owner, repo, filepath string, opts *RepositoryContentGetOptions) (io.ReadCloser, *Response, error) {
|
|
|
|
dir := path.Dir(filepath)
|
|
|
|
filename := path.Base(filepath)
|
|
|
|
_, dirContents, resp, err := s.GetContents(ctx, owner, repo, dir, opts)
|
|
|
|
if err != nil {
|
|
|
|
return nil, resp, err
|
|
|
|
}
|
|
|
|
for _, contents := range dirContents {
|
|
|
|
if *contents.Name == filename {
|
|
|
|
if contents.DownloadURL == nil || *contents.DownloadURL == "" {
|
|
|
|
return nil, resp, fmt.Errorf("No download link found for %s", filepath)
|
|
|
|
}
|
|
|
|
dlResp, err := s.client.client.Get(*contents.DownloadURL)
|
|
|
|
if err != nil {
|
|
|
|
return nil, &Response{Response: dlResp}, err
|
|
|
|
}
|
|
|
|
return dlResp.Body, &Response{Response: dlResp}, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, resp, fmt.Errorf("No file named %s found in %s", filename, dir)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DownloadContentsWithMeta is identical to DownloadContents but additionally
|
|
|
|
// returns the RepositoryContent of the requested file. This additional data
|
|
|
|
// is useful for future operations involving the requested file. For merely
|
|
|
|
// reading the content of a file, DownloadContents is perfectly adequate.
|
|
|
|
//
|
|
|
|
// It is possible for the download to result in a failed response when the
|
|
|
|
// returned error is nil. Callers should check the returned Response status
|
|
|
|
// code to verify the content is from a successful response.
|
|
|
|
func (s *RepositoriesService) DownloadContentsWithMeta(ctx context.Context, owner, repo, filepath string, opts *RepositoryContentGetOptions) (io.ReadCloser, *RepositoryContent, *Response, error) {
|
2019-05-07 09:12:51 +08:00
|
|
|
dir := path.Dir(filepath)
|
|
|
|
filename := path.Base(filepath)
|
2021-08-10 17:49:43 +08:00
|
|
|
_, dirContents, resp, err := s.GetContents(ctx, owner, repo, dir, opts)
|
2019-05-07 09:12:51 +08:00
|
|
|
if err != nil {
|
2021-08-10 17:49:43 +08:00
|
|
|
return nil, nil, resp, err
|
2019-05-07 09:12:51 +08:00
|
|
|
}
|
|
|
|
for _, contents := range dirContents {
|
|
|
|
if *contents.Name == filename {
|
|
|
|
if contents.DownloadURL == nil || *contents.DownloadURL == "" {
|
2021-08-10 17:49:43 +08:00
|
|
|
return nil, contents, resp, fmt.Errorf("No download link found for %s", filepath)
|
2019-05-07 09:12:51 +08:00
|
|
|
}
|
2021-08-10 17:49:43 +08:00
|
|
|
dlResp, err := s.client.client.Get(*contents.DownloadURL)
|
2019-05-07 09:12:51 +08:00
|
|
|
if err != nil {
|
2021-08-10 17:49:43 +08:00
|
|
|
return nil, contents, &Response{Response: dlResp}, err
|
2019-05-07 09:12:51 +08:00
|
|
|
}
|
2021-08-10 17:49:43 +08:00
|
|
|
return dlResp.Body, contents, &Response{Response: dlResp}, nil
|
2019-05-07 09:12:51 +08:00
|
|
|
}
|
|
|
|
}
|
2021-08-10 17:49:43 +08:00
|
|
|
return nil, nil, resp, fmt.Errorf("No file named %s found in %s", filename, dir)
|
2019-05-07 09:12:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetContents can return either the metadata and content of a single file
|
|
|
|
// (when path references a file) or the metadata of all the files and/or
|
|
|
|
// subdirectories of a directory (when path references a directory). To make it
|
|
|
|
// easy to distinguish between both result types and to mimic the API as much
|
|
|
|
// as possible, both result types will be returned but only one will contain a
|
|
|
|
// value and the other will be nil.
|
|
|
|
//
|
2021-08-10 17:49:43 +08:00
|
|
|
// GitHub API docs: https://docs.github.com/en/free-pro-team@latest/rest/reference/repos/#get-repository-content
|
2020-07-31 22:22:34 +08:00
|
|
|
func (s *RepositoriesService) GetContents(ctx context.Context, owner, repo, path string, opts *RepositoryContentGetOptions) (fileContent *RepositoryContent, directoryContent []*RepositoryContent, resp *Response, err error) {
|
2021-08-10 17:49:43 +08:00
|
|
|
escapedPath := (&url.URL{Path: strings.TrimSuffix(path, "/")}).String()
|
2019-05-07 09:12:51 +08:00
|
|
|
u := fmt.Sprintf("repos/%s/%s/contents/%s", owner, repo, escapedPath)
|
2020-07-31 22:22:34 +08:00
|
|
|
u, err = addOptions(u, opts)
|
2019-05-07 09:12:51 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, nil, err
|
|
|
|
}
|
|
|
|
req, err := s.client.NewRequest("GET", u, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, nil, err
|
|
|
|
}
|
|
|
|
var rawJSON json.RawMessage
|
|
|
|
resp, err = s.client.Do(ctx, req, &rawJSON)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, resp, err
|
|
|
|
}
|
|
|
|
fileUnmarshalError := json.Unmarshal(rawJSON, &fileContent)
|
|
|
|
if fileUnmarshalError == nil {
|
|
|
|
return fileContent, nil, resp, nil
|
|
|
|
}
|
|
|
|
directoryUnmarshalError := json.Unmarshal(rawJSON, &directoryContent)
|
|
|
|
if directoryUnmarshalError == nil {
|
|
|
|
return nil, directoryContent, resp, nil
|
|
|
|
}
|
|
|
|
return nil, nil, resp, fmt.Errorf("unmarshalling failed for both file and directory content: %s and %s", fileUnmarshalError, directoryUnmarshalError)
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateFile creates a new file in a repository at the given path and returns
|
|
|
|
// the commit and file metadata.
|
|
|
|
//
|
2021-08-10 17:49:43 +08:00
|
|
|
// GitHub API docs: https://docs.github.com/en/free-pro-team@latest/rest/reference/repos/#create-or-update-file-contents
|
2020-07-31 22:22:34 +08:00
|
|
|
func (s *RepositoriesService) CreateFile(ctx context.Context, owner, repo, path string, opts *RepositoryContentFileOptions) (*RepositoryContentResponse, *Response, error) {
|
2019-05-07 09:12:51 +08:00
|
|
|
u := fmt.Sprintf("repos/%s/%s/contents/%s", owner, repo, path)
|
2020-07-31 22:22:34 +08:00
|
|
|
req, err := s.client.NewRequest("PUT", u, opts)
|
2019-05-07 09:12:51 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
createResponse := new(RepositoryContentResponse)
|
|
|
|
resp, err := s.client.Do(ctx, req, createResponse)
|
|
|
|
if err != nil {
|
|
|
|
return nil, resp, err
|
|
|
|
}
|
|
|
|
return createResponse, resp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateFile updates a file in a repository at the given path and returns the
|
|
|
|
// commit and file metadata. Requires the blob SHA of the file being updated.
|
|
|
|
//
|
2021-08-10 17:49:43 +08:00
|
|
|
// GitHub API docs: https://docs.github.com/en/free-pro-team@latest/rest/reference/repos/#create-or-update-file-contents
|
2020-07-31 22:22:34 +08:00
|
|
|
func (s *RepositoriesService) UpdateFile(ctx context.Context, owner, repo, path string, opts *RepositoryContentFileOptions) (*RepositoryContentResponse, *Response, error) {
|
2019-05-07 09:12:51 +08:00
|
|
|
u := fmt.Sprintf("repos/%s/%s/contents/%s", owner, repo, path)
|
2020-07-31 22:22:34 +08:00
|
|
|
req, err := s.client.NewRequest("PUT", u, opts)
|
2019-05-07 09:12:51 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
updateResponse := new(RepositoryContentResponse)
|
|
|
|
resp, err := s.client.Do(ctx, req, updateResponse)
|
|
|
|
if err != nil {
|
|
|
|
return nil, resp, err
|
|
|
|
}
|
|
|
|
return updateResponse, resp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteFile deletes a file from a repository and returns the commit.
|
|
|
|
// Requires the blob SHA of the file to be deleted.
|
|
|
|
//
|
2021-08-10 17:49:43 +08:00
|
|
|
// GitHub API docs: https://docs.github.com/en/free-pro-team@latest/rest/reference/repos/#delete-a-file
|
2020-07-31 22:22:34 +08:00
|
|
|
func (s *RepositoriesService) DeleteFile(ctx context.Context, owner, repo, path string, opts *RepositoryContentFileOptions) (*RepositoryContentResponse, *Response, error) {
|
2019-05-07 09:12:51 +08:00
|
|
|
u := fmt.Sprintf("repos/%s/%s/contents/%s", owner, repo, path)
|
2020-07-31 22:22:34 +08:00
|
|
|
req, err := s.client.NewRequest("DELETE", u, opts)
|
2019-05-07 09:12:51 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
deleteResponse := new(RepositoryContentResponse)
|
|
|
|
resp, err := s.client.Do(ctx, req, deleteResponse)
|
|
|
|
if err != nil {
|
|
|
|
return nil, resp, err
|
|
|
|
}
|
|
|
|
return deleteResponse, resp, nil
|
|
|
|
}
|
|
|
|
|
2020-07-31 22:22:34 +08:00
|
|
|
// ArchiveFormat is used to define the archive type when calling GetArchiveLink.
|
|
|
|
type ArchiveFormat string
|
2019-05-07 09:12:51 +08:00
|
|
|
|
|
|
|
const (
|
|
|
|
// Tarball specifies an archive in gzipped tar format.
|
2020-07-31 22:22:34 +08:00
|
|
|
Tarball ArchiveFormat = "tarball"
|
2019-05-07 09:12:51 +08:00
|
|
|
|
|
|
|
// Zipball specifies an archive in zip format.
|
2020-07-31 22:22:34 +08:00
|
|
|
Zipball ArchiveFormat = "zipball"
|
2019-05-07 09:12:51 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
// GetArchiveLink returns an URL to download a tarball or zipball archive for a
|
|
|
|
// repository. The archiveFormat can be specified by either the github.Tarball
|
|
|
|
// or github.Zipball constant.
|
|
|
|
//
|
2021-08-10 17:49:43 +08:00
|
|
|
// GitHub API docs: https://docs.github.com/en/free-pro-team@latest/rest/reference/repos/contents/#get-archive-link
|
2020-07-31 22:22:34 +08:00
|
|
|
func (s *RepositoriesService) GetArchiveLink(ctx context.Context, owner, repo string, archiveformat ArchiveFormat, opts *RepositoryContentGetOptions, followRedirects bool) (*url.URL, *Response, error) {
|
2019-05-07 09:12:51 +08:00
|
|
|
u := fmt.Sprintf("repos/%s/%s/%s", owner, repo, archiveformat)
|
2020-07-31 22:22:34 +08:00
|
|
|
if opts != nil && opts.Ref != "" {
|
|
|
|
u += fmt.Sprintf("/%s", opts.Ref)
|
2019-05-07 09:12:51 +08:00
|
|
|
}
|
2020-07-31 22:22:34 +08:00
|
|
|
resp, err := s.getArchiveLinkFromURL(ctx, u, followRedirects)
|
2019-05-07 09:12:51 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
2020-07-31 22:22:34 +08:00
|
|
|
if resp.StatusCode != http.StatusFound {
|
|
|
|
return nil, newResponse(resp), fmt.Errorf("unexpected status code: %s", resp.Status)
|
|
|
|
}
|
|
|
|
parsedURL, err := url.Parse(resp.Header.Get("Location"))
|
|
|
|
return parsedURL, newResponse(resp), err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *RepositoriesService) getArchiveLinkFromURL(ctx context.Context, u string, followRedirects bool) (*http.Response, error) {
|
|
|
|
req, err := s.client.NewRequest("GET", u, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-05-07 09:12:51 +08:00
|
|
|
var resp *http.Response
|
|
|
|
// Use http.DefaultTransport if no custom Transport is configured
|
|
|
|
req = withContext(ctx, req)
|
|
|
|
if s.client.client.Transport == nil {
|
|
|
|
resp, err = http.DefaultTransport.RoundTrip(req)
|
|
|
|
} else {
|
|
|
|
resp, err = s.client.client.Transport.RoundTrip(req)
|
|
|
|
}
|
|
|
|
if err != nil {
|
2020-07-31 22:22:34 +08:00
|
|
|
return nil, err
|
2019-05-07 09:12:51 +08:00
|
|
|
}
|
|
|
|
resp.Body.Close()
|
2020-07-31 22:22:34 +08:00
|
|
|
|
|
|
|
// If redirect response is returned, follow it
|
|
|
|
if followRedirects && resp.StatusCode == http.StatusMovedPermanently {
|
|
|
|
u = resp.Header.Get("Location")
|
|
|
|
resp, err = s.getArchiveLinkFromURL(ctx, u, false)
|
2019-05-07 09:12:51 +08:00
|
|
|
}
|
2020-07-31 22:22:34 +08:00
|
|
|
return resp, err
|
2019-05-07 09:12:51 +08:00
|
|
|
}
|