// Copyright 2020 The Gitea Authors. All rights reserved. // SPDX-License-Identifier: MIT package config import ( "crypto/tls" "errors" "fmt" "log" "net/http" "net/http/cookiejar" "net/url" "os" "strings" "code.gitea.io/sdk/gitea" "code.gitea.io/tea/modules/utils" "github.com/AlecAivazis/survey/v2" ) // Login represents a login to a gitea server, you even could add multiple logins for one gitea server type Login struct { Name string `yaml:"name"` URL string `yaml:"url"` Token string `yaml:"token"` Default bool `yaml:"default"` SSHHost string `yaml:"ssh_host"` // optional path to the private key SSHKey string `yaml:"ssh_key"` Insecure bool `yaml:"insecure"` SSHCertPrincipal string `yaml:"ssh_certificate_principal"` SSHAgent bool `yaml:"ssh_agent"` SSHKeyFingerprint string `yaml:"ssh_key_agent_pub"` SSHPassphrase string `yaml:"-"` VersionCheck bool `yaml:"version_check"` // User is username from gitea User string `yaml:"user"` // Created is auto created unix timestamp Created int64 `yaml:"created"` } // GetLogins return all login available by config func GetLogins() ([]Login, error) { if err := loadConfig(); err != nil { return nil, err } return config.Logins, nil } // GetDefaultLogin return the default login func GetDefaultLogin() (*Login, error) { if err := loadConfig(); err != nil { return nil, err } if len(config.Logins) == 0 { return nil, errors.New("No available login") } for _, l := range config.Logins { if l.Default { return &l, nil } } return &config.Logins[0], nil } // SetDefaultLogin set the default login by name (case insensitive) func SetDefaultLogin(name string) error { if err := loadConfig(); err != nil { return err } loginExist := false for i := range config.Logins { config.Logins[i].Default = false if strings.ToLower(config.Logins[i].Name) == strings.ToLower(name) { config.Logins[i].Default = true loginExist = true } } if !loginExist { return fmt.Errorf("login '%s' not found", name) } return saveConfig() } // GetLoginByName get login by name (case insensitive) func GetLoginByName(name string) *Login { err := loadConfig() if err != nil { log.Fatal(err) } for _, l := range config.Logins { if strings.ToLower(l.Name) == strings.ToLower(name) { return &l } } return nil } // GetLoginByToken get login by token func GetLoginByToken(token string) *Login { err := loadConfig() if err != nil { log.Fatal(err) } for _, l := range config.Logins { if l.Token == token { return &l } } return nil } // GetLoginByHost finds a login by it's server URL func GetLoginByHost(host string) *Login { err := loadConfig() if err != nil { log.Fatal(err) } for _, l := range config.Logins { loginURL, err := url.Parse(l.URL) if err != nil { log.Fatal(err) } if loginURL.Host == host { return &l } } return nil } // DeleteLogin delete a login by name from config func DeleteLogin(name string) error { idx := -1 for i, l := range config.Logins { if l.Name == name { idx = i break } } if idx == -1 { return fmt.Errorf("can not delete login '%s', does not exist", name) } config.Logins = append(config.Logins[:idx], config.Logins[idx+1:]...) return saveConfig() } // AddLogin save a login to config func AddLogin(login *Login) error { if err := loadConfig(); err != nil { return err } // save login to global var config.Logins = append(config.Logins, *login) // save login to config file return saveConfig() } // Client returns a client to operate Gitea API. You may provide additional modifiers // for the client like gitea.SetBasicAuth() for customization func (l *Login) Client(options ...gitea.ClientOption) *gitea.Client { httpClient := &http.Client{} if l.Insecure { cookieJar, _ := cookiejar.New(nil) httpClient = &http.Client{ Jar: cookieJar, Transport: &http.Transport{ TLSClientConfig: &tls.Config{InsecureSkipVerify: true}, }, } } // versioncheck must be prepended in options to make sure we don't hit any version checks in the sdk if !l.VersionCheck { options = append([]gitea.ClientOption{gitea.SetGiteaVersion("")}, options...) } options = append(options, gitea.SetToken(l.Token), gitea.SetHTTPClient(httpClient)) if ok, err := utils.IsKeyEncrypted(l.SSHKey); ok && err == nil && l.SSHPassphrase == "" { promptPW := &survey.Password{Message: "ssh-key is encrypted please enter the passphrase: "} if err = survey.AskOne(promptPW, &l.SSHPassphrase, survey.WithValidator(survey.Required)); err != nil { log.Fatal(err) } } if l.SSHCertPrincipal != "" { options = append(options, gitea.UseSSHCert(l.SSHCertPrincipal, l.SSHKey, l.SSHPassphrase)) } if l.SSHKeyFingerprint != "" { options = append(options, gitea.UseSSHPubkey(l.SSHKeyFingerprint, l.SSHKey, l.SSHPassphrase)) } client, err := gitea.NewClient(l.URL, options...) if err != nil { var versionError *gitea.ErrUnknownVersion if !errors.As(err, &versionError) { log.Fatal(err) } fmt.Fprintf(os.Stderr, "WARNING: could not detect gitea version: %s\nINFO: set gitea version: to last supported one\n", versionError) } return client } // GetSSHHost returns SSH host name func (l *Login) GetSSHHost() string { if l.SSHHost != "" { return l.SSHHost } u, err := url.Parse(l.URL) if err != nil { return "" } return u.Host }