commit
252e29e577
27 changed files with 630 additions and 2094 deletions
5
Makefile
5
Makefile
|
|
@ -5,3 +5,8 @@ BUILD_FILES = $(shell go list -f '{{range .GoFiles}}{{$$.Dir}}/{{.}}\
|
|||
|
||||
bin/gh: $(BUILD_FILES)
|
||||
go build -o "$@"
|
||||
|
||||
test:
|
||||
go test ./...
|
||||
|
||||
.PHONY: test
|
||||
|
|
@ -7,9 +7,8 @@ import (
|
|||
"io/ioutil"
|
||||
"net/http"
|
||||
"os"
|
||||
"os/user"
|
||||
"regexp"
|
||||
|
||||
"github.com/github/gh-cli/context"
|
||||
"github.com/github/gh-cli/version"
|
||||
)
|
||||
|
||||
|
|
@ -57,7 +56,7 @@ var GraphQL = func(query string, variables map[string]string, data interface{})
|
|||
return err
|
||||
}
|
||||
|
||||
token, err := getToken()
|
||||
token, err := context.Current().AuthToken()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
|
@ -138,20 +137,3 @@ func debugResponse(resp *http.Response, body string) {
|
|||
|
||||
fmt.Printf("DEBUG: GraphQL response:\n%+v\n\n%s\n\n", resp, body)
|
||||
}
|
||||
|
||||
// TODO: Everything below this line will be removed when Nate's context work is complete
|
||||
func getToken() (string, error) {
|
||||
usr, err := user.Current()
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
content, err := ioutil.ReadFile(usr.HomeDir + "/.config/hub")
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
r := regexp.MustCompile(`oauth_token: (\w+)`)
|
||||
token := r.FindStringSubmatch(string(content))
|
||||
return token[1], nil
|
||||
}
|
||||
|
|
|
|||
|
|
@ -2,10 +2,8 @@ package api
|
|||
|
||||
import (
|
||||
"fmt"
|
||||
"strings"
|
||||
|
||||
"github.com/github/gh-cli/git"
|
||||
"github.com/github/gh-cli/github"
|
||||
"github.com/github/gh-cli/context"
|
||||
)
|
||||
|
||||
type PullRequestsPayload struct {
|
||||
|
|
@ -21,7 +19,7 @@ type PullRequest struct {
|
|||
HeadRefName string
|
||||
}
|
||||
|
||||
func PullRequests() (PullRequestsPayload, error) {
|
||||
func PullRequests() (*PullRequestsPayload, error) {
|
||||
type edges struct {
|
||||
Edges []struct {
|
||||
Node PullRequest
|
||||
|
|
@ -81,13 +79,24 @@ func PullRequests() (PullRequestsPayload, error) {
|
|||
}
|
||||
`
|
||||
|
||||
project := project()
|
||||
owner := project.Owner
|
||||
repo := project.Name
|
||||
currentBranch := currentBranch()
|
||||
ghRepo, err := context.Current().BaseRepo()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
currentBranch, err := context.Current().Branch()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
currentUsername, err := context.Current().AuthLogin()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
viewerQuery := fmt.Sprintf("repo:%s/%s state:open is:pr author:%s", owner, repo, currentUsername())
|
||||
reviewerQuery := fmt.Sprintf("repo:%s/%s state:open review-requested:%s", owner, repo, currentUsername())
|
||||
owner := ghRepo.Owner
|
||||
repo := ghRepo.Name
|
||||
|
||||
viewerQuery := fmt.Sprintf("repo:%s/%s state:open is:pr author:%s", owner, repo, currentUsername)
|
||||
reviewerQuery := fmt.Sprintf("repo:%s/%s state:open review-requested:%s", owner, repo, currentUsername)
|
||||
|
||||
variables := map[string]string{
|
||||
"viewerQuery": viewerQuery,
|
||||
|
|
@ -98,9 +107,9 @@ func PullRequests() (PullRequestsPayload, error) {
|
|||
}
|
||||
|
||||
var resp response
|
||||
err := GraphQL(query, variables, &resp)
|
||||
err = GraphQL(query, variables, &resp)
|
||||
if err != nil {
|
||||
return PullRequestsPayload{}, err
|
||||
return nil, err
|
||||
}
|
||||
|
||||
var viewerCreated []PullRequest
|
||||
|
|
@ -124,37 +133,5 @@ func PullRequests() (PullRequestsPayload, error) {
|
|||
currentPR,
|
||||
}
|
||||
|
||||
return payload, nil
|
||||
}
|
||||
|
||||
// TODO: Everything below this line will be removed when Nate's context work is complete
|
||||
func project() github.Project {
|
||||
remotes, error := github.Remotes()
|
||||
if error != nil {
|
||||
panic(error)
|
||||
}
|
||||
for _, remote := range remotes {
|
||||
if project, error := remote.Project(); error == nil {
|
||||
return *project
|
||||
}
|
||||
}
|
||||
|
||||
panic("Could not get the project. What is a project? I don't know, it's kind of like a git repository I think?")
|
||||
}
|
||||
|
||||
func currentBranch() string {
|
||||
currentBranch, err := git.Head()
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
return strings.Replace(currentBranch, "refs/heads/", "", 1)
|
||||
}
|
||||
|
||||
func currentUsername() string {
|
||||
host, err := github.CurrentConfig().DefaultHost()
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
return host.User
|
||||
return &payload, nil
|
||||
}
|
||||
|
|
|
|||
|
|
@ -2,15 +2,11 @@ package command
|
|||
|
||||
import (
|
||||
"fmt"
|
||||
"net/url"
|
||||
"os"
|
||||
"os/exec"
|
||||
"strconv"
|
||||
"strings"
|
||||
|
||||
"github.com/github/gh-cli/api"
|
||||
"github.com/github/gh-cli/git"
|
||||
"github.com/github/gh-cli/github"
|
||||
"github.com/github/gh-cli/context"
|
||||
"github.com/github/gh-cli/utils"
|
||||
"github.com/spf13/cobra"
|
||||
)
|
||||
|
|
@ -52,7 +48,11 @@ func prList(cmd *cobra.Command, args []string) error {
|
|||
if prPayload.CurrentPR != nil {
|
||||
printPrs(*prPayload.CurrentPR)
|
||||
} else {
|
||||
message := fmt.Sprintf(" There is no pull request associated with %s", utils.Cyan("["+currentBranch()+"]"))
|
||||
currentBranch, err := context.Current().Branch()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
message := fmt.Sprintf(" There is no pull request associated with %s", utils.Cyan("["+currentBranch+"]"))
|
||||
printMessage(message)
|
||||
}
|
||||
fmt.Println()
|
||||
|
|
@ -77,19 +77,26 @@ func prList(cmd *cobra.Command, args []string) error {
|
|||
}
|
||||
|
||||
func prView(cmd *cobra.Command, args []string) error {
|
||||
project := project()
|
||||
baseRepo, err := context.Current().BaseRepo()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
var openURL string
|
||||
if len(args) > 0 {
|
||||
if prNumber, err := strconv.Atoi(args[0]); err == nil {
|
||||
openURL = project.WebURL("", "", fmt.Sprintf("pull/%d", prNumber))
|
||||
// TODO: move URL generation into GitHubRepository
|
||||
openURL = fmt.Sprintf("https://github.com/%s/%s/pull/%d", baseRepo.Owner, baseRepo.Name, prNumber)
|
||||
} else {
|
||||
return fmt.Errorf("invalid pull request number: '%s'", args[0])
|
||||
}
|
||||
} else {
|
||||
prPayload, err := api.PullRequests()
|
||||
if err != nil || prPayload.CurrentPR == nil {
|
||||
branch := currentBranch()
|
||||
branch, err := context.Current().Branch()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return fmt.Errorf("The [%s] branch has no open PRs", branch)
|
||||
}
|
||||
openURL = prPayload.CurrentPR.URL
|
||||
|
|
@ -130,41 +137,3 @@ func openInBrowser(url string) error {
|
|||
endingArgs := append(launcher[1:], url)
|
||||
return exec.Command(launcher[0], endingArgs...).Run()
|
||||
}
|
||||
|
||||
// The functions below should be replaced at some point by the context package
|
||||
// 🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨🧨
|
||||
func currentBranch() string {
|
||||
currentBranch, err := git.Head()
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
return strings.Replace(currentBranch, "refs/heads/", "", 1)
|
||||
}
|
||||
|
||||
func project() github.Project {
|
||||
if repoFromEnv := os.Getenv("GH_REPO"); repoFromEnv != "" {
|
||||
repoURL, err := url.Parse(fmt.Sprintf("https://github.com/%s.git", repoFromEnv))
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
project, err := github.NewProjectFromURL(repoURL)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
return *project
|
||||
}
|
||||
|
||||
remotes, err := github.Remotes()
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
for _, remote := range remotes {
|
||||
if project, err := remote.Project(); err == nil {
|
||||
return *project
|
||||
}
|
||||
}
|
||||
|
||||
panic("Could not get the project. What is a project? I don't know, it's kind of like a git repository I think?")
|
||||
}
|
||||
|
|
|
|||
|
|
@ -4,16 +4,18 @@ import (
|
|||
"regexp"
|
||||
"testing"
|
||||
|
||||
"github.com/github/gh-cli/context"
|
||||
"github.com/github/gh-cli/test"
|
||||
)
|
||||
|
||||
func TestPRList(t *testing.T) {
|
||||
ctx := context.InitBlankContext()
|
||||
ctx.SetBaseRepo("github/FAKE-GITHUB-REPO-NAME")
|
||||
ctx.SetBranch("master")
|
||||
|
||||
teardown := test.MockGraphQLResponse("test/fixtures/pr.json")
|
||||
defer teardown()
|
||||
|
||||
gitRepo := test.UseTempGitRepo()
|
||||
defer gitRepo.TearDown()
|
||||
|
||||
output, err := test.RunCommand(RootCmd, "pr list")
|
||||
if err != nil {
|
||||
t.Errorf("error running command `pr list`: %v", err)
|
||||
|
|
|
|||
|
|
@ -2,10 +2,34 @@ package command
|
|||
|
||||
import (
|
||||
"fmt"
|
||||
"os"
|
||||
|
||||
"github.com/github/gh-cli/context"
|
||||
"github.com/github/gh-cli/git"
|
||||
"github.com/spf13/cobra"
|
||||
)
|
||||
|
||||
var (
|
||||
currentRepo string
|
||||
currentBranch string
|
||||
)
|
||||
|
||||
func init() {
|
||||
RootCmd.PersistentFlags().StringVarP(¤tRepo, "repo", "R", "", "current GitHub repository")
|
||||
RootCmd.PersistentFlags().StringVarP(¤tBranch, "current-branch", "B", "", "current git branch")
|
||||
|
||||
ctx := context.InitDefaultContext()
|
||||
ctx.SetBranch(currentBranch)
|
||||
repo := currentRepo
|
||||
if repo == "" {
|
||||
repo = os.Getenv("GH_REPO")
|
||||
}
|
||||
ctx.SetBaseRepo(repo)
|
||||
|
||||
git.InitSSHAliasMap(nil)
|
||||
}
|
||||
|
||||
// RootCmd is the entry point of command-line execution
|
||||
var RootCmd = &cobra.Command{
|
||||
Use: "gh",
|
||||
Short: "GitHub CLI",
|
||||
|
|
|
|||
63
context/blank_context.go
Normal file
63
context/blank_context.go
Normal file
|
|
@ -0,0 +1,63 @@
|
|||
package context
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"strings"
|
||||
)
|
||||
|
||||
// InitBlankContext initializes a blank context for testing
|
||||
func InitBlankContext() Context {
|
||||
currentContext = &blankContext{
|
||||
authToken: "OTOKEN",
|
||||
authLogin: "monalisa",
|
||||
}
|
||||
return currentContext
|
||||
}
|
||||
|
||||
// A Context implementation that queries the filesystem
|
||||
type blankContext struct {
|
||||
authToken string
|
||||
authLogin string
|
||||
branch string
|
||||
baseRepo *GitHubRepository
|
||||
}
|
||||
|
||||
func (c *blankContext) AuthToken() (string, error) {
|
||||
return c.authToken, nil
|
||||
}
|
||||
|
||||
func (c *blankContext) AuthLogin() (string, error) {
|
||||
return c.authLogin, nil
|
||||
}
|
||||
|
||||
func (c *blankContext) Branch() (string, error) {
|
||||
if c.branch == "" {
|
||||
return "", fmt.Errorf("branch was not initialized")
|
||||
}
|
||||
return c.branch, nil
|
||||
}
|
||||
|
||||
func (c *blankContext) SetBranch(b string) {
|
||||
c.branch = b
|
||||
}
|
||||
|
||||
func (c *blankContext) Remotes() (Remotes, error) {
|
||||
return Remotes{}, nil
|
||||
}
|
||||
|
||||
func (c *blankContext) BaseRepo() (*GitHubRepository, error) {
|
||||
if c.baseRepo == nil {
|
||||
return nil, fmt.Errorf("base repo was not initialized")
|
||||
}
|
||||
return c.baseRepo, nil
|
||||
}
|
||||
|
||||
func (c *blankContext) SetBaseRepo(nwo string) {
|
||||
parts := strings.SplitN(nwo, "/", 2)
|
||||
if len(parts) == 2 {
|
||||
c.baseRepo = &GitHubRepository{
|
||||
Owner: parts[0],
|
||||
Name: parts[1],
|
||||
}
|
||||
}
|
||||
}
|
||||
50
context/config_file.go
Normal file
50
context/config_file.go
Normal file
|
|
@ -0,0 +1,50 @@
|
|||
package context
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"os"
|
||||
|
||||
"gopkg.in/yaml.v3"
|
||||
)
|
||||
|
||||
type configEntry struct {
|
||||
User string
|
||||
Token string `yaml:"oauth_token"`
|
||||
}
|
||||
|
||||
func parseConfigFile(fn string) (*configEntry, error) {
|
||||
f, err := os.Open(fn)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer f.Close()
|
||||
return parseConfig(f)
|
||||
}
|
||||
|
||||
func parseConfig(r io.Reader) (*configEntry, error) {
|
||||
data, err := ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
var config yaml.Node
|
||||
err = yaml.Unmarshal(data, &config)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if len(config.Content) < 1 {
|
||||
return nil, fmt.Errorf("malformed config")
|
||||
}
|
||||
for i := 0; i < len(config.Content[0].Content)-1; i = i + 2 {
|
||||
if config.Content[0].Content[i].Value == defaultHostname {
|
||||
var entries []configEntry
|
||||
err = config.Content[0].Content[i+1].Decode(&entries)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &entries[0], nil
|
||||
}
|
||||
}
|
||||
return nil, fmt.Errorf("could not find config entry for %q", defaultHostname)
|
||||
}
|
||||
54
context/config_file_test.go
Normal file
54
context/config_file_test.go
Normal file
|
|
@ -0,0 +1,54 @@
|
|||
package context
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"reflect"
|
||||
"strings"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func eq(t *testing.T, got interface{}, expected interface{}) {
|
||||
if !reflect.DeepEqual(got, expected) {
|
||||
t.Errorf("expected: %v, got: %v", expected, got)
|
||||
}
|
||||
}
|
||||
|
||||
func Test_parseConfig(t *testing.T) {
|
||||
c := strings.NewReader(`---
|
||||
github.com:
|
||||
- user: monalisa
|
||||
oauth_token: OTOKEN
|
||||
protocol: https
|
||||
- user: wronguser
|
||||
oauth_token: NOTTHIS
|
||||
`)
|
||||
entry, err := parseConfig(c)
|
||||
eq(t, err, nil)
|
||||
eq(t, entry.User, "monalisa")
|
||||
eq(t, entry.Token, "OTOKEN")
|
||||
}
|
||||
|
||||
func Test_parseConfig_multipleHosts(t *testing.T) {
|
||||
c := strings.NewReader(`---
|
||||
example.com:
|
||||
- user: wronguser
|
||||
oauth_token: NOTTHIS
|
||||
github.com:
|
||||
- user: monalisa
|
||||
oauth_token: OTOKEN
|
||||
`)
|
||||
entry, err := parseConfig(c)
|
||||
eq(t, err, nil)
|
||||
eq(t, entry.User, "monalisa")
|
||||
eq(t, entry.Token, "OTOKEN")
|
||||
}
|
||||
|
||||
func Test_parseConfig_notFound(t *testing.T) {
|
||||
c := strings.NewReader(`---
|
||||
example.com:
|
||||
- user: wronguser
|
||||
oauth_token: NOTTHIS
|
||||
`)
|
||||
_, err := parseConfig(c)
|
||||
eq(t, err, errors.New(`could not find config entry for "github.com"`))
|
||||
}
|
||||
135
context/context.go
Normal file
135
context/context.go
Normal file
|
|
@ -0,0 +1,135 @@
|
|||
package context
|
||||
|
||||
import (
|
||||
"strings"
|
||||
|
||||
"github.com/github/gh-cli/git"
|
||||
"github.com/mitchellh/go-homedir"
|
||||
)
|
||||
|
||||
// Context represents the interface for querying information about the current environment
|
||||
type Context interface {
|
||||
AuthToken() (string, error)
|
||||
AuthLogin() (string, error)
|
||||
Branch() (string, error)
|
||||
SetBranch(string)
|
||||
Remotes() (Remotes, error)
|
||||
BaseRepo() (*GitHubRepository, error)
|
||||
SetBaseRepo(string)
|
||||
}
|
||||
|
||||
var currentContext Context
|
||||
|
||||
// Current returns the currently initialized Context instance
|
||||
func Current() Context {
|
||||
return currentContext
|
||||
}
|
||||
|
||||
// InitDefaultContext initializes the default filesystem context
|
||||
func InitDefaultContext() Context {
|
||||
ctx := &fsContext{}
|
||||
if currentContext == nil {
|
||||
currentContext = ctx
|
||||
}
|
||||
return ctx
|
||||
}
|
||||
|
||||
// A Context implementation that queries the filesystem
|
||||
type fsContext struct {
|
||||
config *configEntry
|
||||
remotes Remotes
|
||||
branch string
|
||||
baseRepo *GitHubRepository
|
||||
}
|
||||
|
||||
func (c *fsContext) configFile() string {
|
||||
dir, _ := homedir.Expand("~/.config/hub")
|
||||
return dir
|
||||
}
|
||||
|
||||
func (c *fsContext) getConfig() (*configEntry, error) {
|
||||
if c.config == nil {
|
||||
entry, err := parseConfigFile(c.configFile())
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
c.config = entry
|
||||
}
|
||||
return c.config, nil
|
||||
}
|
||||
|
||||
func (c *fsContext) AuthToken() (string, error) {
|
||||
config, err := c.getConfig()
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
return config.Token, nil
|
||||
}
|
||||
|
||||
func (c *fsContext) AuthLogin() (string, error) {
|
||||
config, err := c.getConfig()
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
return config.User, nil
|
||||
}
|
||||
|
||||
func (c *fsContext) Branch() (string, error) {
|
||||
if c.branch != "" {
|
||||
return c.branch, nil
|
||||
}
|
||||
|
||||
currentBranch, err := git.Head()
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
c.branch = strings.Replace(currentBranch, "refs/heads/", "", 1)
|
||||
return c.branch, nil
|
||||
}
|
||||
|
||||
func (c *fsContext) SetBranch(b string) {
|
||||
c.branch = b
|
||||
}
|
||||
|
||||
func (c *fsContext) Remotes() (Remotes, error) {
|
||||
if c.remotes == nil {
|
||||
gitRemotes, err := git.Remotes()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
c.remotes = parseRemotes(gitRemotes)
|
||||
}
|
||||
return c.remotes, nil
|
||||
}
|
||||
|
||||
func (c *fsContext) BaseRepo() (*GitHubRepository, error) {
|
||||
if c.baseRepo != nil {
|
||||
return c.baseRepo, nil
|
||||
}
|
||||
|
||||
remotes, err := c.Remotes()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
rem, err := remotes.FindByName("upstream", "github", "origin", "*")
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
c.baseRepo = &GitHubRepository{
|
||||
Owner: rem.Owner,
|
||||
Name: rem.Repo,
|
||||
}
|
||||
return c.baseRepo, nil
|
||||
}
|
||||
|
||||
func (c *fsContext) SetBaseRepo(nwo string) {
|
||||
parts := strings.SplitN(nwo, "/", 2)
|
||||
if len(parts) == 2 {
|
||||
c.baseRepo = &GitHubRepository{
|
||||
Owner: parts[0],
|
||||
Name: parts[1],
|
||||
}
|
||||
}
|
||||
}
|
||||
100
context/remote.go
Normal file
100
context/remote.go
Normal file
|
|
@ -0,0 +1,100 @@
|
|||
package context
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"regexp"
|
||||
"strings"
|
||||
|
||||
"github.com/github/gh-cli/git"
|
||||
)
|
||||
|
||||
const defaultHostname = "github.com"
|
||||
|
||||
// Remotes represents a set of git remotes
|
||||
type Remotes []*Remote
|
||||
|
||||
// FindByName returns the first Remote whose name matches the list
|
||||
func (r Remotes) FindByName(names ...string) (*Remote, error) {
|
||||
for _, name := range names {
|
||||
for _, rem := range r {
|
||||
if rem.Name == name || name == "*" {
|
||||
return rem, nil
|
||||
}
|
||||
}
|
||||
}
|
||||
return nil, fmt.Errorf("no GitHub remotes found")
|
||||
}
|
||||
|
||||
// Remote represents a git remote mapped to a GitHub repository
|
||||
type Remote struct {
|
||||
Name string
|
||||
Owner string
|
||||
Repo string
|
||||
}
|
||||
|
||||
func (r *Remote) String() string {
|
||||
return r.Name
|
||||
}
|
||||
|
||||
// GitHubRepository represents a GitHub respository
|
||||
type GitHubRepository struct {
|
||||
Name string
|
||||
Owner string
|
||||
}
|
||||
|
||||
func parseRemotes(gitRemotes []string) (remotes Remotes) {
|
||||
re := regexp.MustCompile(`(.+)\s+(.+)\s+\((push|fetch)\)`)
|
||||
|
||||
names := []string{}
|
||||
remotesMap := make(map[string]map[string]string)
|
||||
for _, r := range gitRemotes {
|
||||
if re.MatchString(r) {
|
||||
match := re.FindStringSubmatch(r)
|
||||
name := strings.TrimSpace(match[1])
|
||||
url := strings.TrimSpace(match[2])
|
||||
urlType := strings.TrimSpace(match[3])
|
||||
utm, ok := remotesMap[name]
|
||||
if !ok {
|
||||
utm = make(map[string]string)
|
||||
remotesMap[name] = utm
|
||||
names = append(names, name)
|
||||
}
|
||||
utm[urlType] = url
|
||||
}
|
||||
}
|
||||
|
||||
for _, name := range names {
|
||||
urlMap := remotesMap[name]
|
||||
repo, err := repoFromURL(urlMap["fetch"])
|
||||
if err != nil {
|
||||
repo, err = repoFromURL(urlMap["push"])
|
||||
}
|
||||
if err == nil {
|
||||
remotes = append(remotes, &Remote{
|
||||
Name: name,
|
||||
Owner: repo.Owner,
|
||||
Repo: repo.Name,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func repoFromURL(u string) (*GitHubRepository, error) {
|
||||
url, err := git.ParseURL(u)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if url.Hostname() != defaultHostname {
|
||||
return nil, fmt.Errorf("invalid hostname: %s", url.Hostname())
|
||||
}
|
||||
parts := strings.SplitN(strings.TrimPrefix(url.Path, "/"), "/", 3)
|
||||
if len(parts) < 2 {
|
||||
return nil, fmt.Errorf("invalid path: %s", url.Path)
|
||||
}
|
||||
return &GitHubRepository{
|
||||
Owner: parts[0],
|
||||
Name: strings.TrimSuffix(parts[1], ".git"),
|
||||
}, nil
|
||||
}
|
||||
78
context/remote_test.go
Normal file
78
context/remote_test.go
Normal file
|
|
@ -0,0 +1,78 @@
|
|||
package context
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"testing"
|
||||
|
||||
"github.com/github/gh-cli/git"
|
||||
)
|
||||
|
||||
func Test_repoFromURL(t *testing.T) {
|
||||
git.InitSSHAliasMap(nil)
|
||||
|
||||
r, err := repoFromURL("http://github.com/monalisa/octo-cat.git")
|
||||
eq(t, err, nil)
|
||||
eq(t, r, &GitHubRepository{Owner: "monalisa", Name: "octo-cat"})
|
||||
}
|
||||
|
||||
func Test_repoFromURL_invalid(t *testing.T) {
|
||||
git.InitSSHAliasMap(nil)
|
||||
|
||||
_, err := repoFromURL("https://example.com/one/two")
|
||||
eq(t, err, errors.New(`invalid hostname: example.com`))
|
||||
|
||||
_, err = repoFromURL("/path/to/disk")
|
||||
eq(t, err, errors.New(`invalid hostname: `))
|
||||
}
|
||||
|
||||
func Test_repoFromURL_SSH(t *testing.T) {
|
||||
git.InitSSHAliasMap(map[string]string{
|
||||
"gh": "github.com",
|
||||
"github.com": "ssh.github.com",
|
||||
})
|
||||
|
||||
r, err := repoFromURL("git@gh:monalisa/octo-cat")
|
||||
eq(t, err, nil)
|
||||
eq(t, r, &GitHubRepository{Owner: "monalisa", Name: "octo-cat"})
|
||||
|
||||
r, err = repoFromURL("git@github.com:monalisa/octo-cat")
|
||||
eq(t, err, nil)
|
||||
eq(t, r, &GitHubRepository{Owner: "monalisa", Name: "octo-cat"})
|
||||
}
|
||||
|
||||
func Test_parseRemotes(t *testing.T) {
|
||||
git.InitSSHAliasMap(nil)
|
||||
|
||||
remoteList := []string{
|
||||
"mona\tgit@github.com:monalisa/myfork.git (fetch)",
|
||||
"origin\thttps://github.com/monalisa/octo-cat.git (fetch)",
|
||||
"origin\thttps://github.com/monalisa/octo-cat-push.git (push)",
|
||||
"upstream\thttps://example.com/nowhere.git (fetch)",
|
||||
"upstream\thttps://github.com/hubot/tools (push)",
|
||||
}
|
||||
r := parseRemotes(remoteList)
|
||||
eq(t, len(r), 3)
|
||||
|
||||
eq(t, r[0], &Remote{Name: "mona", Owner: "monalisa", Repo: "myfork"})
|
||||
eq(t, r[1], &Remote{Name: "origin", Owner: "monalisa", Repo: "octo-cat"})
|
||||
eq(t, r[2], &Remote{Name: "upstream", Owner: "hubot", Repo: "tools"})
|
||||
}
|
||||
|
||||
func Test_Remotes_FindByName(t *testing.T) {
|
||||
list := Remotes{
|
||||
&Remote{Name: "mona", Owner: "monalisa", Repo: "myfork"},
|
||||
&Remote{Name: "origin", Owner: "monalisa", Repo: "octo-cat"},
|
||||
&Remote{Name: "upstream", Owner: "hubot", Repo: "tools"},
|
||||
}
|
||||
|
||||
r, err := list.FindByName("upstream", "origin")
|
||||
eq(t, err, nil)
|
||||
eq(t, r.Name, "upstream")
|
||||
|
||||
r, err = list.FindByName("nonexist", "*")
|
||||
eq(t, err, nil)
|
||||
eq(t, r.Name, "mona")
|
||||
|
||||
_, err = list.FindByName("nonexist")
|
||||
eq(t, err, errors.New(`no GitHub remotes found`))
|
||||
}
|
||||
|
|
@ -2,6 +2,7 @@ package git
|
|||
|
||||
import (
|
||||
"bufio"
|
||||
"io"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"regexp"
|
||||
|
|
@ -10,13 +11,19 @@ import (
|
|||
"github.com/mitchellh/go-homedir"
|
||||
)
|
||||
|
||||
const (
|
||||
hostReStr = "(?i)^[ \t]*(host|hostname)[ \t]+(.+)$"
|
||||
var (
|
||||
sshHostRE,
|
||||
sshTokenRE *regexp.Regexp
|
||||
)
|
||||
|
||||
type SSHConfig map[string]string
|
||||
func init() {
|
||||
sshHostRE = regexp.MustCompile("(?i)^[ \t]*(host|hostname)[ \t]+(.+)$")
|
||||
sshTokenRE = regexp.MustCompile(`%[%h]`)
|
||||
}
|
||||
|
||||
func newSSHConfigReader() *SSHConfigReader {
|
||||
type sshAliasMap map[string]string
|
||||
|
||||
func sshParseFiles() sshAliasMap {
|
||||
configFiles := []string{
|
||||
"/etc/ssh_config",
|
||||
"/etc/ssh/ssh_config",
|
||||
|
|
@ -25,38 +32,33 @@ func newSSHConfigReader() *SSHConfigReader {
|
|||
userConfig := filepath.Join(homedir, ".ssh", "config")
|
||||
configFiles = append([]string{userConfig}, configFiles...)
|
||||
}
|
||||
return &SSHConfigReader{
|
||||
Files: configFiles,
|
||||
|
||||
openFiles := []io.Reader{}
|
||||
for _, file := range configFiles {
|
||||
f, err := os.Open(file)
|
||||
if err != nil {
|
||||
continue
|
||||
}
|
||||
defer f.Close()
|
||||
openFiles = append(openFiles, f)
|
||||
}
|
||||
return sshParse(openFiles...)
|
||||
}
|
||||
|
||||
type SSHConfigReader struct {
|
||||
Files []string
|
||||
}
|
||||
|
||||
func (r *SSHConfigReader) Read() SSHConfig {
|
||||
config := make(SSHConfig)
|
||||
hostRe := regexp.MustCompile(hostReStr)
|
||||
|
||||
for _, filename := range r.Files {
|
||||
r.readFile(config, hostRe, filename)
|
||||
func sshParse(r ...io.Reader) sshAliasMap {
|
||||
config := sshAliasMap{}
|
||||
for _, file := range r {
|
||||
sshParseConfig(config, file)
|
||||
}
|
||||
|
||||
return config
|
||||
}
|
||||
|
||||
func (r *SSHConfigReader) readFile(c SSHConfig, re *regexp.Regexp, f string) error {
|
||||
file, err := os.Open(f)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer file.Close()
|
||||
|
||||
func sshParseConfig(c sshAliasMap, file io.Reader) error {
|
||||
hosts := []string{"*"}
|
||||
scanner := bufio.NewScanner(file)
|
||||
for scanner.Scan() {
|
||||
line := scanner.Text()
|
||||
match := re.FindStringSubmatch(line)
|
||||
match := sshHostRE.FindStringSubmatch(line)
|
||||
if match == nil {
|
||||
continue
|
||||
}
|
||||
|
|
@ -67,7 +69,7 @@ func (r *SSHConfigReader) readFile(c SSHConfig, re *regexp.Regexp, f string) err
|
|||
} else {
|
||||
for _, host := range hosts {
|
||||
for _, name := range names {
|
||||
c[host] = expandTokens(name, host)
|
||||
c[host] = sshExpandTokens(name, host)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -76,9 +78,8 @@ func (r *SSHConfigReader) readFile(c SSHConfig, re *regexp.Regexp, f string) err
|
|||
return scanner.Err()
|
||||
}
|
||||
|
||||
func expandTokens(text, host string) string {
|
||||
re := regexp.MustCompile(`%[%h]`)
|
||||
return re.ReplaceAllStringFunc(text, func(match string) string {
|
||||
func sshExpandTokens(text, host string) string {
|
||||
return sshTokenRE.ReplaceAllStringFunc(text, func(match string) string {
|
||||
switch match {
|
||||
case "%h":
|
||||
return host
|
||||
|
|
|
|||
27
git/ssh_config_test.go
Normal file
27
git/ssh_config_test.go
Normal file
|
|
@ -0,0 +1,27 @@
|
|||
package git
|
||||
|
||||
import (
|
||||
"reflect"
|
||||
"strings"
|
||||
"testing"
|
||||
)
|
||||
|
||||
// TODO: extract assertion helpers into a shared package
|
||||
func eq(t *testing.T, got interface{}, expected interface{}) {
|
||||
if !reflect.DeepEqual(got, expected) {
|
||||
t.Errorf("expected: %v, got: %v", expected, got)
|
||||
}
|
||||
}
|
||||
|
||||
func Test_sshParse(t *testing.T) {
|
||||
m := sshParse(strings.NewReader(`
|
||||
Host foo bar
|
||||
HostName example.com
|
||||
`), strings.NewReader(`
|
||||
Host bar baz
|
||||
hostname %%%h.net%%
|
||||
`))
|
||||
eq(t, m["foo"], "example.com")
|
||||
eq(t, m["bar"], "%bar.net%")
|
||||
eq(t, m["nonexist"], "")
|
||||
}
|
||||
30
git/url.go
30
git/url.go
|
|
@ -7,15 +7,12 @@ import (
|
|||
)
|
||||
|
||||
var (
|
||||
cachedSSHConfig SSHConfig
|
||||
cachedSSHConfig sshAliasMap
|
||||
protocolRe = regexp.MustCompile("^[a-zA-Z_+-]+://")
|
||||
)
|
||||
|
||||
type URLParser struct {
|
||||
SSHConfig SSHConfig
|
||||
}
|
||||
|
||||
func (p *URLParser) Parse(rawURL string) (u *url.URL, err error) {
|
||||
// ParseURL normalizes git remote urls
|
||||
func ParseURL(rawURL string) (u *url.URL, err error) {
|
||||
if !protocolRe.MatchString(rawURL) &&
|
||||
strings.Contains(rawURL, ":") &&
|
||||
// not a Windows path
|
||||
|
|
@ -44,7 +41,10 @@ func (p *URLParser) Parse(rawURL string) (u *url.URL, err error) {
|
|||
u.Host = u.Host[0:idx]
|
||||
}
|
||||
|
||||
sshHost := p.SSHConfig[u.Host]
|
||||
if cachedSSHConfig == nil {
|
||||
return
|
||||
}
|
||||
sshHost := cachedSSHConfig[u.Host]
|
||||
// ignore replacing host that fixes for limited network
|
||||
// https://help.github.com/articles/using-ssh-over-the-https-port
|
||||
ignoredHost := u.Host == "github.com" && sshHost == "ssh.github.com"
|
||||
|
|
@ -55,12 +55,14 @@ func (p *URLParser) Parse(rawURL string) (u *url.URL, err error) {
|
|||
return
|
||||
}
|
||||
|
||||
func ParseURL(rawURL string) (u *url.URL, err error) {
|
||||
if cachedSSHConfig == nil {
|
||||
cachedSSHConfig = newSSHConfigReader().Read()
|
||||
// InitSSHAliasMap prepares globally cached SSH hostname alias mappings
|
||||
func InitSSHAliasMap(m map[string]string) {
|
||||
if m == nil {
|
||||
cachedSSHConfig = sshParseFiles()
|
||||
return
|
||||
}
|
||||
cachedSSHConfig = sshAliasMap{}
|
||||
for k, v := range m {
|
||||
cachedSSHConfig[k] = v
|
||||
}
|
||||
|
||||
p := &URLParser{cachedSSHConfig}
|
||||
|
||||
return p.Parse(rawURL)
|
||||
}
|
||||
|
|
|
|||
585
github/client.go
585
github/client.go
|
|
@ -1,585 +0,0 @@
|
|||
package github
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"io"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"os"
|
||||
"os/exec"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/github/gh-cli/version"
|
||||
)
|
||||
|
||||
const (
|
||||
GitHubHost string = "github.com"
|
||||
OAuthAppURL string = "https://github.com/github/gh-cli"
|
||||
)
|
||||
|
||||
var userAgent = "GitHub CLI " + version.Version
|
||||
|
||||
func NewClient(h string) *Client {
|
||||
return NewClientWithHost(&Host{Host: h})
|
||||
}
|
||||
|
||||
func NewClientWithHost(host *Host) *Client {
|
||||
return &Client{Host: host}
|
||||
}
|
||||
|
||||
type Client struct {
|
||||
Host *Host
|
||||
cachedClient *simpleClient
|
||||
}
|
||||
|
||||
func (client *Client) FetchPullRequests(project *Project, filterParams map[string]interface{}, limit int, filter func(*PullRequest) bool) (pulls []PullRequest, err error) {
|
||||
api, err := client.simpleApi()
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
path := fmt.Sprintf("repos/%s/%s/pulls?per_page=%d", project.Owner, project.Name, perPage(limit, 100))
|
||||
if filterParams != nil {
|
||||
path = addQuery(path, filterParams)
|
||||
}
|
||||
|
||||
pulls = []PullRequest{}
|
||||
var res *simpleResponse
|
||||
|
||||
for path != "" {
|
||||
res, err = api.GetFile(path, draftsType)
|
||||
if err = checkStatus(200, "fetching pull requests", res, err); err != nil {
|
||||
return
|
||||
}
|
||||
path = res.Link("next")
|
||||
|
||||
pullsPage := []PullRequest{}
|
||||
if err = res.Unmarshal(&pullsPage); err != nil {
|
||||
return
|
||||
}
|
||||
for _, pr := range pullsPage {
|
||||
if filter == nil || filter(&pr) {
|
||||
pulls = append(pulls, pr)
|
||||
if limit > 0 && len(pulls) == limit {
|
||||
path = ""
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func (client *Client) PullRequest(project *Project, id string) (pr *PullRequest, err error) {
|
||||
api, err := client.simpleApi()
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
res, err := api.Get(fmt.Sprintf("repos/%s/%s/pulls/%s", project.Owner, project.Name, id))
|
||||
if err = checkStatus(200, "getting pull request", res, err); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
pr = &PullRequest{}
|
||||
err = res.Unmarshal(pr)
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func (client *Client) CreatePullRequest(project *Project, params map[string]interface{}) (pr *PullRequest, err error) {
|
||||
api, err := client.simpleApi()
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
res, err := api.PostJSONPreview(fmt.Sprintf("repos/%s/%s/pulls", project.Owner, project.Name), params, draftsType)
|
||||
if err = checkStatus(201, "creating pull request", res, err); err != nil {
|
||||
if res != nil && res.StatusCode == 404 {
|
||||
projectUrl := strings.SplitN(project.WebURL("", "", ""), "://", 2)[1]
|
||||
err = fmt.Errorf("%s\nAre you sure that %s exists?", err, projectUrl)
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
pr = &PullRequest{}
|
||||
err = res.Unmarshal(pr)
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func (client *Client) UpdatePullRequest(pr *PullRequest, params map[string]interface{}) (updatedPullRequest *PullRequest, err error) {
|
||||
api, err := client.simpleApi()
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
res, err := api.PatchJSON(pr.ApiUrl, params)
|
||||
if err = checkStatus(200, "updating pull request", res, err); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
updatedPullRequest = &PullRequest{}
|
||||
err = res.Unmarshal(updatedPullRequest)
|
||||
return
|
||||
}
|
||||
|
||||
func (client *Client) RequestReview(project *Project, prNumber int, params map[string]interface{}) (err error) {
|
||||
api, err := client.simpleApi()
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
res, err := api.PostJSON(fmt.Sprintf("repos/%s/%s/pulls/%d/requested_reviewers", project.Owner, project.Name, prNumber), params)
|
||||
if err = checkStatus(201, "requesting reviewer", res, err); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
res.Body.Close()
|
||||
return
|
||||
}
|
||||
|
||||
func (client *Client) Repository(project *Project) (repo *Repository, err error) {
|
||||
api, err := client.simpleApi()
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
res, err := api.Get(fmt.Sprintf("repos/%s/%s", project.Owner, project.Name))
|
||||
if err = checkStatus(200, "getting repository info", res, err); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
repo = &Repository{}
|
||||
err = res.Unmarshal(&repo)
|
||||
return
|
||||
}
|
||||
|
||||
type Repository struct {
|
||||
Name string `json:"name"`
|
||||
FullName string `json:"full_name"`
|
||||
Parent *Repository `json:"parent"`
|
||||
Owner *User `json:"owner"`
|
||||
Private bool `json:"private"`
|
||||
HasWiki bool `json:"has_wiki"`
|
||||
Permissions *RepositoryPermissions `json:"permissions"`
|
||||
HtmlUrl string `json:"html_url"`
|
||||
DefaultBranch string `json:"default_branch"`
|
||||
}
|
||||
|
||||
type RepositoryPermissions struct {
|
||||
Admin bool `json:"admin"`
|
||||
Push bool `json:"push"`
|
||||
Pull bool `json:"pull"`
|
||||
}
|
||||
|
||||
type Issue struct {
|
||||
Number int `json:"number"`
|
||||
State string `json:"state"`
|
||||
Title string `json:"title"`
|
||||
Body string `json:"body"`
|
||||
User *User `json:"user"`
|
||||
|
||||
PullRequest *PullRequest `json:"pull_request"`
|
||||
Head *PullRequestSpec `json:"head"`
|
||||
Base *PullRequestSpec `json:"base"`
|
||||
|
||||
MergeCommitSha string `json:"merge_commit_sha"`
|
||||
MaintainerCanModify bool `json:"maintainer_can_modify"`
|
||||
Draft bool `json:"draft"`
|
||||
|
||||
Comments int `json:"comments"`
|
||||
Labels []IssueLabel `json:"labels"`
|
||||
Assignees []User `json:"assignees"`
|
||||
Milestone *Milestone `json:"milestone"`
|
||||
CreatedAt time.Time `json:"created_at"`
|
||||
UpdatedAt time.Time `json:"updated_at"`
|
||||
MergedAt time.Time `json:"merged_at"`
|
||||
|
||||
RequestedReviewers []User `json:"requested_reviewers"`
|
||||
RequestedTeams []Team `json:"requested_teams"`
|
||||
|
||||
ApiUrl string `json:"url"`
|
||||
HtmlUrl string `json:"html_url"`
|
||||
|
||||
ClosedBy *User `json:"closed_by"`
|
||||
}
|
||||
|
||||
type PullRequest Issue
|
||||
|
||||
type PullRequestSpec struct {
|
||||
Label string `json:"label"`
|
||||
Ref string `json:"ref"`
|
||||
Sha string `json:"sha"`
|
||||
Repo *Repository `json:"repo"`
|
||||
}
|
||||
|
||||
func (pr *PullRequest) IsSameRepo() bool {
|
||||
return pr.Head != nil && pr.Head.Repo != nil &&
|
||||
pr.Head.Repo.Name == pr.Base.Repo.Name &&
|
||||
pr.Head.Repo.Owner.Login == pr.Base.Repo.Owner.Login
|
||||
}
|
||||
|
||||
func (pr *PullRequest) HasRequestedReviewer(name string) bool {
|
||||
for _, user := range pr.RequestedReviewers {
|
||||
if strings.EqualFold(user.Login, name) {
|
||||
return true
|
||||
}
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
func (pr *PullRequest) HasRequestedTeam(name string) bool {
|
||||
for _, team := range pr.RequestedTeams {
|
||||
if strings.EqualFold(team.Slug, name) {
|
||||
return true
|
||||
}
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
type IssueLabel struct {
|
||||
Name string `json:"name"`
|
||||
Color string `json:"color"`
|
||||
}
|
||||
|
||||
type User struct {
|
||||
Login string `json:"login"`
|
||||
}
|
||||
|
||||
type Team struct {
|
||||
Name string `json:"name"`
|
||||
Slug string `json:"slug"`
|
||||
}
|
||||
|
||||
type Milestone struct {
|
||||
Number int `json:"number"`
|
||||
Title string `json:"title"`
|
||||
}
|
||||
|
||||
func (client *Client) GenericAPIRequest(method, path string, data interface{}, headers map[string]string, ttl int) (*simpleResponse, error) {
|
||||
api, err := client.simpleApi()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
api.CacheTTL = ttl
|
||||
|
||||
var body io.Reader
|
||||
switch d := data.(type) {
|
||||
case map[string]interface{}:
|
||||
if method == "GET" {
|
||||
path = addQuery(path, d)
|
||||
} else if len(d) > 0 {
|
||||
json, err := json.Marshal(d)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
body = bytes.NewBuffer(json)
|
||||
}
|
||||
case io.Reader:
|
||||
body = d
|
||||
}
|
||||
|
||||
return api.performRequest(method, path, body, func(req *http.Request) {
|
||||
if body != nil {
|
||||
req.Header.Set("Content-Type", "application/json; charset=utf-8")
|
||||
}
|
||||
for key, value := range headers {
|
||||
req.Header.Set(key, value)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func (client *Client) CurrentUser() (user *User, err error) {
|
||||
api, err := client.simpleApi()
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
res, err := api.Get("user")
|
||||
if err = checkStatus(200, "getting current user", res, err); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
user = &User{}
|
||||
err = res.Unmarshal(user)
|
||||
return
|
||||
}
|
||||
|
||||
type AuthorizationEntry struct {
|
||||
Token string `json:"token"`
|
||||
}
|
||||
|
||||
func isToken(api *simpleClient, password string) bool {
|
||||
api.PrepareRequest = func(req *http.Request) {
|
||||
req.Header.Set("Authorization", "token "+password)
|
||||
}
|
||||
|
||||
res, _ := api.Get("user")
|
||||
if res != nil && res.StatusCode == 200 {
|
||||
return true
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
func (client *Client) FindOrCreateToken(user, password, twoFactorCode string) (token string, err error) {
|
||||
api := client.apiClient()
|
||||
|
||||
if len(password) >= 40 && isToken(api, password) {
|
||||
return password, nil
|
||||
}
|
||||
|
||||
params := map[string]interface{}{
|
||||
"scopes": []string{"repo"},
|
||||
"note_url": OAuthAppURL,
|
||||
}
|
||||
|
||||
api.PrepareRequest = func(req *http.Request) {
|
||||
req.SetBasicAuth(user, password)
|
||||
if twoFactorCode != "" {
|
||||
req.Header.Set("X-GitHub-OTP", twoFactorCode)
|
||||
}
|
||||
}
|
||||
|
||||
count := 1
|
||||
maxTries := 9
|
||||
for {
|
||||
params["note"], err = authTokenNote(count)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
res, postErr := api.PostJSON("authorizations", params)
|
||||
if postErr != nil {
|
||||
err = postErr
|
||||
break
|
||||
}
|
||||
|
||||
if res.StatusCode == 201 {
|
||||
auth := &AuthorizationEntry{}
|
||||
if err = res.Unmarshal(auth); err != nil {
|
||||
return
|
||||
}
|
||||
token = auth.Token
|
||||
break
|
||||
} else if res.StatusCode == 422 && count < maxTries {
|
||||
count++
|
||||
} else {
|
||||
errInfo, e := res.ErrorInfo()
|
||||
if e == nil {
|
||||
err = errInfo
|
||||
} else {
|
||||
err = e
|
||||
}
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func (client *Client) ensureAccessToken() error {
|
||||
if client.Host.AccessToken == "" {
|
||||
host, err := CurrentConfig().PromptForHost(client.Host.Host)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
client.Host = host
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (client *Client) simpleApi() (c *simpleClient, err error) {
|
||||
err = client.ensureAccessToken()
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
if client.cachedClient != nil {
|
||||
c = client.cachedClient
|
||||
return
|
||||
}
|
||||
|
||||
c = client.apiClient()
|
||||
c.PrepareRequest = func(req *http.Request) {
|
||||
clientDomain := normalizeHost(client.Host.Host)
|
||||
if strings.HasPrefix(clientDomain, "api.github.") {
|
||||
clientDomain = strings.TrimPrefix(clientDomain, "api.")
|
||||
}
|
||||
requestHost := strings.ToLower(req.URL.Host)
|
||||
if requestHost == clientDomain || strings.HasSuffix(requestHost, "."+clientDomain) {
|
||||
req.Header.Set("Authorization", "token "+client.Host.AccessToken)
|
||||
}
|
||||
}
|
||||
|
||||
client.cachedClient = c
|
||||
return
|
||||
}
|
||||
|
||||
func (client *Client) apiClient() *simpleClient {
|
||||
unixSocket := os.ExpandEnv(client.Host.UnixSocket)
|
||||
httpClient := newHttpClient(os.Getenv("HUB_TEST_HOST"), os.Getenv("HUB_VERBOSE") != "", unixSocket)
|
||||
apiRoot := client.absolute(normalizeHost(client.Host.Host))
|
||||
if !strings.HasPrefix(apiRoot.Host, "api.github.") {
|
||||
apiRoot.Path = "/api/v3/"
|
||||
}
|
||||
|
||||
return &simpleClient{
|
||||
httpClient: httpClient,
|
||||
rootUrl: apiRoot,
|
||||
}
|
||||
}
|
||||
|
||||
func (client *Client) absolute(host string) *url.URL {
|
||||
u, err := url.Parse("https://" + host + "/")
|
||||
if err != nil {
|
||||
panic(err)
|
||||
} else if client.Host != nil && client.Host.Protocol != "" {
|
||||
u.Scheme = client.Host.Protocol
|
||||
}
|
||||
return u
|
||||
}
|
||||
|
||||
func normalizeHost(host string) string {
|
||||
if host == "" {
|
||||
return GitHubHost
|
||||
} else if strings.EqualFold(host, GitHubHost) {
|
||||
return "api.github.com"
|
||||
} else if strings.EqualFold(host, "github.localhost") {
|
||||
return "api.github.localhost"
|
||||
} else {
|
||||
return strings.ToLower(host)
|
||||
}
|
||||
}
|
||||
|
||||
func checkStatus(expectedStatus int, action string, response *simpleResponse, err error) error {
|
||||
if err != nil {
|
||||
return fmt.Errorf("Error %s: %s", action, err.Error())
|
||||
} else if response.StatusCode != expectedStatus {
|
||||
errInfo, err := response.ErrorInfo()
|
||||
if err == nil {
|
||||
return FormatError(action, errInfo)
|
||||
} else {
|
||||
return fmt.Errorf("Error %s: %s (HTTP %d)", action, err.Error(), response.StatusCode)
|
||||
}
|
||||
} else {
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
func FormatError(action string, err error) (ee error) {
|
||||
switch e := err.(type) {
|
||||
default:
|
||||
ee = err
|
||||
case *errorInfo:
|
||||
statusCode := e.Response.StatusCode
|
||||
var reason string
|
||||
if s := strings.SplitN(e.Response.Status, " ", 2); len(s) >= 2 {
|
||||
reason = strings.TrimSpace(s[1])
|
||||
}
|
||||
|
||||
errStr := fmt.Sprintf("Error %s: %s (HTTP %d)", action, reason, statusCode)
|
||||
|
||||
var errorSentences []string
|
||||
for _, err := range e.Errors {
|
||||
switch err.Code {
|
||||
case "custom":
|
||||
errorSentences = append(errorSentences, err.Message)
|
||||
case "missing_field":
|
||||
errorSentences = append(errorSentences, fmt.Sprintf("Missing field: \"%s\"", err.Field))
|
||||
case "already_exists":
|
||||
errorSentences = append(errorSentences, fmt.Sprintf("Duplicate value for \"%s\"", err.Field))
|
||||
case "invalid":
|
||||
errorSentences = append(errorSentences, fmt.Sprintf("Invalid value for \"%s\"", err.Field))
|
||||
case "unauthorized":
|
||||
errorSentences = append(errorSentences, fmt.Sprintf("Not allowed to change field \"%s\"", err.Field))
|
||||
}
|
||||
}
|
||||
|
||||
var errorMessage string
|
||||
if len(errorSentences) > 0 {
|
||||
errorMessage = strings.Join(errorSentences, "\n")
|
||||
} else {
|
||||
errorMessage = e.Message
|
||||
if action == "getting current user" && e.Message == "Resource not accessible by integration" {
|
||||
errorMessage = errorMessage + "\nYou must specify GITHUB_USER via environment variable."
|
||||
}
|
||||
}
|
||||
|
||||
if errorMessage != "" {
|
||||
errStr = fmt.Sprintf("%s\n%s", errStr, errorMessage)
|
||||
}
|
||||
|
||||
ee = fmt.Errorf(errStr)
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func authTokenNote(num int) (string, error) {
|
||||
n := os.Getenv("USER")
|
||||
|
||||
if n == "" {
|
||||
n = os.Getenv("USERNAME")
|
||||
}
|
||||
|
||||
if n == "" {
|
||||
whoami := exec.Command("whoami")
|
||||
whoamiOut, err := whoami.Output()
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
n = strings.TrimSpace(string(whoamiOut))
|
||||
}
|
||||
|
||||
h, err := os.Hostname()
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
if num > 1 {
|
||||
return fmt.Sprintf("hub for %s@%s %d", n, h, num), nil
|
||||
}
|
||||
|
||||
return fmt.Sprintf("hub for %s@%s", n, h), nil
|
||||
}
|
||||
|
||||
func perPage(limit, max int) int {
|
||||
if limit > 0 {
|
||||
limit = limit + (limit / 2)
|
||||
if limit < max {
|
||||
return limit
|
||||
}
|
||||
}
|
||||
return max
|
||||
}
|
||||
|
||||
func addQuery(path string, params map[string]interface{}) string {
|
||||
if len(params) == 0 {
|
||||
return path
|
||||
}
|
||||
|
||||
query := url.Values{}
|
||||
for key, value := range params {
|
||||
switch v := value.(type) {
|
||||
case string:
|
||||
query.Add(key, v)
|
||||
case nil:
|
||||
query.Add(key, "")
|
||||
case int:
|
||||
query.Add(key, fmt.Sprintf("%d", v))
|
||||
case bool:
|
||||
query.Add(key, fmt.Sprintf("%v", v))
|
||||
}
|
||||
}
|
||||
|
||||
sep := "?"
|
||||
if strings.Contains(path, sep) {
|
||||
sep = "&"
|
||||
}
|
||||
return path + sep + query.Encode()
|
||||
}
|
||||
409
github/config.go
409
github/config.go
|
|
@ -1,409 +0,0 @@
|
|||
package github
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"fmt"
|
||||
"io/ioutil"
|
||||
"net/url"
|
||||
"os"
|
||||
"os/signal"
|
||||
"path/filepath"
|
||||
"strconv"
|
||||
"strings"
|
||||
"syscall"
|
||||
|
||||
"github.com/github/gh-cli/ui"
|
||||
"github.com/github/gh-cli/utils"
|
||||
"github.com/mitchellh/go-homedir"
|
||||
"golang.org/x/crypto/ssh/terminal"
|
||||
)
|
||||
|
||||
type yamlHost struct {
|
||||
User string `yaml:"user"`
|
||||
OAuthToken string `yaml:"oauth_token"`
|
||||
Protocol string `yaml:"protocol"`
|
||||
UnixSocket string `yaml:"unix_socket,omitempty"`
|
||||
}
|
||||
|
||||
type Host struct {
|
||||
Host string `toml:"host"`
|
||||
User string `toml:"user"`
|
||||
AccessToken string `toml:"access_token"`
|
||||
Protocol string `toml:"protocol"`
|
||||
UnixSocket string `toml:"unix_socket,omitempty"`
|
||||
}
|
||||
|
||||
type Config struct {
|
||||
Hosts []*Host `toml:"hosts"`
|
||||
}
|
||||
|
||||
func (c *Config) PromptForHost(host string) (h *Host, err error) {
|
||||
token := c.DetectToken()
|
||||
tokenFromEnv := token != ""
|
||||
|
||||
if host != GitHubHost {
|
||||
if _, e := url.Parse("https://" + host); e != nil {
|
||||
err = fmt.Errorf("invalid hostname: %q", host)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
h = c.Find(host)
|
||||
if h != nil {
|
||||
if h.User == "" {
|
||||
utils.Check(CheckWriteable(configsFile()))
|
||||
// User is missing from the config: this is a broken config probably
|
||||
// because it was created with an old (broken) version of hub. Let's fix
|
||||
// it now. See issue #1007 for details.
|
||||
user := c.PromptForUser(host)
|
||||
if user == "" {
|
||||
utils.Check(fmt.Errorf("missing user"))
|
||||
}
|
||||
h.User = user
|
||||
err := newConfigService().Save(configsFile(), c)
|
||||
utils.Check(err)
|
||||
}
|
||||
if tokenFromEnv {
|
||||
h.AccessToken = token
|
||||
} else {
|
||||
return
|
||||
}
|
||||
} else {
|
||||
h = &Host{
|
||||
Host: host,
|
||||
AccessToken: token,
|
||||
Protocol: "https",
|
||||
}
|
||||
c.Hosts = append(c.Hosts, h)
|
||||
}
|
||||
|
||||
client := NewClientWithHost(h)
|
||||
|
||||
if !tokenFromEnv {
|
||||
utils.Check(CheckWriteable(configsFile()))
|
||||
err = c.authorizeClient(client, host)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
userFromEnv := os.Getenv("GITHUB_USER")
|
||||
repoFromEnv := os.Getenv("GITHUB_REPOSITORY")
|
||||
if userFromEnv == "" && repoFromEnv != "" {
|
||||
repoParts := strings.SplitN(repoFromEnv, "/", 2)
|
||||
if len(repoParts) > 0 {
|
||||
userFromEnv = repoParts[0]
|
||||
}
|
||||
}
|
||||
if tokenFromEnv && userFromEnv != "" {
|
||||
h.User = userFromEnv
|
||||
} else {
|
||||
var currentUser *User
|
||||
currentUser, err = client.CurrentUser()
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
h.User = currentUser.Login
|
||||
}
|
||||
|
||||
if !tokenFromEnv {
|
||||
err = newConfigService().Save(configsFile(), c)
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func (c *Config) authorizeClient(client *Client, host string) (err error) {
|
||||
user := c.PromptForUser(host)
|
||||
pass := c.PromptForPassword(host, user)
|
||||
|
||||
var code, token string
|
||||
for {
|
||||
token, err = client.FindOrCreateToken(user, pass, code)
|
||||
if err == nil {
|
||||
break
|
||||
}
|
||||
|
||||
if ae, ok := err.(*errorInfo); ok && strings.HasPrefix(ae.Response.Header.Get("X-GitHub-OTP"), "required;") {
|
||||
if code != "" {
|
||||
ui.Errorln("warning: invalid two-factor code")
|
||||
}
|
||||
code = c.PromptForOTP()
|
||||
} else {
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
if err == nil {
|
||||
client.Host.AccessToken = token
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func (c *Config) DetectToken() string {
|
||||
return os.Getenv("GITHUB_TOKEN")
|
||||
}
|
||||
|
||||
func (c *Config) PromptForUser(host string) (user string) {
|
||||
user = os.Getenv("GITHUB_USER")
|
||||
if user != "" {
|
||||
return
|
||||
}
|
||||
|
||||
ui.Printf("%s username: ", host)
|
||||
user = c.scanLine()
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func (c *Config) PromptForPassword(host, user string) (pass string) {
|
||||
pass = os.Getenv("GITHUB_PASSWORD")
|
||||
if pass != "" {
|
||||
return
|
||||
}
|
||||
|
||||
ui.Printf("%s password for %s (never stored): ", host, user)
|
||||
if ui.IsTerminal(os.Stdin) {
|
||||
if password, err := getPassword(); err == nil {
|
||||
pass = password
|
||||
}
|
||||
} else {
|
||||
pass = c.scanLine()
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func (c *Config) PromptForOTP() string {
|
||||
fmt.Print("two-factor authentication code: ")
|
||||
return c.scanLine()
|
||||
}
|
||||
|
||||
func (c *Config) scanLine() string {
|
||||
var line string
|
||||
scanner := bufio.NewScanner(os.Stdin)
|
||||
if scanner.Scan() {
|
||||
line = scanner.Text()
|
||||
}
|
||||
utils.Check(scanner.Err())
|
||||
|
||||
return line
|
||||
}
|
||||
|
||||
func getPassword() (string, error) {
|
||||
stdin := int(syscall.Stdin)
|
||||
initialTermState, err := terminal.GetState(stdin)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
c := make(chan os.Signal)
|
||||
signal.Notify(c, os.Interrupt, syscall.SIGTERM)
|
||||
go func() {
|
||||
s := <-c
|
||||
terminal.Restore(stdin, initialTermState)
|
||||
switch sig := s.(type) {
|
||||
case syscall.Signal:
|
||||
if int(sig) == 2 {
|
||||
fmt.Println("^C")
|
||||
}
|
||||
}
|
||||
os.Exit(1)
|
||||
}()
|
||||
|
||||
passBytes, err := terminal.ReadPassword(stdin)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
signal.Stop(c)
|
||||
fmt.Print("\n")
|
||||
return string(passBytes), nil
|
||||
}
|
||||
|
||||
func (c *Config) Find(host string) *Host {
|
||||
for _, h := range c.Hosts {
|
||||
if h.Host == host {
|
||||
return h
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (c *Config) selectHost() *Host {
|
||||
options := len(c.Hosts)
|
||||
|
||||
if options == 1 {
|
||||
return c.Hosts[0]
|
||||
}
|
||||
|
||||
prompt := "Select host:\n"
|
||||
for idx, host := range c.Hosts {
|
||||
prompt += fmt.Sprintf(" %d. %s\n", idx+1, host.Host)
|
||||
}
|
||||
prompt += fmt.Sprint("> ")
|
||||
|
||||
ui.Printf(prompt)
|
||||
index := c.scanLine()
|
||||
i, err := strconv.Atoi(index)
|
||||
if err != nil || i < 1 || i > options {
|
||||
utils.Check(fmt.Errorf("Error: must enter a number [1-%d]", options))
|
||||
}
|
||||
|
||||
return c.Hosts[i-1]
|
||||
}
|
||||
|
||||
var defaultConfigsFile string
|
||||
|
||||
func configsFile() string {
|
||||
if configFromEnv := os.Getenv("HUB_CONFIG"); configFromEnv != "" {
|
||||
return configFromEnv
|
||||
}
|
||||
if defaultConfigsFile == "" {
|
||||
var err error
|
||||
defaultConfigsFile, err = determineConfigLocation()
|
||||
utils.Check(err)
|
||||
}
|
||||
return defaultConfigsFile
|
||||
}
|
||||
|
||||
func homeConfig() (string, error) {
|
||||
if home, err := homedir.Dir(); err != nil {
|
||||
return "", err
|
||||
} else {
|
||||
return filepath.Join(home, ".config"), nil
|
||||
}
|
||||
}
|
||||
|
||||
func determineConfigLocation() (string, error) {
|
||||
var err error
|
||||
|
||||
xdgHome := os.Getenv("XDG_CONFIG_HOME")
|
||||
configDir := xdgHome
|
||||
if configDir == "" {
|
||||
if configDir, err = homeConfig(); err != nil {
|
||||
return "", err
|
||||
}
|
||||
}
|
||||
|
||||
xdgDirs := os.Getenv("XDG_CONFIG_DIRS")
|
||||
if xdgDirs == "" {
|
||||
xdgDirs = "/etc/xdg"
|
||||
}
|
||||
searchDirs := append([]string{configDir}, strings.Split(xdgDirs, ":")...)
|
||||
|
||||
for _, dir := range searchDirs {
|
||||
filename := filepath.Join(dir, "hub")
|
||||
if _, err := os.Stat(filename); err == nil {
|
||||
return filename, nil
|
||||
}
|
||||
}
|
||||
|
||||
configFile := filepath.Join(configDir, "hub")
|
||||
|
||||
if configDir == xdgHome {
|
||||
if homeDir, _ := homeConfig(); homeDir != "" {
|
||||
legacyConfig := filepath.Join(homeDir, "hub")
|
||||
if _, err = os.Stat(legacyConfig); err == nil {
|
||||
ui.Errorf("Notice: config file found but not respected at: %s\n", legacyConfig)
|
||||
ui.Errorf("You might want to move it to `%s' to avoid re-authenticating.\n", configFile)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return configFile, nil
|
||||
}
|
||||
|
||||
var currentConfig *Config
|
||||
var configLoadedFrom = ""
|
||||
|
||||
func CurrentConfig() *Config {
|
||||
filename := configsFile()
|
||||
if configLoadedFrom != filename {
|
||||
currentConfig = &Config{}
|
||||
newConfigService().Load(filename, currentConfig)
|
||||
configLoadedFrom = filename
|
||||
}
|
||||
|
||||
return currentConfig
|
||||
}
|
||||
|
||||
func (c *Config) DefaultHost() (host *Host, err error) {
|
||||
if GitHubHostEnv != "" {
|
||||
host, err = c.PromptForHost(GitHubHostEnv)
|
||||
} else if len(c.Hosts) > 0 {
|
||||
host = c.selectHost()
|
||||
// HACK: forces host to inherit GITHUB_TOKEN if applicable
|
||||
host, err = c.PromptForHost(host.Host)
|
||||
} else {
|
||||
host, err = c.PromptForHost(DefaultGitHubHost())
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func (c *Config) DefaultHostNoPrompt() (*Host, error) {
|
||||
if GitHubHostEnv != "" {
|
||||
return c.PromptForHost(GitHubHostEnv)
|
||||
} else if len(c.Hosts) > 0 {
|
||||
host := c.Hosts[0]
|
||||
// HACK: forces host to inherit GITHUB_TOKEN if applicable
|
||||
return c.PromptForHost(host.Host)
|
||||
} else {
|
||||
return c.PromptForHost(GitHubHost)
|
||||
}
|
||||
}
|
||||
|
||||
// CheckWriteable checks if config file is writeable. This should
|
||||
// be called before asking for credentials and only if current
|
||||
// operation needs to update the file. See issue #1314 for details.
|
||||
func CheckWriteable(filename string) error {
|
||||
// Check if file exists already. if it doesn't, we will delete it after
|
||||
// checking for writeabilty
|
||||
fileExistsAlready := false
|
||||
|
||||
if _, err := os.Stat(filename); err == nil {
|
||||
fileExistsAlready = true
|
||||
}
|
||||
|
||||
err := os.MkdirAll(filepath.Dir(filename), 0771)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
w, err := os.OpenFile(filename, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0600)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
w.Close()
|
||||
|
||||
if !fileExistsAlready {
|
||||
err := os.Remove(filename)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// Public for testing purpose
|
||||
func CreateTestConfigs(user, token string) *Config {
|
||||
f, _ := ioutil.TempFile("", "test-config")
|
||||
os.Setenv("HUB_CONFIG", f.Name())
|
||||
|
||||
host := &Host{
|
||||
User: "jingweno",
|
||||
AccessToken: "123",
|
||||
Host: GitHubHost,
|
||||
}
|
||||
|
||||
c := &Config{Hosts: []*Host{host}}
|
||||
err := newConfigService().Save(f.Name(), c)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
return c
|
||||
}
|
||||
|
|
@ -1,61 +0,0 @@
|
|||
package github
|
||||
|
||||
import (
|
||||
"io"
|
||||
"io/ioutil"
|
||||
|
||||
"github.com/BurntSushi/toml"
|
||||
"gopkg.in/yaml.v2"
|
||||
)
|
||||
|
||||
type configDecoder interface {
|
||||
Decode(r io.Reader, c *Config) error
|
||||
}
|
||||
|
||||
type tomlConfigDecoder struct {
|
||||
}
|
||||
|
||||
func (t *tomlConfigDecoder) Decode(r io.Reader, c *Config) error {
|
||||
_, err := toml.DecodeReader(r, c)
|
||||
return err
|
||||
}
|
||||
|
||||
type yamlConfigDecoder struct {
|
||||
}
|
||||
|
||||
func (y *yamlConfigDecoder) Decode(r io.Reader, c *Config) error {
|
||||
d, err := ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
yc := yaml.MapSlice{}
|
||||
err = yaml.Unmarshal(d, &yc)
|
||||
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
for _, hostEntry := range yc {
|
||||
v := hostEntry.Value.([]interface{})
|
||||
if len(v) < 1 {
|
||||
continue
|
||||
}
|
||||
host := &Host{Host: hostEntry.Key.(string)}
|
||||
for _, prop := range v[0].(yaml.MapSlice) {
|
||||
switch prop.Key.(string) {
|
||||
case "user":
|
||||
host.User = prop.Value.(string)
|
||||
case "oauth_token":
|
||||
host.AccessToken = prop.Value.(string)
|
||||
case "protocol":
|
||||
host.Protocol = prop.Value.(string)
|
||||
case "unix_socket":
|
||||
host.UnixSocket = prop.Value.(string)
|
||||
}
|
||||
}
|
||||
c.Hosts = append(c.Hosts, host)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
|
@ -1,52 +0,0 @@
|
|||
package github
|
||||
|
||||
import (
|
||||
"io"
|
||||
|
||||
"github.com/BurntSushi/toml"
|
||||
"gopkg.in/yaml.v2"
|
||||
)
|
||||
|
||||
type configEncoder interface {
|
||||
Encode(w io.Writer, c *Config) error
|
||||
}
|
||||
|
||||
type tomlConfigEncoder struct {
|
||||
}
|
||||
|
||||
func (t *tomlConfigEncoder) Encode(w io.Writer, c *Config) error {
|
||||
enc := toml.NewEncoder(w)
|
||||
return enc.Encode(c)
|
||||
}
|
||||
|
||||
type yamlConfigEncoder struct {
|
||||
}
|
||||
|
||||
func (y *yamlConfigEncoder) Encode(w io.Writer, c *Config) error {
|
||||
yc := yaml.MapSlice{}
|
||||
for _, h := range c.Hosts {
|
||||
yc = append(yc, yaml.MapItem{
|
||||
Key: h.Host,
|
||||
Value: []yamlHost{
|
||||
{
|
||||
User: h.User,
|
||||
OAuthToken: h.AccessToken,
|
||||
Protocol: h.Protocol,
|
||||
UnixSocket: h.UnixSocket,
|
||||
},
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
d, err := yaml.Marshal(yc)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
n, err := w.Write(d)
|
||||
if err == nil && n < len(d) {
|
||||
err = io.ErrShortWrite
|
||||
}
|
||||
|
||||
return err
|
||||
}
|
||||
|
|
@ -1,43 +0,0 @@
|
|||
package github
|
||||
|
||||
import (
|
||||
"os"
|
||||
"path/filepath"
|
||||
)
|
||||
|
||||
func newConfigService() *configService {
|
||||
return &configService{
|
||||
Encoder: &yamlConfigEncoder{},
|
||||
Decoder: &yamlConfigDecoder{},
|
||||
}
|
||||
}
|
||||
|
||||
type configService struct {
|
||||
Encoder configEncoder
|
||||
Decoder configDecoder
|
||||
}
|
||||
|
||||
func (s *configService) Save(filename string, c *Config) error {
|
||||
err := os.MkdirAll(filepath.Dir(filename), 0771)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
w, err := os.OpenFile(filename, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0600)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer w.Close()
|
||||
|
||||
return s.Encoder.Encode(w, c)
|
||||
}
|
||||
|
||||
func (s *configService) Load(filename string, c *Config) error {
|
||||
r, err := os.Open(filename)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer r.Close()
|
||||
|
||||
return s.Decoder.Decode(r, c)
|
||||
}
|
||||
|
|
@ -1,64 +0,0 @@
|
|||
package github
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"net/url"
|
||||
"os"
|
||||
"strings"
|
||||
|
||||
"github.com/github/gh-cli/git"
|
||||
)
|
||||
|
||||
var (
|
||||
GitHubHostEnv = os.Getenv("GITHUB_HOST")
|
||||
cachedHosts []string
|
||||
)
|
||||
|
||||
type GithubHostError struct {
|
||||
url *url.URL
|
||||
}
|
||||
|
||||
func (e *GithubHostError) Error() string {
|
||||
return fmt.Sprintf("Invalid GitHub URL: %s", e.url)
|
||||
}
|
||||
|
||||
func knownGitHubHostsInclude(host string) bool {
|
||||
for _, hh := range knownGitHubHosts() {
|
||||
if hh == host {
|
||||
return true
|
||||
}
|
||||
}
|
||||
|
||||
return false
|
||||
}
|
||||
|
||||
func knownGitHubHosts() []string {
|
||||
if cachedHosts != nil {
|
||||
return cachedHosts
|
||||
}
|
||||
|
||||
hosts := []string{}
|
||||
defaultHost := DefaultGitHubHost()
|
||||
hosts = append(hosts, defaultHost)
|
||||
hosts = append(hosts, "ssh.github.com")
|
||||
|
||||
ghHosts, _ := git.ConfigAll("hub.host")
|
||||
for _, ghHost := range ghHosts {
|
||||
ghHost = strings.TrimSpace(ghHost)
|
||||
if ghHost != "" {
|
||||
hosts = append(hosts, ghHost)
|
||||
}
|
||||
}
|
||||
|
||||
cachedHosts = hosts
|
||||
return hosts
|
||||
}
|
||||
|
||||
func DefaultGitHubHost() string {
|
||||
defaultHost := GitHubHostEnv
|
||||
if defaultHost == "" {
|
||||
defaultHost = GitHubHost
|
||||
}
|
||||
|
||||
return defaultHost
|
||||
}
|
||||
490
github/http.go
490
github/http.go
|
|
@ -1,490 +0,0 @@
|
|||
package github
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"context"
|
||||
"crypto/md5"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"net"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"os"
|
||||
"path"
|
||||
"path/filepath"
|
||||
"regexp"
|
||||
"sort"
|
||||
"strconv"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/github/gh-cli/ui"
|
||||
"github.com/github/gh-cli/utils"
|
||||
)
|
||||
|
||||
const apiPayloadVersion = "application/vnd.github.v3+json;charset=utf-8"
|
||||
const patchMediaType = "application/vnd.github.v3.patch;charset=utf-8"
|
||||
const textMediaType = "text/plain;charset=utf-8"
|
||||
const checksType = "application/vnd.github.antiope-preview+json;charset=utf-8"
|
||||
const draftsType = "application/vnd.github.shadow-cat-preview+json;charset=utf-8"
|
||||
const cacheVersion = 2
|
||||
|
||||
var inspectHeaders = []string{
|
||||
"Authorization",
|
||||
"X-GitHub-OTP",
|
||||
"Location",
|
||||
"Link",
|
||||
"Accept",
|
||||
}
|
||||
|
||||
type verboseTransport struct {
|
||||
Transport *http.Transport
|
||||
Verbose bool
|
||||
OverrideURL *url.URL
|
||||
Out io.Writer
|
||||
Colorized bool
|
||||
}
|
||||
|
||||
func (t *verboseTransport) RoundTrip(req *http.Request) (resp *http.Response, err error) {
|
||||
if t.Verbose {
|
||||
t.dumpRequest(req)
|
||||
}
|
||||
|
||||
if t.OverrideURL != nil {
|
||||
port := "80"
|
||||
if s := strings.Split(req.URL.Host, ":"); len(s) > 1 {
|
||||
port = s[1]
|
||||
}
|
||||
|
||||
req = cloneRequest(req)
|
||||
req.Header.Set("X-Original-Scheme", req.URL.Scheme)
|
||||
req.Header.Set("X-Original-Port", port)
|
||||
req.Host = req.URL.Host
|
||||
req.URL.Scheme = t.OverrideURL.Scheme
|
||||
req.URL.Host = t.OverrideURL.Host
|
||||
}
|
||||
|
||||
resp, err = t.Transport.RoundTrip(req)
|
||||
|
||||
if err == nil && t.Verbose {
|
||||
t.dumpResponse(resp)
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func (t *verboseTransport) dumpRequest(req *http.Request) {
|
||||
info := fmt.Sprintf("> %s %s://%s%s", req.Method, req.URL.Scheme, req.URL.Host, req.URL.RequestURI())
|
||||
t.verbosePrintln(info)
|
||||
t.dumpHeaders(req.Header, ">")
|
||||
body := t.dumpBody(req.Body)
|
||||
if body != nil {
|
||||
// reset body since it's been read
|
||||
req.Body = body
|
||||
}
|
||||
}
|
||||
|
||||
func (t *verboseTransport) dumpResponse(resp *http.Response) {
|
||||
info := fmt.Sprintf("< HTTP %d", resp.StatusCode)
|
||||
t.verbosePrintln(info)
|
||||
t.dumpHeaders(resp.Header, "<")
|
||||
body := t.dumpBody(resp.Body)
|
||||
if body != nil {
|
||||
// reset body since it's been read
|
||||
resp.Body = body
|
||||
}
|
||||
}
|
||||
|
||||
func (t *verboseTransport) dumpHeaders(header http.Header, indent string) {
|
||||
for _, listed := range inspectHeaders {
|
||||
for name, vv := range header {
|
||||
if !strings.EqualFold(name, listed) {
|
||||
continue
|
||||
}
|
||||
for _, v := range vv {
|
||||
if v != "" {
|
||||
r := regexp.MustCompile("(?i)^(basic|token) (.+)")
|
||||
if r.MatchString(v) {
|
||||
v = r.ReplaceAllString(v, "$1 [REDACTED]")
|
||||
}
|
||||
|
||||
info := fmt.Sprintf("%s %s: %s", indent, name, v)
|
||||
t.verbosePrintln(info)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func (t *verboseTransport) dumpBody(body io.ReadCloser) io.ReadCloser {
|
||||
if body == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
defer body.Close()
|
||||
buf := new(bytes.Buffer)
|
||||
_, err := io.Copy(buf, body)
|
||||
utils.Check(err)
|
||||
|
||||
if buf.Len() > 0 {
|
||||
t.verbosePrintln(buf.String())
|
||||
}
|
||||
|
||||
return ioutil.NopCloser(buf)
|
||||
}
|
||||
|
||||
func (t *verboseTransport) verbosePrintln(msg string) {
|
||||
if t.Colorized {
|
||||
msg = fmt.Sprintf("\033[36m%s\033[0m", msg)
|
||||
}
|
||||
|
||||
fmt.Fprintln(t.Out, msg)
|
||||
}
|
||||
|
||||
func newHttpClient(testHost string, verbose bool, unixSocket string) *http.Client {
|
||||
var testURL *url.URL
|
||||
if testHost != "" {
|
||||
testURL, _ = url.Parse(testHost)
|
||||
}
|
||||
var httpTransport *http.Transport
|
||||
if unixSocket != "" {
|
||||
dialFunc := func(network, addr string) (net.Conn, error) {
|
||||
return net.Dial("unix", unixSocket)
|
||||
}
|
||||
dialContext := func(_ context.Context, _, _ string) (net.Conn, error) {
|
||||
return net.Dial("unix", unixSocket)
|
||||
}
|
||||
httpTransport = &http.Transport{
|
||||
DialContext: dialContext,
|
||||
DialTLS: dialFunc,
|
||||
ResponseHeaderTimeout: 30 * time.Second,
|
||||
ExpectContinueTimeout: 10 * time.Second,
|
||||
TLSHandshakeTimeout: 10 * time.Second,
|
||||
}
|
||||
} else {
|
||||
httpTransport = &http.Transport{
|
||||
DialContext: (&net.Dialer{
|
||||
Timeout: 30 * time.Second,
|
||||
KeepAlive: 30 * time.Second,
|
||||
}).DialContext,
|
||||
TLSHandshakeTimeout: 10 * time.Second,
|
||||
}
|
||||
}
|
||||
tr := &verboseTransport{
|
||||
Transport: httpTransport,
|
||||
Verbose: verbose,
|
||||
OverrideURL: testURL,
|
||||
Out: ui.Stderr,
|
||||
Colorized: ui.IsTerminal(os.Stderr),
|
||||
}
|
||||
|
||||
return &http.Client{
|
||||
Transport: tr,
|
||||
}
|
||||
}
|
||||
|
||||
func cloneRequest(req *http.Request) *http.Request {
|
||||
dup := new(http.Request)
|
||||
*dup = *req
|
||||
dup.URL, _ = url.Parse(req.URL.String())
|
||||
dup.Header = make(http.Header)
|
||||
for k, s := range req.Header {
|
||||
dup.Header[k] = s
|
||||
}
|
||||
return dup
|
||||
}
|
||||
|
||||
type simpleClient struct {
|
||||
httpClient *http.Client
|
||||
rootUrl *url.URL
|
||||
PrepareRequest func(*http.Request)
|
||||
CacheTTL int
|
||||
}
|
||||
|
||||
func (c *simpleClient) performRequest(method, path string, body io.Reader, configure func(*http.Request)) (*simpleResponse, error) {
|
||||
url, err := url.Parse(path)
|
||||
if err == nil {
|
||||
url = c.rootUrl.ResolveReference(url)
|
||||
return c.performRequestUrl(method, url, body, configure)
|
||||
} else {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
func (c *simpleClient) performRequestUrl(method string, url *url.URL, body io.Reader, configure func(*http.Request)) (res *simpleResponse, err error) {
|
||||
req, err := http.NewRequest(method, url.String(), body)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
if c.PrepareRequest != nil {
|
||||
c.PrepareRequest(req)
|
||||
}
|
||||
req.Header.Set("User-Agent", userAgent)
|
||||
req.Header.Set("Accept", apiPayloadVersion)
|
||||
|
||||
if configure != nil {
|
||||
configure(req)
|
||||
}
|
||||
|
||||
key := cacheKey(req)
|
||||
if cachedResponse := c.cacheRead(key, req); cachedResponse != nil {
|
||||
res = &simpleResponse{cachedResponse}
|
||||
return
|
||||
}
|
||||
|
||||
httpResponse, err := c.httpClient.Do(req)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
c.cacheWrite(key, httpResponse)
|
||||
res = &simpleResponse{httpResponse}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func isGraphQL(req *http.Request) bool {
|
||||
return req.URL.Path == "/graphql"
|
||||
}
|
||||
|
||||
func canCache(req *http.Request) bool {
|
||||
return strings.EqualFold(req.Method, "GET") || isGraphQL(req)
|
||||
}
|
||||
|
||||
func (c *simpleClient) cacheRead(key string, req *http.Request) (res *http.Response) {
|
||||
if c.CacheTTL > 0 && canCache(req) {
|
||||
f := cacheFile(key)
|
||||
cacheInfo, err := os.Stat(f)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
if time.Since(cacheInfo.ModTime()).Seconds() > float64(c.CacheTTL) {
|
||||
return
|
||||
}
|
||||
cf, err := os.Open(f)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
defer cf.Close()
|
||||
|
||||
cb, err := ioutil.ReadAll(cf)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
parts := strings.SplitN(string(cb), "\r\n\r\n", 2)
|
||||
if len(parts) < 2 {
|
||||
return
|
||||
}
|
||||
|
||||
res = &http.Response{
|
||||
Body: ioutil.NopCloser(bytes.NewBufferString(parts[1])),
|
||||
Header: http.Header{},
|
||||
}
|
||||
headerLines := strings.Split(parts[0], "\r\n")
|
||||
if len(headerLines) < 1 {
|
||||
return
|
||||
}
|
||||
if proto := strings.SplitN(headerLines[0], " ", 3); len(proto) >= 3 {
|
||||
res.Proto = proto[0]
|
||||
res.Status = fmt.Sprintf("%s %s", proto[1], proto[2])
|
||||
if code, _ := strconv.Atoi(proto[1]); code > 0 {
|
||||
res.StatusCode = code
|
||||
}
|
||||
}
|
||||
for _, line := range headerLines[1:] {
|
||||
kv := strings.SplitN(line, ":", 2)
|
||||
if len(kv) >= 2 {
|
||||
res.Header.Add(kv[0], strings.TrimLeft(kv[1], " "))
|
||||
}
|
||||
}
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
func (c *simpleClient) cacheWrite(key string, res *http.Response) {
|
||||
if c.CacheTTL > 0 && canCache(res.Request) && res.StatusCode < 500 && res.StatusCode != 403 {
|
||||
bodyCopy := &bytes.Buffer{}
|
||||
bodyReplacement := readCloserCallback{
|
||||
Reader: io.TeeReader(res.Body, bodyCopy),
|
||||
Closer: res.Body,
|
||||
Callback: func() {
|
||||
f := cacheFile(key)
|
||||
err := os.MkdirAll(filepath.Dir(f), 0771)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
cf, err := os.OpenFile(f, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0600)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
defer cf.Close()
|
||||
fmt.Fprintf(cf, "%s %s\r\n", res.Proto, res.Status)
|
||||
res.Header.Write(cf)
|
||||
fmt.Fprintf(cf, "\r\n")
|
||||
io.Copy(cf, bodyCopy)
|
||||
},
|
||||
}
|
||||
res.Body = &bodyReplacement
|
||||
}
|
||||
}
|
||||
|
||||
type readCloserCallback struct {
|
||||
Callback func()
|
||||
Closer io.Closer
|
||||
io.Reader
|
||||
}
|
||||
|
||||
func (rc *readCloserCallback) Close() error {
|
||||
err := rc.Closer.Close()
|
||||
if err == nil {
|
||||
rc.Callback()
|
||||
}
|
||||
return err
|
||||
}
|
||||
|
||||
func cacheKey(req *http.Request) string {
|
||||
path := strings.Replace(req.URL.EscapedPath(), "/", "-", -1)
|
||||
if len(path) > 1 {
|
||||
path = strings.TrimPrefix(path, "-")
|
||||
}
|
||||
host := req.Host
|
||||
if host == "" {
|
||||
host = req.URL.Host
|
||||
}
|
||||
hash := md5.New()
|
||||
fmt.Fprintf(hash, "%d:", cacheVersion)
|
||||
io.WriteString(hash, req.Header.Get("Accept"))
|
||||
io.WriteString(hash, req.Header.Get("Authorization"))
|
||||
queryParts := strings.Split(req.URL.RawQuery, "&")
|
||||
sort.Strings(queryParts)
|
||||
for _, q := range queryParts {
|
||||
fmt.Fprintf(hash, "%s&", q)
|
||||
}
|
||||
if isGraphQL(req) && req.Body != nil {
|
||||
if b, err := ioutil.ReadAll(req.Body); err == nil {
|
||||
req.Body = ioutil.NopCloser(bytes.NewBuffer(b))
|
||||
hash.Write(b)
|
||||
}
|
||||
}
|
||||
return fmt.Sprintf("%s/%s_%x", host, path, hash.Sum(nil))
|
||||
}
|
||||
|
||||
func cacheFile(key string) string {
|
||||
return path.Join(os.TempDir(), "hub", "api", key)
|
||||
}
|
||||
|
||||
func (c *simpleClient) jsonRequest(method, path string, body interface{}, configure func(*http.Request)) (*simpleResponse, error) {
|
||||
json, err := json.Marshal(body)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
buf := bytes.NewBuffer(json)
|
||||
|
||||
return c.performRequest(method, path, buf, func(req *http.Request) {
|
||||
req.Header.Set("Content-Type", "application/json; charset=utf-8")
|
||||
if configure != nil {
|
||||
configure(req)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func (c *simpleClient) Get(path string) (*simpleResponse, error) {
|
||||
return c.performRequest("GET", path, nil, nil)
|
||||
}
|
||||
|
||||
func (c *simpleClient) GetFile(path string, mimeType string) (*simpleResponse, error) {
|
||||
return c.performRequest("GET", path, nil, func(req *http.Request) {
|
||||
req.Header.Set("Accept", mimeType)
|
||||
})
|
||||
}
|
||||
|
||||
func (c *simpleClient) PostJSON(path string, payload interface{}) (*simpleResponse, error) {
|
||||
return c.jsonRequest("POST", path, payload, nil)
|
||||
}
|
||||
|
||||
func (c *simpleClient) PostJSONPreview(path string, payload interface{}, mimeType string) (*simpleResponse, error) {
|
||||
return c.jsonRequest("POST", path, payload, func(req *http.Request) {
|
||||
req.Header.Set("Accept", mimeType)
|
||||
})
|
||||
}
|
||||
|
||||
func (c *simpleClient) PatchJSON(path string, payload interface{}) (*simpleResponse, error) {
|
||||
return c.jsonRequest("PATCH", path, payload, nil)
|
||||
}
|
||||
|
||||
type simpleResponse struct {
|
||||
*http.Response
|
||||
}
|
||||
|
||||
type errorInfo struct {
|
||||
Message string `json:"message"`
|
||||
Errors []fieldError `json:"errors"`
|
||||
Response *http.Response
|
||||
}
|
||||
type errorInfoSimple struct {
|
||||
Message string `json:"message"`
|
||||
Errors []string `json:"errors"`
|
||||
}
|
||||
type fieldError struct {
|
||||
Resource string `json:"resource"`
|
||||
Message string `json:"message"`
|
||||
Code string `json:"code"`
|
||||
Field string `json:"field"`
|
||||
}
|
||||
|
||||
func (e *errorInfo) Error() string {
|
||||
return e.Message
|
||||
}
|
||||
|
||||
func (res *simpleResponse) Unmarshal(dest interface{}) (err error) {
|
||||
defer res.Body.Close()
|
||||
|
||||
body, err := ioutil.ReadAll(res.Body)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
return json.Unmarshal(body, dest)
|
||||
}
|
||||
|
||||
func (res *simpleResponse) ErrorInfo() (msg *errorInfo, err error) {
|
||||
defer res.Body.Close()
|
||||
|
||||
body, err := ioutil.ReadAll(res.Body)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
msg = &errorInfo{}
|
||||
err = json.Unmarshal(body, msg)
|
||||
if err != nil {
|
||||
msgSimple := &errorInfoSimple{}
|
||||
if err = json.Unmarshal(body, msgSimple); err == nil {
|
||||
msg.Message = msgSimple.Message
|
||||
for _, errMsg := range msgSimple.Errors {
|
||||
msg.Errors = append(msg.Errors, fieldError{
|
||||
Code: "custom",
|
||||
Message: errMsg,
|
||||
})
|
||||
}
|
||||
}
|
||||
}
|
||||
if err == nil {
|
||||
msg.Response = res.Response
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func (res *simpleResponse) Link(name string) string {
|
||||
linkVal := res.Header.Get("Link")
|
||||
re := regexp.MustCompile(`<([^>]+)>; rel="([^"]+)"`)
|
||||
for _, match := range re.FindAllStringSubmatch(linkVal, -1) {
|
||||
if match[2] == name {
|
||||
return match[1]
|
||||
}
|
||||
}
|
||||
return ""
|
||||
}
|
||||
|
|
@ -1,119 +0,0 @@
|
|||
package github
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"net/url"
|
||||
"strings"
|
||||
|
||||
"github.com/github/gh-cli/utils"
|
||||
)
|
||||
|
||||
type Project struct {
|
||||
Name string
|
||||
Owner string
|
||||
Host string
|
||||
Protocol string
|
||||
}
|
||||
|
||||
func (p Project) String() string {
|
||||
return fmt.Sprintf("%s/%s", p.Owner, p.Name)
|
||||
}
|
||||
|
||||
func (p *Project) SameAs(other *Project) bool {
|
||||
return strings.ToLower(p.Owner) == strings.ToLower(other.Owner) &&
|
||||
strings.ToLower(p.Name) == strings.ToLower(other.Name) &&
|
||||
strings.ToLower(p.Host) == strings.ToLower(other.Host)
|
||||
}
|
||||
|
||||
func (p *Project) WebURL(name, owner, path string) string {
|
||||
if owner == "" {
|
||||
owner = p.Owner
|
||||
}
|
||||
if name == "" {
|
||||
name = p.Name
|
||||
}
|
||||
|
||||
ownerWithName := fmt.Sprintf("%s/%s", owner, name)
|
||||
url := fmt.Sprintf("%s://%s", p.Protocol, utils.ConcatPaths(p.Host, ownerWithName))
|
||||
if path != "" {
|
||||
url = utils.ConcatPaths(url, path)
|
||||
}
|
||||
|
||||
return url
|
||||
}
|
||||
|
||||
func (p *Project) GitURL(name, owner string, isSSH bool) string {
|
||||
return p.WebURL(name, owner, "") + ".git"
|
||||
}
|
||||
|
||||
func NewProjectFromURL(url *url.URL) (p *Project, err error) {
|
||||
if !knownGitHubHostsInclude(url.Host) {
|
||||
err = &GithubHostError{url}
|
||||
return
|
||||
}
|
||||
|
||||
parts := strings.SplitN(url.Path, "/", 4)
|
||||
if len(parts) <= 2 {
|
||||
err = fmt.Errorf("Invalid GitHub URL: %s", url)
|
||||
return
|
||||
}
|
||||
|
||||
name := strings.TrimSuffix(parts[2], ".git")
|
||||
p = newProject(parts[1], name, url.Host, url.Scheme)
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func NewProject(owner, name, host string) *Project {
|
||||
return newProject(owner, name, host, "")
|
||||
}
|
||||
|
||||
func newProject(owner, name, host, protocol string) *Project {
|
||||
if strings.Contains(owner, "/") {
|
||||
result := strings.SplitN(owner, "/", 2)
|
||||
owner = result[0]
|
||||
if name == "" {
|
||||
name = result[1]
|
||||
}
|
||||
} else if strings.Contains(name, "/") {
|
||||
result := strings.SplitN(name, "/", 2)
|
||||
if owner == "" {
|
||||
owner = result[0]
|
||||
}
|
||||
name = result[1]
|
||||
}
|
||||
|
||||
if host == "" {
|
||||
host = DefaultGitHubHost()
|
||||
}
|
||||
if host == "ssh.github.com" {
|
||||
host = GitHubHost
|
||||
}
|
||||
|
||||
if protocol != "http" && protocol != "https" {
|
||||
protocol = ""
|
||||
}
|
||||
if protocol == "" {
|
||||
h := CurrentConfig().Find(host)
|
||||
if h != nil {
|
||||
protocol = h.Protocol
|
||||
}
|
||||
}
|
||||
if protocol == "" {
|
||||
protocol = "https"
|
||||
}
|
||||
|
||||
if owner == "" {
|
||||
h := CurrentConfig().Find(host)
|
||||
if h != nil {
|
||||
owner = h.User
|
||||
}
|
||||
}
|
||||
|
||||
return &Project{
|
||||
Name: name,
|
||||
Owner: owner,
|
||||
Host: host,
|
||||
Protocol: protocol,
|
||||
}
|
||||
}
|
||||
101
github/remote.go
101
github/remote.go
|
|
@ -1,101 +0,0 @@
|
|||
package github
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"net/url"
|
||||
"regexp"
|
||||
"strings"
|
||||
|
||||
"github.com/github/gh-cli/git"
|
||||
)
|
||||
|
||||
var (
|
||||
OriginNamesInLookupOrder = []string{"upstream", "github", "origin"}
|
||||
)
|
||||
|
||||
type Remote struct {
|
||||
Name string
|
||||
URL *url.URL
|
||||
PushURL *url.URL
|
||||
}
|
||||
|
||||
func (remote *Remote) String() string {
|
||||
return remote.Name
|
||||
}
|
||||
|
||||
func (remote *Remote) Project() (*Project, error) {
|
||||
p, err := NewProjectFromURL(remote.URL)
|
||||
if _, ok := err.(*GithubHostError); ok {
|
||||
return NewProjectFromURL(remote.PushURL)
|
||||
}
|
||||
return p, err
|
||||
}
|
||||
|
||||
func Remotes() (remotes []Remote, err error) {
|
||||
re := regexp.MustCompile(`(.+)\s+(.+)\s+\((push|fetch)\)`)
|
||||
|
||||
rs, err := git.Remotes()
|
||||
if err != nil {
|
||||
err = fmt.Errorf("Can't load git remote")
|
||||
return
|
||||
}
|
||||
|
||||
// build the remotes map
|
||||
remotesMap := make(map[string]map[string]string)
|
||||
for _, r := range rs {
|
||||
if re.MatchString(r) {
|
||||
match := re.FindStringSubmatch(r)
|
||||
name := strings.TrimSpace(match[1])
|
||||
url := strings.TrimSpace(match[2])
|
||||
urlType := strings.TrimSpace(match[3])
|
||||
utm, ok := remotesMap[name]
|
||||
if !ok {
|
||||
utm = make(map[string]string)
|
||||
remotesMap[name] = utm
|
||||
}
|
||||
utm[urlType] = url
|
||||
}
|
||||
}
|
||||
|
||||
// construct remotes in priority order
|
||||
names := OriginNamesInLookupOrder
|
||||
for _, name := range names {
|
||||
if u, ok := remotesMap[name]; ok {
|
||||
r, err := newRemote(name, u)
|
||||
if err == nil {
|
||||
remotes = append(remotes, r)
|
||||
delete(remotesMap, name)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// the rest of the remotes
|
||||
for n, u := range remotesMap {
|
||||
r, err := newRemote(n, u)
|
||||
if err == nil {
|
||||
remotes = append(remotes, r)
|
||||
}
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func newRemote(name string, urlMap map[string]string) (Remote, error) {
|
||||
r := Remote{}
|
||||
|
||||
fetchURL, ferr := git.ParseURL(urlMap["fetch"])
|
||||
pushURL, perr := git.ParseURL(urlMap["push"])
|
||||
if ferr != nil && perr != nil {
|
||||
return r, fmt.Errorf("No valid remote URLs")
|
||||
}
|
||||
|
||||
r.Name = name
|
||||
if ferr == nil {
|
||||
r.URL = fetchURL
|
||||
}
|
||||
if perr == nil {
|
||||
r.PushURL = pushURL
|
||||
}
|
||||
|
||||
return r, nil
|
||||
}
|
||||
4
go.mod
4
go.mod
|
|
@ -3,13 +3,11 @@ module github.com/github/gh-cli
|
|||
go 1.13
|
||||
|
||||
require (
|
||||
github.com/BurntSushi/toml v0.3.1
|
||||
github.com/gookit/color v1.2.0
|
||||
github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51
|
||||
github.com/mattn/go-colorable v0.1.2
|
||||
github.com/mattn/go-isatty v0.0.9
|
||||
github.com/mitchellh/go-homedir v1.1.0
|
||||
github.com/spf13/cobra v0.0.5
|
||||
golang.org/x/crypto v0.0.0-20190926180335-cea2066c6411
|
||||
gopkg.in/yaml.v2 v2.2.2
|
||||
gopkg.in/yaml.v3 v3.0.0-20191010095647-fc94e3f71652
|
||||
)
|
||||
|
|
|
|||
8
go.sum
8
go.sum
|
|
@ -44,14 +44,8 @@ github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UV
|
|||
github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0=
|
||||
github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q=
|
||||
golang.org/x/crypto v0.0.0-20181203042331-505ab145d0a9/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
||||
golang.org/x/crypto v0.0.0-20190926180335-cea2066c6411 h1:kuW9k4QvBJpRjC3rxEytsfIYPs8oGY3Jw7iR36h0FIY=
|
||||
golang.org/x/crypto v0.0.0-20190926180335-cea2066c6411/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
||||
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
||||
golang.org/x/sys v0.0.0-20181205085412-a5c9d58dba9a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||
golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20190813064441-fde4db37ae7a h1:aYOabOQFp6Vj6W1F80affTUvO9UxmJRx8K0gsfABByQ=
|
||||
golang.org/x/sys v0.0.0-20190813064441-fde4db37ae7a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||
|
|
@ -59,3 +53,5 @@ gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+
|
|||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||
gopkg.in/yaml.v2 v2.2.2 h1:ZCJp+EgiOT7lHqUV2J862kp8Qj64Jo6az82+3Td9dZw=
|
||||
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||
gopkg.in/yaml.v3 v3.0.0-20191010095647-fc94e3f71652 h1:VKvJ/mQ4BgCjZUDggYFxTe0qv9jPMHsZPD4Xt91Y5H4=
|
||||
gopkg.in/yaml.v3 v3.0.0-20191010095647-fc94e3f71652/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||
|
|
|
|||
|
|
@ -10,7 +10,6 @@ import (
|
|||
"strings"
|
||||
|
||||
"github.com/github/gh-cli/api"
|
||||
"github.com/github/gh-cli/github"
|
||||
"github.com/spf13/cobra"
|
||||
)
|
||||
|
||||
|
|
@ -20,8 +19,6 @@ type TempGitRepo struct {
|
|||
}
|
||||
|
||||
func UseTempGitRepo() *TempGitRepo {
|
||||
github.CreateTestConfigs("mario", "i-love-peach")
|
||||
|
||||
pwd, _ := os.Getwd()
|
||||
oldEnv := make(map[string]string)
|
||||
overrideEnv := func(name, value string) {
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue