37 Star 411 Fork 76

GVPrancher/rancher

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
client.go 14.95 KB
一键复制 编辑 原始数据 按行查看 历史
Dan Ramich 提交于 2019-06-12 13:38 . goimport linting changes
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552
package bitbucketserver
import (
"bytes"
"crypto/x509"
"encoding/json"
"encoding/pem"
"fmt"
"io"
"io/ioutil"
"mime/multipart"
"net/http"
"regexp"
"strconv"
"strings"
"github.com/mrjones/oauth"
"github.com/pkg/errors"
"github.com/rancher/norman/httperror"
"github.com/rancher/rancher/pkg/pipeline/remote/model"
"github.com/rancher/rancher/pkg/pipeline/utils"
"github.com/rancher/rancher/pkg/ref"
"github.com/rancher/rancher/pkg/settings"
v3 "github.com/rancher/types/apis/project.cattle.io/v3"
"github.com/sirupsen/logrus"
)
const (
maxPerPage = "100"
requestTokenURL = "%s/plugins/servlet/oauth/request-token"
authorizeTokenURL = "%s/plugins/servlet/oauth/authorize"
accessTokenURL = "%s/plugins/servlet/oauth/access-token"
)
type client struct {
BaseURL string
ConsumerKey string
PrivateKey string
RedirectURL string
}
func New(config *v3.BitbucketServerPipelineConfig) (model.Remote, error) {
if config == nil {
return nil, errors.New("empty bitbucket server config")
}
bsClient := &client{
ConsumerKey: config.ConsumerKey,
PrivateKey: config.PrivateKey,
RedirectURL: config.RedirectURL,
}
if config.TLS {
bsClient.BaseURL = "https://" + config.Hostname
} else {
bsClient.BaseURL = "http://" + config.Hostname
}
return bsClient, nil
}
func (c *client) Type() string {
return model.BitbucketServerType
}
func (c *client) getOauthConsumer() (*oauth.Consumer, error) {
keyBytes := []byte(c.PrivateKey)
block, _ := pem.Decode(keyBytes)
privateKey, err := x509.ParsePKCS1PrivateKey(block.Bytes)
if err != nil {
return nil, err
}
bitbucketOauthConsumer := oauth.NewRSAConsumer(
c.ConsumerKey,
privateKey,
oauth.ServiceProvider{
RequestTokenUrl: fmt.Sprintf(requestTokenURL, c.BaseURL),
AuthorizeTokenUrl: fmt.Sprintf(authorizeTokenURL, c.BaseURL),
AccessTokenUrl: fmt.Sprintf(accessTokenURL, c.BaseURL),
HttpMethod: http.MethodPost,
})
return bitbucketOauthConsumer, nil
}
func (c *client) Login(code string) (*v3.SourceCodeCredential, error) {
splits := strings.SplitN(code, ":", 3)
projectID := splits[0]
OAuthToken := splits[1]
OAuthVerifier := ""
if len(splits) == 3 {
OAuthVerifier = splits[2]
}
consumer, err := c.getOauthConsumer()
if err != nil {
return nil, err
}
requestToken := &oauth.RequestToken{
Token: OAuthToken,
}
token, err := consumer.AuthorizeToken(requestToken, OAuthVerifier)
if err != nil {
return nil, err
}
user, err := c.getUser(token.Token)
if err != nil {
return nil, err
}
cloneToken, err := c.getCloneToken(projectID, user.Name, token.Token)
if err != nil {
return nil, err
}
cred := convertUser(user)
cred.Spec.AccessToken = token.Token
cred.Spec.GitCloneToken = cloneToken
return cred, nil
}
func (c *client) Repos(account *v3.SourceCodeCredential) ([]v3.SourceCodeRepository, error) {
if account == nil {
return nil, fmt.Errorf("empty account")
}
url := c.BaseURL + "/rest/api/1.0/repos?permission=REPO_ADMIN"
hasNext := true
var repos []Repository
for hasNext {
b, err := c.getFromBitbucket(url, account.Spec.AccessToken)
if err != nil {
return nil, err
}
var pageRepos PaginatedRepositories
if err := json.Unmarshal(b, &pageRepos); err != nil {
return nil, err
}
hasNext = !pageRepos.IsLastPage
url = fmt.Sprintf("%s/rest/api/1.0/repos?permission=REPO_ADMIN&start=%d", c.BaseURL, pageRepos.NextPageStart)
repos = append(repos, pageRepos.Values...)
}
return convertRepos(repos), nil
}
func (c *client) CreateHook(pipeline *v3.Pipeline, accessToken string) (string, error) {
user, repo, err := getUserRepoFromURL(pipeline.Spec.RepositoryURL)
if err != nil {
return "", err
}
hookURL := fmt.Sprintf("%s/hooks?pipelineId=%s", settings.ServerURL.Get(), ref.Ref(pipeline))
hook := Hook{
Name: "pipeline webhook",
URL: hookURL,
Active: true,
Configuration: HookConfiguration{
Secret: pipeline.Status.Token,
},
Events: []string{
"repo:refs_changed",
"pr:opened",
"pr:modified",
},
}
url := fmt.Sprintf("%s/rest/api/1.0/projects/%s/repos/%s/webhooks", c.BaseURL, user, repo)
b, err := json.Marshal(hook)
if err != nil {
return "", err
}
reader := bytes.NewReader(b)
resp, err := c.doRequestToBitbucket(http.MethodPost, url, accessToken, nil, reader)
if err != nil {
return "", err
}
err = json.Unmarshal(resp, &hook)
if err != nil {
return "", err
}
return strconv.Itoa(hook.ID), nil
}
func (c *client) DeleteHook(pipeline *v3.Pipeline, accessToken string) error {
user, repo, err := getUserRepoFromURL(pipeline.Spec.RepositoryURL)
if err != nil {
return err
}
hook, err := c.getHook(pipeline, accessToken)
if err != nil {
return err
}
if hook != nil {
url := fmt.Sprintf("%s/rest/api/1.0/projects/%s/repos/%s/webhooks/%d", c.BaseURL, user, repo, hook.ID)
_, err := c.doRequestToBitbucket(http.MethodDelete, url, accessToken, nil, nil)
if err != nil {
return err
}
}
return nil
}
func (c *client) getHook(pipeline *v3.Pipeline, accessToken string) (*Hook, error) {
user, repo, err := getUserRepoFromURL(pipeline.Spec.RepositoryURL)
if err != nil {
return nil, err
}
var hooks PaginatedHooks
var result *Hook
url := fmt.Sprintf("%s/rest/api/1.0/projects/%s/repos/%s/webhooks", c.BaseURL, user, repo)
b, err := c.getFromBitbucket(url, accessToken)
if err != nil {
return nil, err
}
if err := json.Unmarshal(b, &hooks); err != nil {
return nil, err
}
for _, hook := range hooks.Values {
if strings.HasSuffix(hook.URL, fmt.Sprintf("hooks?pipelineId=%s", ref.Ref(pipeline))) {
result = &hook
break
}
}
return result, nil
}
func (c *client) getCloneToken(projectID string, user string, accessToken string) (string, error) {
url := fmt.Sprintf("%s/rest/access-tokens/1.0/users/%s", c.BaseURL, user)
if err := c.cleanCloneToken(projectID, user, accessToken); err != nil {
return "", err
}
token := AccessToken{
Name: "pipeline-" + projectID,
Permissions: []string{"REPO_READ", "PROJECT_READ"},
}
data, err := json.Marshal(token)
if err != nil {
return "", err
}
reader := bytes.NewReader(data)
b, err := c.doRequestToBitbucket(http.MethodPut, url, accessToken, nil, reader)
if err != nil {
return "", err
}
if err := json.Unmarshal(b, &token); err != nil {
return "", err
}
return token.Token, nil
}
func (c *client) cleanCloneToken(projectID string, user string, accessToken string) error {
url := fmt.Sprintf("%s/rest/access-tokens/1.0/users/%s", c.BaseURL, user)
tokens := &PaginatedAccessToken{}
name := "pipeline-" + projectID
b, err := c.getFromBitbucket(url, accessToken)
if err != nil {
return err
}
if err := json.Unmarshal(b, tokens); err != nil {
return err
}
for _, token := range tokens.Values {
if token.Name == name {
todeleteURL := fmt.Sprintf("%s/%s", url, token.ID)
if _, err = c.doRequestToBitbucket(http.MethodDelete, todeleteURL, accessToken, nil, nil); err != nil {
return err
}
break
}
}
return nil
}
func (c *client) getFileFromRepo(filename string, owner string, repo string, branch string, accessToken string) ([]byte, error) {
url := fmt.Sprintf("%s/rest/api/1.0/projects/%s/repos/%s/raw/%s?at=%s", c.BaseURL, owner, repo, filename, branch)
return c.getFromBitbucket(url, accessToken)
}
func (c *client) GetPipelineFileInRepo(repoURL string, branch string, accessToken string) ([]byte, error) {
owner, repo, err := getUserRepoFromURL(repoURL)
if err != nil {
return nil, err
}
content, err := c.getFileFromRepo(utils.PipelineFileYaml, owner, repo, branch, accessToken)
if err != nil {
//look for both suffix
content, err = c.getFileFromRepo(utils.PipelineFileYml, owner, repo, branch, accessToken)
}
if err != nil {
logrus.Debugf("error GetPipelineFileInRepo - %v", err)
return nil, nil
}
return content, nil
}
func (c *client) SetPipelineFileInRepo(repoURL string, branch string, accessToken string, content []byte) error {
owner, repo, err := getUserRepoFromURL(repoURL)
if err != nil {
return err
}
currentContent, err := c.getFileFromRepo(utils.PipelineFileYml, owner, repo, branch, accessToken)
currentFileName := utils.PipelineFileYml
if err != nil {
if httpErr, ok := err.(*httperror.APIError); !ok || httpErr.Code.Status != http.StatusNotFound {
return err
}
//look for both suffix
currentContent, err = c.getFileFromRepo(utils.PipelineFileYaml, owner, repo, branch, accessToken)
if err != nil {
if httpErr, ok := err.(*httperror.APIError); !ok || httpErr.Code.Status != http.StatusNotFound {
return err
}
} else {
currentFileName = utils.PipelineFileYaml
}
}
apiurl := fmt.Sprintf("%s/rest/api/1.0/projects/%s/repos/%s/browse/%s", c.BaseURL, owner, repo, currentFileName)
message := "Create .rancher-pipeline.yml file"
var b bytes.Buffer
w := multipart.NewWriter(&b)
w.FormDataContentType()
values := map[string]string{
"content": string(content),
"branch": branch,
"message": message,
}
if currentContent != nil {
//update pipeline file
message = fmt.Sprintf("Update %s file", currentFileName)
commitID, err := c.getFileLastCommit(owner, repo, branch, currentFileName, accessToken)
if err != nil {
return err
}
values["sourceCommitId"] = commitID
values["message"] = message
}
for key, v := range values {
var fw io.Writer
if fw, err = w.CreateFormField(key); err != nil {
return err
}
fw.Write([]byte(v))
}
if err := w.Close(); err != nil {
return err
}
header := map[string]string{"Content-Type": w.FormDataContentType()}
_, err = c.doRequestToBitbucket(http.MethodPut, apiurl, accessToken, header, &b)
return err
}
func (c *client) getFileLastCommit(owner string, repo string, branch string, fileName string, accessToken string) (string, error) {
url := fmt.Sprintf("%s/rest/api/1.0/projects/%s/repos/%s/last-modified?at=%s", c.BaseURL, owner, repo, branch)
b, err := c.getFromBitbucket(url, accessToken)
if err != nil {
return "", err
}
var lastModified LastModified
if err := json.Unmarshal(b, &lastModified); err != nil {
return "", err
}
for fname, commit := range lastModified.Files {
if fname == fileName {
return commit.ID, nil
}
}
return "", fmt.Errorf("fail to get last commit id of '%s' file", fileName)
}
func (c *client) GetBranches(repoURL string, accessToken string) ([]string, error) {
owner, repo, err := getUserRepoFromURL(repoURL)
if err != nil {
return nil, err
}
url := fmt.Sprintf("%s/rest/api/1.0/projects/%s/repos/%s/branches", c.BaseURL, owner, repo)
b, err := c.getFromBitbucket(url, accessToken)
if err != nil {
return nil, err
}
var branches PaginatedBranches
if err := json.Unmarshal(b, &branches); err != nil {
return nil, err
}
result := []string{}
for _, b := range branches.Values {
if b.Type != "BRANCH" {
continue
}
result = append(result, b.DisplayID)
}
return result, nil
}
func (c *client) GetHeadInfo(repoURL string, branch string, accessToken string) (*model.BuildInfo, error) {
owner, repo, err := getUserRepoFromURL(repoURL)
if err != nil {
return nil, err
}
url := fmt.Sprintf("%s/rest/api/1.0/projects/%s/repos/%s/branches", c.BaseURL, owner, repo)
b, err := c.getFromBitbucket(url, accessToken)
if err != nil {
return nil, err
}
var branches PaginatedBranches
if err := json.Unmarshal(b, &branches); err != nil {
return nil, err
}
headCommit := ""
for _, b := range branches.Values {
if b.DisplayID == branch {
headCommit = b.LatestCommit
break
}
}
if headCommit == "" {
return nil, fmt.Errorf("cannot find head commit of branch '%s'", branch)
}
url = fmt.Sprintf("%s/rest/api/1.0/projects/%s/repos/%s/commits/%s", c.BaseURL, owner, repo, headCommit)
b, err = c.getFromBitbucket(url, accessToken)
if err != nil {
return nil, err
}
var commit Commit
if err := json.Unmarshal(b, &commit); err != nil {
return nil, err
}
info := &model.BuildInfo{}
info.Commit = commit.ID
info.Ref = "refs/heads/" + branch
info.Branch = branch
info.Message = commit.Message
info.HTMLLink = fmt.Sprintf("%s/projects/%s/repos/%s/commits/%s", c.BaseURL, owner, repo, headCommit)
info.AvatarURL = fmt.Sprintf("%s/users/%s/avatar.png", c.BaseURL, commit.Author.Name)
info.Author = commit.Author.Name
return info, nil
}
func convertUser(bitbucketUser *User) *v3.SourceCodeCredential {
if bitbucketUser == nil {
return nil
}
cred := &v3.SourceCodeCredential{}
cred.Spec.SourceCodeType = model.BitbucketServerType
cred.Spec.AvatarURL = bitbucketUser.Links.Avatar.Href
if len(bitbucketUser.Links.Self) > 0 {
cred.Spec.HTMLURL = bitbucketUser.Links.Self[0].Href
cred.Spec.AvatarURL = cred.Spec.HTMLURL + "/avatar.png"
}
cred.Spec.LoginName = bitbucketUser.Name
cred.Spec.GitLoginName = bitbucketUser.Name
cred.Spec.DisplayName = bitbucketUser.DisplayName
return cred
}
func (c *client) getUser(accessToken string) (*User, error) {
url := c.BaseURL + "/plugins/servlet/applinks/whoami"
b, err := c.getFromBitbucket(url, accessToken)
if err != nil {
return nil, err
}
username := string(b)
url = fmt.Sprintf("%s/rest/api/1.0/users/%s", c.BaseURL, username)
b, err = c.getFromBitbucket(url, accessToken)
if err != nil {
return nil, err
}
user := &User{}
if err := json.Unmarshal(b, user); err != nil {
return nil, err
}
return user, nil
}
func convertRepos(repos []Repository) []v3.SourceCodeRepository {
result := []v3.SourceCodeRepository{}
for _, repo := range repos {
r := v3.SourceCodeRepository{}
for _, link := range repo.Links.Clone {
if strings.HasPrefix(link.Name, "http") {
r.Spec.URL = link.Href
break
}
}
r.Spec.Permissions.Admin = true
r.Spec.Permissions.Pull = true
r.Spec.Permissions.Push = true
result = append(result, r)
}
return result
}
func (c *client) getFromBitbucket(url string, accessToken string) ([]byte, error) {
return c.doRequestToBitbucket(http.MethodGet, url, accessToken, nil, nil)
}
func (c *client) doRequestToBitbucket(method string, url string, accessToken string, header map[string]string, body io.Reader) ([]byte, error) {
req, err := http.NewRequest(method, url, body)
if err != nil {
return nil, err
}
consumer, err := c.getOauthConsumer()
if err != nil {
return nil, err
}
var token oauth.AccessToken
token.Token = accessToken
client, err := consumer.MakeHttpClient(&token)
if err != nil {
return nil, err
}
q := req.URL.Query()
if method == http.MethodGet {
q.Set("limit", maxPerPage)
}
req.URL.RawQuery = q.Encode()
req.Header.Set("Content-Type", "application/json")
for k, v := range header {
req.Header.Set(k, v)
}
resp, err := client.Do(req)
if err != nil {
return nil, err
}
defer resp.Body.Close()
if resp.StatusCode < http.StatusOK || resp.StatusCode >= http.StatusBadRequest {
var body bytes.Buffer
io.Copy(&body, resp.Body)
return nil, httperror.NewAPIErrorLong(resp.StatusCode, "", body.String())
}
r, err := ioutil.ReadAll(resp.Body)
return r, err
}
func getUserRepoFromURL(repoURL string) (string, string, error) {
reg := regexp.MustCompile(".*/([^/]*?)/([^/]*?).git")
match := reg.FindStringSubmatch(repoURL)
if len(match) != 3 {
return "", "", fmt.Errorf("error getting user/repo from gitrepoUrl:%v", repoURL)
}
return match[1], match[2], nil
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
Go
1
https://gitee.com/rancher/rancher.git
git@gitee.com:rancher/rancher.git
rancher
rancher
rancher
v2.2.5-rc5

搜索帮助

0d507c66 1850385 C8b1a773 1850385