// Copyright 2019 Sorint.lab // // 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 agolagit import ( "crypto/tls" "fmt" "io" "io/ioutil" "net" "net/http" "net/url" "regexp" "strings" "time" gitsource "agola.io/agola/internal/gitsources" "agola.io/agola/internal/services/types" errors "golang.org/x/xerrors" ) var ( branchRefPrefix = "refs/heads/" tagRefPrefix = "refs/tags/" ) type Client struct { url string client *http.Client pullRequestRefRegexes []*regexp.Regexp } // NewClient initializes and returns a API client. func New(url string, pullRequestRefRegexes []*regexp.Regexp) *Client { // copied from net/http until it has a clone function: https://github.com/golang/go/issues/26013 transport := &http.Transport{ Proxy: http.ProxyFromEnvironment, DialContext: (&net.Dialer{ Timeout: 30 * time.Second, KeepAlive: 30 * time.Second, DualStack: true, }).DialContext, MaxIdleConns: 100, IdleConnTimeout: 90 * time.Second, TLSHandshakeTimeout: 10 * time.Second, ExpectContinueTimeout: 1 * time.Second, TLSClientConfig: &tls.Config{InsecureSkipVerify: true}, } httpClient := &http.Client{Transport: transport} return &Client{ url: strings.TrimSuffix(url, "/"), client: httpClient, pullRequestRefRegexes: pullRequestRefRegexes, } } // SetHTTPClient replaces default http.Client with user given one. func (c *Client) SetHTTPClient(client *http.Client) { c.client = client } func (c *Client) doRequest(method, path string, query url.Values, header http.Header, ibody io.Reader) (*http.Response, error) { u, err := url.Parse(c.url + "/" + path) if err != nil { return nil, err } u.RawQuery = query.Encode() req, err := http.NewRequest(method, u.String(), ibody) if err != nil { return nil, err } for k, v := range header { req.Header[k] = v } return c.client.Do(req) } func (c *Client) getResponse(method, path string, query url.Values, header http.Header, ibody io.Reader) (*http.Response, error) { resp, err := c.doRequest(method, path, query, header, ibody) if err != nil { return nil, err } if resp.StatusCode/100 != 2 { defer resp.Body.Close() data, err := ioutil.ReadAll(resp.Body) if err != nil { return nil, err } if len(data) <= 1 { return resp, errors.New(resp.Status) } // TODO(sgotti) use a json error response return resp, errors.New(string(data)) } return resp, nil } func (c *Client) GetUserInfo() (*gitsource.UserInfo, error) { return nil, nil } func (c *Client) GetRepoInfo(repopath string) (*gitsource.RepoInfo, error) { return nil, nil } func (c *Client) GetFile(repopath, commit, file string) ([]byte, error) { resp, err := c.getResponse("GET", fmt.Sprintf("%s.git/raw/%s/%s", repopath, commit, file), nil, nil, nil) if err != nil { return nil, err } defer resp.Body.Close() data, err := ioutil.ReadAll(resp.Body) return data, err } func (c *Client) CreateDeployKey(repopath, title, pubKey string, readonly bool) error { return nil } func (c *Client) DeleteDeployKey(repopath, title string) error { return nil } func (c *Client) UpdateDeployKey(repopath, title, pubKey string, readonly bool) error { return nil } func (c *Client) CreateRepoWebhook(repopath, url, secret string) error { return nil } func (c *Client) ParseWebhook(r *http.Request, secret string) (*types.WebhookData, error) { return nil, nil } func (c *Client) DeleteRepoWebhook(repopath, u string) error { return nil } func (c *Client) CreateCommitStatus(repopath, commitSHA string, status gitsource.CommitStatus, targetURL, description, context string) error { return nil } func (c *Client) ListUserRepos() ([]*gitsource.RepoInfo, error) { return nil, nil } func (c *Client) GetRef(repopath, ref string) (*gitsource.Ref, error) { return nil, nil } func (c *Client) RefType(ref string) (gitsource.RefType, string, error) { if strings.HasPrefix(ref, branchRefPrefix) { return gitsource.RefTypeBranch, strings.TrimPrefix(ref, branchRefPrefix), nil } if strings.HasPrefix(ref, tagRefPrefix) { return gitsource.RefTypeTag, strings.TrimPrefix(ref, tagRefPrefix), nil } for _, re := range c.pullRequestRefRegexes { if re.MatchString(ref) { m := re.FindStringSubmatch(ref) return gitsource.RefTypePullRequest, m[1], nil } } return -1, "", fmt.Errorf("unsupported ref: %s", ref) } func (c *Client) GetCommit(repopath, commitSHA string) (*gitsource.Commit, error) { return nil, nil } func (c *Client) BranchRef(branch string) string { return branchRefPrefix + branch } func (c *Client) TagRef(tag string) string { return tagRefPrefix + tag } func (c *Client) PullRequestRef(prID string) string { return "" } func (c *Client) CommitLink(repoInfo *gitsource.RepoInfo, commitSHA string) string { return "" } func (c *Client) BranchLink(repoInfo *gitsource.RepoInfo, branch string) string { return "" } func (c *Client) TagLink(repoInfo *gitsource.RepoInfo, tag string) string { return "" } func (c *Client) PullRequestLink(repoInfo *gitsource.RepoInfo, prID string) string { return "" }