2014-06-04 21:25:38 +00:00
|
|
|
package gitlab
|
|
|
|
|
|
|
|
import (
|
2015-02-23 21:27:17 +00:00
|
|
|
"crypto/tls"
|
2015-01-23 18:51:37 +00:00
|
|
|
"fmt"
|
2014-09-02 07:18:17 +00:00
|
|
|
"io/ioutil"
|
2014-06-04 21:25:38 +00:00
|
|
|
"net/http"
|
2014-06-09 22:47:35 +00:00
|
|
|
"net/url"
|
2014-09-05 17:37:29 +00:00
|
|
|
"strconv"
|
2015-02-04 13:42:24 +00:00
|
|
|
"strings"
|
2015-01-26 23:32:42 +00:00
|
|
|
"time"
|
2014-06-04 21:25:38 +00:00
|
|
|
|
2015-01-26 23:32:42 +00:00
|
|
|
"code.google.com/p/goauth2/oauth"
|
2014-09-02 07:18:17 +00:00
|
|
|
"github.com/Bugagazavr/go-gitlab-client"
|
2015-01-23 18:51:37 +00:00
|
|
|
"github.com/drone/drone/shared/httputil"
|
2014-09-02 07:18:17 +00:00
|
|
|
"github.com/drone/drone/shared/model"
|
2014-06-04 21:25:38 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type Gitlab struct {
|
2014-11-26 22:21:52 +00:00
|
|
|
url string
|
|
|
|
SkipVerify bool
|
2015-01-12 22:59:06 +00:00
|
|
|
Open bool
|
2015-01-23 18:51:37 +00:00
|
|
|
Client string
|
|
|
|
Secret string
|
2014-06-04 21:25:38 +00:00
|
|
|
}
|
|
|
|
|
2015-01-23 18:51:37 +00:00
|
|
|
func New(url string, skipVerify, open bool, client, secret string) *Gitlab {
|
2014-11-26 22:21:52 +00:00
|
|
|
return &Gitlab{
|
|
|
|
url: url,
|
|
|
|
SkipVerify: skipVerify,
|
2015-01-12 22:59:06 +00:00
|
|
|
Open: open,
|
2015-01-23 18:51:37 +00:00
|
|
|
Client: client,
|
|
|
|
Secret: secret,
|
2014-11-26 22:21:52 +00:00
|
|
|
}
|
2014-06-04 21:25:38 +00:00
|
|
|
}
|
|
|
|
|
2014-09-02 07:18:17 +00:00
|
|
|
// Authorize handles authentication with thrid party remote systems,
|
|
|
|
// such as github or bitbucket, and returns user data.
|
|
|
|
func (r *Gitlab) Authorize(res http.ResponseWriter, req *http.Request) (*model.Login, error) {
|
2015-01-26 23:32:42 +00:00
|
|
|
host := httputil.GetURL(req)
|
|
|
|
config := NewOauthConfig(r, host)
|
2015-01-23 18:51:37 +00:00
|
|
|
|
|
|
|
var code = req.FormValue("code")
|
|
|
|
var state = req.FormValue("state")
|
|
|
|
|
|
|
|
if len(code) == 0 {
|
|
|
|
var random = GetRandom()
|
|
|
|
httputil.SetCookie(res, req, "gitlab_state", random)
|
|
|
|
http.Redirect(res, req, config.AuthCodeURL(random), http.StatusSeeOther)
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cookieState := httputil.GetCookie(req, "gitlab_state")
|
|
|
|
httputil.DelCookie(res, req, "gitlab_state")
|
|
|
|
if cookieState != state {
|
|
|
|
return nil, fmt.Errorf("Error matching state in OAuth2 redirect")
|
|
|
|
}
|
2014-09-02 07:18:17 +00:00
|
|
|
|
2015-01-23 18:51:37 +00:00
|
|
|
var trans = &oauth.Transport{Config: config}
|
|
|
|
var token, err = trans.Exchange(code)
|
2014-06-09 22:47:35 +00:00
|
|
|
if err != nil {
|
2015-01-23 18:51:37 +00:00
|
|
|
return nil, fmt.Errorf("Error exchanging token. %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var client = NewClient(r.url, token.AccessToken, r.SkipVerify)
|
|
|
|
|
|
|
|
var user, errr = client.CurrentUser()
|
|
|
|
if errr != nil {
|
|
|
|
return nil, fmt.Errorf("Error retrieving current user. %s", errr)
|
2014-06-09 22:47:35 +00:00
|
|
|
}
|
2014-09-02 07:18:17 +00:00
|
|
|
|
|
|
|
var login = new(model.Login)
|
2015-01-23 18:51:37 +00:00
|
|
|
login.ID = int64(user.Id)
|
|
|
|
login.Access = token.AccessToken
|
2015-01-26 23:32:42 +00:00
|
|
|
login.Secret = token.RefreshToken
|
2015-01-23 18:51:37 +00:00
|
|
|
login.Login = user.Username
|
|
|
|
login.Email = user.Email
|
2014-09-02 07:18:17 +00:00
|
|
|
return login, nil
|
2014-06-09 22:47:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-02 07:18:17 +00:00
|
|
|
// GetKind returns the identifier of this remote GitHub instane.
|
|
|
|
func (r *Gitlab) GetKind() string {
|
|
|
|
return model.RemoteGitlab
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetHost returns the hostname of this remote GitHub instance.
|
|
|
|
func (r *Gitlab) GetHost() string {
|
|
|
|
uri, _ := url.Parse(r.url)
|
|
|
|
return uri.Host
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetRepos fetches all repositories that the specified
|
|
|
|
// user has access to in the remote system.
|
|
|
|
func (r *Gitlab) GetRepos(user *model.User) ([]*model.Repo, error) {
|
|
|
|
|
|
|
|
var repos []*model.Repo
|
2014-11-26 22:21:52 +00:00
|
|
|
var client = NewClient(r.url, user.Access, r.SkipVerify)
|
2014-09-02 07:18:17 +00:00
|
|
|
var list, err = client.AllProjects()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var remote = r.GetKind()
|
|
|
|
var hostname = r.GetHost()
|
|
|
|
|
|
|
|
for _, item := range list {
|
|
|
|
var repo = model.Repo{
|
|
|
|
UserID: user.ID,
|
|
|
|
Remote: remote,
|
|
|
|
Host: hostname,
|
|
|
|
Owner: item.Namespace.Path,
|
|
|
|
Name: item.Path,
|
|
|
|
Private: !item.Public,
|
|
|
|
CloneURL: item.HttpRepoUrl,
|
|
|
|
GitURL: item.HttpRepoUrl,
|
|
|
|
SSHURL: item.SshRepoUrl,
|
2014-10-27 11:32:44 +00:00
|
|
|
URL: item.Url,
|
2014-09-02 07:18:17 +00:00
|
|
|
Role: &model.Perm{},
|
|
|
|
}
|
|
|
|
|
|
|
|
if repo.Private {
|
|
|
|
repo.CloneURL = repo.SSHURL
|
|
|
|
}
|
|
|
|
|
2014-09-03 08:23:45 +00:00
|
|
|
// if the user is the owner we can assume full access,
|
|
|
|
// otherwise check for the permission items.
|
|
|
|
if repo.Owner == user.Login {
|
|
|
|
repo.Role = new(model.Perm)
|
|
|
|
repo.Role.Admin = true
|
|
|
|
repo.Role.Write = true
|
|
|
|
repo.Role.Read = true
|
|
|
|
} else {
|
2014-09-06 19:07:47 +00:00
|
|
|
// Fetch current project
|
|
|
|
project, err := client.Project(strconv.Itoa(item.Id))
|
|
|
|
if err != nil || project.Permissions == nil {
|
|
|
|
continue
|
|
|
|
}
|
2014-09-05 17:37:29 +00:00
|
|
|
repo.Role.Admin = IsAdmin(project)
|
|
|
|
repo.Role.Write = IsWrite(project)
|
|
|
|
repo.Role.Read = IsRead(project)
|
2014-09-03 08:23:45 +00:00
|
|
|
}
|
|
|
|
|
2014-09-02 07:18:17 +00:00
|
|
|
repos = append(repos, &repo)
|
|
|
|
}
|
|
|
|
|
|
|
|
return repos, err
|
2014-06-04 21:25:38 +00:00
|
|
|
}
|
|
|
|
|
2014-09-02 07:18:17 +00:00
|
|
|
// GetScript fetches the build script (.drone.yml) from the remote
|
|
|
|
// repository and returns in string format.
|
|
|
|
func (r *Gitlab) GetScript(user *model.User, repo *model.Repo, hook *model.Hook) ([]byte, error) {
|
2014-11-26 22:21:52 +00:00
|
|
|
var client = NewClient(r.url, user.Access, r.SkipVerify)
|
2014-09-02 07:18:17 +00:00
|
|
|
var path = ns(repo.Owner, repo.Name)
|
|
|
|
return client.RepoRawFile(path, hook.Sha, ".drone.yml")
|
2014-06-04 21:25:38 +00:00
|
|
|
}
|
|
|
|
|
2014-09-02 07:18:17 +00:00
|
|
|
// Activate activates a repository by adding a Post-commit hook and
|
|
|
|
// a Public Deploy key, if applicable.
|
|
|
|
func (r *Gitlab) Activate(user *model.User, repo *model.Repo, link string) error {
|
2014-11-26 22:21:52 +00:00
|
|
|
var client = NewClient(r.url, user.Access, r.SkipVerify)
|
2014-09-02 07:18:17 +00:00
|
|
|
var path = ns(repo.Owner, repo.Name)
|
|
|
|
var title, err = GetKeyTitle(link)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// if the repository is private we'll need
|
|
|
|
// to upload a github key to the repository
|
|
|
|
if repo.Private {
|
|
|
|
var err = client.AddProjectDeployKey(path, title, repo.PublicKey)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// append the repo owner / name to the hook url since gitlab
|
|
|
|
// doesn't send this detail in the post-commit hook
|
2014-10-22 07:41:25 +00:00
|
|
|
link += "?owner=" + repo.Owner + "&name=" + repo.Name
|
2014-09-02 07:18:17 +00:00
|
|
|
|
|
|
|
// add the hook
|
|
|
|
return client.AddProjectHook(path, link, true, false, true)
|
2014-06-04 21:25:38 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 13:42:24 +00:00
|
|
|
// Deactivate removes a repository by removing all the post-commit hooks
|
|
|
|
// which are equal to link and removing the SSH deploy key.
|
|
|
|
func (r *Gitlab) Deactivate(user *model.User, repo *model.Repo, link string) error {
|
|
|
|
var client = NewClient(r.url, user.Access, r.SkipVerify)
|
|
|
|
var path = ns(repo.Owner, repo.Name)
|
|
|
|
|
|
|
|
keys, err := client.ProjectDeployKeys(path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
var pubkey = strings.TrimSpace(repo.PublicKey)
|
|
|
|
for _, k := range keys {
|
|
|
|
if pubkey == strings.TrimSpace(k.Key) {
|
|
|
|
if err := client.RemoveProjectDeployKey(path, strconv.Itoa(k.Id)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
hooks, err := client.ProjectHooks(path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
link += "?owner=" + repo.Owner + "&name=" + repo.Name
|
|
|
|
for _, h := range hooks {
|
|
|
|
if link == h.Url {
|
|
|
|
if err := client.RemoveProjectHook(path, strconv.Itoa(h.Id)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-09-02 07:18:17 +00:00
|
|
|
// ParseHook parses the post-commit hook from the Request body
|
|
|
|
// and returns the required data in a standard format.
|
|
|
|
func (r *Gitlab) ParseHook(req *http.Request) (*model.Hook, error) {
|
|
|
|
|
|
|
|
defer req.Body.Close()
|
|
|
|
var payload, _ = ioutil.ReadAll(req.Body)
|
|
|
|
var parsed, err = gogitlab.ParseHook(payload)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2014-09-06 20:08:56 +00:00
|
|
|
if len(parsed.After) == 0 || parsed.TotalCommitsCount == 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2014-09-02 07:18:17 +00:00
|
|
|
if parsed.ObjectKind == "merge_request" {
|
|
|
|
// TODO (bradrydzewski) figure out how to handle merge requests
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(parsed.After) == 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var hook = new(model.Hook)
|
|
|
|
hook.Owner = req.FormValue("owner")
|
|
|
|
hook.Repo = req.FormValue("name")
|
|
|
|
hook.Sha = parsed.After
|
|
|
|
hook.Branch = parsed.Branch()
|
|
|
|
|
|
|
|
var head = parsed.Head()
|
|
|
|
hook.Message = head.Message
|
|
|
|
hook.Timestamp = head.Timestamp
|
|
|
|
|
|
|
|
// extracts the commit author (ideally email)
|
|
|
|
// from the post-commit hook
|
|
|
|
switch {
|
|
|
|
case head.Author != nil:
|
|
|
|
hook.Author = head.Author.Email
|
|
|
|
case head.Author == nil:
|
|
|
|
hook.Author = parsed.UserName
|
|
|
|
}
|
|
|
|
|
|
|
|
return hook, nil
|
2014-06-04 21:25:38 +00:00
|
|
|
}
|
2015-01-12 22:59:06 +00:00
|
|
|
|
|
|
|
func (r *Gitlab) OpenRegistration() bool {
|
|
|
|
return r.Open
|
|
|
|
}
|
2015-01-26 23:32:42 +00:00
|
|
|
|
|
|
|
func (r *Gitlab) GetToken(user *model.User) (*model.Token, error) {
|
2015-01-27 22:42:20 +00:00
|
|
|
expiry := time.Unix(user.TokenExpiry, 0)
|
2015-04-29 00:05:16 +00:00
|
|
|
if user.TokenExpiry == 0 && len(user.Access) != 0 ||
|
|
|
|
expiry.Sub(time.Now()) > (60*time.Second) {
|
2015-01-27 22:42:20 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2015-01-26 23:32:42 +00:00
|
|
|
t := &oauth.Transport{
|
|
|
|
Config: NewOauthConfig(r, ""),
|
|
|
|
Token: &oauth.Token{
|
|
|
|
AccessToken: user.Access,
|
|
|
|
RefreshToken: user.Secret,
|
|
|
|
Expiry: expiry,
|
|
|
|
},
|
2015-02-23 21:27:17 +00:00
|
|
|
Transport: &http.Transport{
|
|
|
|
Proxy: http.ProxyFromEnvironment,
|
|
|
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: r.SkipVerify},
|
|
|
|
},
|
2015-01-26 23:32:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if err := t.Refresh(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var token = new(model.Token)
|
|
|
|
token.AccessToken = t.Token.AccessToken
|
|
|
|
token.RefreshToken = t.Token.RefreshToken
|
2015-01-27 22:42:20 +00:00
|
|
|
token.Expiry = t.Token.Expiry.Unix()
|
2015-01-26 23:32:42 +00:00
|
|
|
return token, nil
|
|
|
|
}
|