Spaces:
Paused
Paused
| // Copyright 2023 Woodpecker Authors | |
| // | |
| // Licensed under the Apache License, Version 2.0 (the "License"); | |
| // you may not use this file except in compliance with the License. | |
| // You may obtain a copy of the License at | |
| // | |
| // http://www.apache.org/licenses/LICENSE-2.0 | |
| // | |
| // Unless required by applicable law or agreed to in writing, software | |
| // distributed under the License is distributed on an "AS IS" BASIS, | |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
| // See the License for the specific language governing permissions and | |
| // limitations under the License. | |
| package internal | |
| import ( | |
| "context" | |
| "crypto/tls" | |
| "crypto/x509" | |
| "fmt" | |
| "net/http" | |
| "os/exec" | |
| "strconv" | |
| "strings" | |
| vsc_url "github.com/gitsight/go-vcsurl" | |
| "github.com/rs/zerolog/log" | |
| "github.com/urfave/cli/v3" | |
| "golang.org/x/net/proxy" | |
| "golang.org/x/oauth2" | |
| "go.woodpecker-ci.org/woodpecker/v3/shared/httputil" | |
| "go.woodpecker-ci.org/woodpecker/v3/woodpecker-go/woodpecker" | |
| ) | |
| // NewClient returns a new client from the CLI context. | |
| func NewClient(ctx context.Context, c *cli.Command) (woodpecker.Client, error) { | |
| var ( | |
| skip = c.Bool("skip-verify") | |
| socks = c.String("socks-proxy") | |
| socksOff = c.Bool("socks-proxy-off") | |
| token = c.String("token") | |
| server = c.String("server") | |
| ) | |
| server = strings.TrimRight(server, "/") | |
| // if no server url is provided we can default | |
| // to the hosted Woodpecker service. | |
| if len(server) == 0 { | |
| return nil, fmt.Errorf("you must provide the Woodpecker server address") | |
| } | |
| if len(token) == 0 { | |
| return nil, fmt.Errorf("you must provide your Woodpecker access token") | |
| } | |
| // attempt to find system CA certs | |
| certs, err := x509.SystemCertPool() | |
| if err != nil { | |
| log.Error().Err(err).Msg("failed to find system CA certs") | |
| } | |
| tlsConfig := &tls.Config{ | |
| RootCAs: certs, | |
| InsecureSkipVerify: skip, | |
| } | |
| config := new(oauth2.Config) | |
| client := config.Client(ctx, | |
| &oauth2.Token{ | |
| AccessToken: token, | |
| }, | |
| ) | |
| trans, _ := client.Transport.(*oauth2.Transport) | |
| var baseTransport http.RoundTripper | |
| if len(socks) != 0 && !socksOff { | |
| dialer, err := proxy.SOCKS5("tcp", socks, nil, proxy.Direct) | |
| if err != nil { | |
| return nil, err | |
| } | |
| baseTransport = &http.Transport{ | |
| TLSClientConfig: tlsConfig, | |
| Proxy: http.ProxyFromEnvironment, | |
| Dial: dialer.Dial, | |
| } | |
| } else { | |
| baseTransport = &http.Transport{ | |
| TLSClientConfig: tlsConfig, | |
| Proxy: http.ProxyFromEnvironment, | |
| } | |
| } | |
| // Wrap the base transport with User-Agent support | |
| trans.Base = httputil.NewUserAgentRoundTripper(baseTransport, "cli") | |
| return woodpecker.NewClient(server, client), nil | |
| } | |
| func getRepoFromGit(remoteName string) (string, error) { | |
| cmd := exec.Command("git", "remote", "get-url", remoteName) | |
| stdout, err := cmd.Output() | |
| if err != nil { | |
| return "", fmt.Errorf("could not get remote url: %w", err) | |
| } | |
| gitRemote := strings.TrimSpace(string(stdout)) | |
| log.Debug().Str("git-remote", gitRemote).Msg("extracted remote url from git") | |
| if len(gitRemote) == 0 { | |
| return "", fmt.Errorf("no repository provided") | |
| } | |
| u, err := vsc_url.Parse(gitRemote) | |
| if err != nil { | |
| return "", fmt.Errorf("could not parse git remote url: %w", err) | |
| } | |
| repoFullName := u.FullName | |
| log.Debug().Str("repo", repoFullName).Msg("extracted repository from remote url") | |
| return repoFullName, nil | |
| } | |
| // ParseRepo parses the repository owner and name from a string. | |
| func ParseRepo(client woodpecker.Client, str string) (repoID int64, err error) { | |
| if str == "" { | |
| str, err = getRepoFromGit("upstream") | |
| if err != nil { | |
| log.Debug().Err(err).Msg("could not get repository from git upstream remote") | |
| } | |
| } | |
| if str == "" { | |
| str, err = getRepoFromGit("origin") | |
| if err != nil { | |
| log.Debug().Err(err).Msg("could not get repository from git origin remote") | |
| } | |
| } | |
| if str == "" { | |
| return 0, fmt.Errorf("no repository provided") | |
| } | |
| if strings.Contains(str, "/") { | |
| repo, err := client.RepoLookup(str) | |
| if err != nil { | |
| return 0, err | |
| } | |
| return repo.ID, nil | |
| } | |
| return strconv.ParseInt(str, 10, 64) | |
| } | |
| // ParseKeyPair parses a key=value pair. | |
| func ParseKeyPair(p []string) map[string]string { | |
| params := map[string]string{} | |
| for _, i := range p { | |
| before, after, ok := strings.Cut(i, "=") | |
| if !ok || before == "" { | |
| continue | |
| } | |
| params[before] = after | |
| } | |
| return params | |
| } | |
| /* | |
| ParseStep parses the step id form a string which may either be the step PID (step number) or a step name. | |
| These rules apply: | |
| - Step PID take precedence over step name when searching for a match. | |
| - First match is used, when there are multiple steps with the same name. | |
| Strictly speaking, this is not parsing, but a lookup. | |
| */ | |
| func ParseStep(client woodpecker.Client, repoID, number int64, stepArg string) (stepID int64, err error) { | |
| pipeline, err := client.Pipeline(repoID, number) | |
| if err != nil { | |
| return 0, err | |
| } | |
| stepPID, err := strconv.ParseInt(stepArg, 10, 64) | |
| if err == nil { | |
| for _, wf := range pipeline.Workflows { | |
| for _, step := range wf.Children { | |
| if int64(step.PID) == stepPID { | |
| return step.ID, nil | |
| } | |
| } | |
| } | |
| } | |
| for _, wf := range pipeline.Workflows { | |
| for _, step := range wf.Children { | |
| if step.Name == stepArg { | |
| return step.ID, nil | |
| } | |
| } | |
| } | |
| return 0, fmt.Errorf("no step with number or name '%s' found", stepArg) | |
| } | |