mirror of
https://github.com/woodpecker-ci/woodpecker.git
synced 2025-01-24 08:08:38 +00:00
refactored how remotes work and how (some) config is loaded
This commit is contained in:
parent
38379992bf
commit
ca3d15bca2
8 changed files with 173 additions and 118 deletions
88
README.md
88
README.md
|
@ -18,40 +18,70 @@ We have optimized the installation process for Ubuntu since that is what we test
|
||||||
wget downloads.drone.io/exp/drone.deb
|
wget downloads.drone.io/exp/drone.deb
|
||||||
sudo dpkg -i drone.deb
|
sudo dpkg -i drone.deb
|
||||||
```
|
```
|
||||||
https://github.com/drone/drone/milestones/v0.3
|
|
||||||
|
|
||||||
## What Changed
|
## Setup
|
||||||
|
|
||||||
This branch introduces major changes, including:
|
We are in the process of moving configuration out of the UI and into configuration
|
||||||
|
files and/or environment variables (your choice which). If you prefer configuration files
|
||||||
|
you can provide Drone with the path to your configuration file:
|
||||||
|
|
||||||
1. modification to project structure
|
```sh
|
||||||
2. api-driven design
|
./drone --config=/path/to/drone.conf
|
||||||
3. interface to abstract github, bitbucket, gitlab code (see /shared/remote)
|
|
||||||
4. handlers are structures with service providers "injected"
|
|
||||||
5. github, bitbucket, etc native permissions are used. No more teams or permissions in Drone
|
|
||||||
6. github, bitbucket, etc authentication is used. No more drone password
|
|
||||||
7. github, bitbucket, etc repository data is cached upon login (and subsequent logins)
|
|
||||||
8. angularjs user interface with modified responsive design
|
|
||||||
|
|
||||||
## Database Compatibility Issues
|
```
|
||||||
|
|
||||||
|
The configuration file is in TOML format:
|
||||||
|
|
||||||
|
```toml
|
||||||
|
[github]
|
||||||
|
client=""
|
||||||
|
secret=""
|
||||||
|
|
||||||
|
[github_enterprise]
|
||||||
|
client=""
|
||||||
|
secret=""
|
||||||
|
api=""
|
||||||
|
url=""
|
||||||
|
|
||||||
|
[bitbucket]
|
||||||
|
client=""
|
||||||
|
secret=""
|
||||||
|
|
||||||
|
[smtp]
|
||||||
|
host=""
|
||||||
|
port=""
|
||||||
|
from=""
|
||||||
|
user=""
|
||||||
|
pass=""
|
||||||
|
```
|
||||||
|
|
||||||
|
Or you can use environment variables
|
||||||
|
|
||||||
|
```sh
|
||||||
|
# github configuration
|
||||||
|
export DRONE_GITHUB_CLIENT=""
|
||||||
|
export DRONE_GITHUB_secret=""
|
||||||
|
|
||||||
|
# github enterprise configuration
|
||||||
|
export DRONE_GITHUB_ENTERPRISE_CLIENT=""
|
||||||
|
export DRONE_GITHUB_ENTERPRISE_SECRET=""
|
||||||
|
export DRONE_GITHUB_ENTERPRISE_API=""
|
||||||
|
export DRONE_GITHUB_ENTERPRISE_URL=""
|
||||||
|
|
||||||
|
# bitbucket configuration
|
||||||
|
export DRONE_BITBUCKET_CLIENT=""
|
||||||
|
export DRONE_BITBUCKET_SECRET=""
|
||||||
|
|
||||||
|
# email configuration
|
||||||
|
export DRONE_SMTP_HOST=""
|
||||||
|
export DRONE_SMTP_PORT=""
|
||||||
|
export DRONE_SMTP_FROM=""
|
||||||
|
export DRONE_SMTP_USER=""
|
||||||
|
export DRONE_SMTP_PASS=""
|
||||||
|
```
|
||||||
|
|
||||||
|
## Compatibility Issues
|
||||||
|
|
||||||
**WARNING**
|
**WARNING**
|
||||||
|
|
||||||
There were some fundamental changes to the application and we decided to introduce breaking changes to the dataabase. Migration would have been difficult and time consuming. Drone is an alpha product and therefore backward compatibility is not a primary goal until we hit a stable release. Apologizes for any inconvenience.
|
There were some fundamental changes to the application and we decided to introduce breaking changes to the dataabase. Migration would have been difficult and time consuming. Drone is an alpha product and therefore backward compatibility is not a primary goal until we hit a stable release. Apologizes for any inconvenience.
|
||||||
|
|
||||||
## Filing Bugs
|
|
||||||
|
|
||||||
This is an experimental branch with known bugs and issues, namely:
|
|
||||||
|
|
||||||
* notifications
|
|
||||||
* github status api updates
|
|
||||||
* gitlab integration
|
|
||||||
* smtp settings screen
|
|
||||||
* github / bitbucket / gitlab settings screen
|
|
||||||
* mysql support
|
|
||||||
|
|
||||||
Please do not log issues for the above items. We are aware and will fix as soon as possible, as they are holding up the 0.3 release. Pull requests, however, are very welcome :)
|
|
||||||
|
|
||||||
You can track progress of this release here:
|
|
||||||
|
|
||||||
https://github.com/drone/drone/milestones/v0.3
|
|
||||||
|
|
|
@ -5,9 +5,9 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"net"
|
"net"
|
||||||
"net/smtp"
|
"net/smtp"
|
||||||
"os"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/drone/config"
|
||||||
"github.com/drone/drone/shared/model"
|
"github.com/drone/drone/shared/model"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -28,11 +28,11 @@ const (
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
DefaultHost = os.Getenv("SMTP_HOST")
|
DefaultHost = config.String("smtp-host", "")
|
||||||
DefaultPort = os.Getenv("SMTP_PORT")
|
DefaultPort = config.String("smtp-port", "")
|
||||||
DefaultFrom = os.Getenv("SMTP_FROM")
|
DefaultFrom = config.String("smtp-from", "")
|
||||||
DefaultUser = os.Getenv("SMTP_USER")
|
DefaultUser = config.String("smtp-user", "")
|
||||||
DefaultPass = os.Getenv("SMTP_PASS")
|
DefaultPass = config.String("smtp-pass", "")
|
||||||
)
|
)
|
||||||
|
|
||||||
type Email struct {
|
type Email struct {
|
||||||
|
@ -139,12 +139,12 @@ func (e *Email) send(subject, body string, recipients []string) error {
|
||||||
// configuration. If None provided, attempt to
|
// configuration. If None provided, attempt to
|
||||||
// use the global configuration set in the environet
|
// use the global configuration set in the environet
|
||||||
// variables.
|
// variables.
|
||||||
if len(DefaultHost) != 0 {
|
if len(*DefaultHost) != 0 {
|
||||||
e.Host = DefaultHost
|
e.Host = *DefaultHost
|
||||||
e.Port = DefaultPort
|
e.Port = *DefaultPort
|
||||||
e.From = DefaultFrom
|
e.From = *DefaultFrom
|
||||||
e.Username = DefaultUser
|
e.Username = *DefaultUser
|
||||||
e.Password = DefaultPass
|
e.Password = *DefaultPass
|
||||||
}
|
}
|
||||||
|
|
||||||
var auth smtp.Auth
|
var auth smtp.Auth
|
||||||
|
|
|
@ -3,21 +3,12 @@ package github
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"code.google.com/p/goauth2/oauth"
|
"code.google.com/p/goauth2/oauth"
|
||||||
"github.com/drone/drone/shared/model"
|
"github.com/drone/drone/shared/model"
|
||||||
"github.com/google/go-github/github"
|
"github.com/google/go-github/github"
|
||||||
)
|
)
|
||||||
|
|
||||||
// TODO (bradrydzewski) explore using the Repo.URL to parse the GitHub
|
|
||||||
// Entperprise Scheme+Hostname, instead of the environment variable. Is
|
|
||||||
// there any reason not to use the environment variable?
|
|
||||||
|
|
||||||
// GitHub enterprise URL
|
|
||||||
var URL = os.Getenv("GITHUB_ENTERPRISE_API")
|
|
||||||
|
|
||||||
const (
|
const (
|
||||||
NotifyDisabled = "disabled"
|
NotifyDisabled = "disabled"
|
||||||
NotifyFalse = "false"
|
NotifyFalse = "false"
|
||||||
|
@ -38,6 +29,10 @@ const (
|
||||||
DescError = "oops, something went wrong"
|
DescError = "oops, something went wrong"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
BaseURL = "https://api.github.com/"
|
||||||
|
)
|
||||||
|
|
||||||
type GitHub string
|
type GitHub string
|
||||||
|
|
||||||
// Send uses the github status API to update the build
|
// Send uses the github status API to update the build
|
||||||
|
@ -70,6 +65,7 @@ func (g GitHub) Send(context *model.Request) error {
|
||||||
)
|
)
|
||||||
|
|
||||||
return send(
|
return send(
|
||||||
|
context.Repo.URL,
|
||||||
context.Repo.Host,
|
context.Repo.Host,
|
||||||
context.Repo.Owner,
|
context.Repo.Owner,
|
||||||
context.Repo.Name,
|
context.Repo.Name,
|
||||||
|
@ -81,7 +77,7 @@ func (g GitHub) Send(context *model.Request) error {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
func send(host, owner, repo, status, desc, target, ref, token string) error {
|
func send(rawurl, host, owner, repo, status, desc, target, ref, token string) error {
|
||||||
transport := &oauth.Transport{
|
transport := &oauth.Transport{
|
||||||
Token: &oauth.Token{AccessToken: token},
|
Token: &oauth.Token{AccessToken: token},
|
||||||
}
|
}
|
||||||
|
@ -99,10 +95,7 @@ func send(host, owner, repo, status, desc, target, ref, token string) error {
|
||||||
// the base url. Per the documentation, we need to
|
// the base url. Per the documentation, we need to
|
||||||
// ensure there is a trailing slash.
|
// ensure there is a trailing slash.
|
||||||
if host != model.RemoteGithub {
|
if host != model.RemoteGithub {
|
||||||
client.BaseURL, _ = url.Parse(URL)
|
client.BaseURL, _ = getEndpoint(rawurl)
|
||||||
if !strings.HasSuffix(client.BaseURL.Path, "/") {
|
|
||||||
client.BaseURL.Path = client.BaseURL.Path + "/"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_, _, err := client.Repositories.CreateStatus(owner, repo, ref, &data)
|
_, _, err := client.Repositories.CreateStatus(owner, repo, ref, &data)
|
||||||
|
@ -151,3 +144,15 @@ func getDesc(status string) string {
|
||||||
func getTarget(url, host, owner, repo, branch, commit string) string {
|
func getTarget(url, host, owner, repo, branch, commit string) string {
|
||||||
return fmt.Sprintf("%s/%s/%s/%s/%s/%s", url, host, owner, repo, branch, commit)
|
return fmt.Sprintf("%s/%s/%s/%s/%s/%s", url, host, owner, repo, branch, commit)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// getEndpoint is a helper funcation that parsed the
|
||||||
|
// repository HTML URL to determine the API URL. It is
|
||||||
|
// intended for use with GitHub enterprise.
|
||||||
|
func getEndpoint(rawurl string) (*url.URL, error) {
|
||||||
|
uri, err := url.Parse(rawurl)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
uri.Path = "/api/v3"
|
||||||
|
return uri, nil
|
||||||
|
}
|
||||||
|
|
|
@ -1,16 +1,24 @@
|
||||||
package bitbucket
|
package bitbucket
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"os"
|
"github.com/drone/config"
|
||||||
|
|
||||||
"github.com/drone/drone/plugin/remote"
|
"github.com/drone/drone/plugin/remote"
|
||||||
)
|
)
|
||||||
|
|
||||||
func init() {
|
var (
|
||||||
var cli = os.Getenv("BITBUCKET_CLIENT")
|
// Bitbucket cloud configuration details
|
||||||
var sec = os.Getenv("BITBUCKET_SECRET")
|
bitbucketClient = config.String("bitbucket-client", "")
|
||||||
if len(cli) == 0 || len(sec) == 0 {
|
bitbucketSecret = config.String("bitbucket-secret", "")
|
||||||
|
)
|
||||||
|
|
||||||
|
// Registers the Bitbucket plugin using the default
|
||||||
|
// settings from the config file or environment
|
||||||
|
// variables.
|
||||||
|
func Register() {
|
||||||
|
if len(*bitbucketClient) == 0 || len(*bitbucketSecret) == 0 {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
remote.Register(NewDefault(cli, sec))
|
remote.Register(
|
||||||
|
NewDefault(*bitbucketClient, *bitbucketSecret),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -140,7 +140,7 @@ func (r *GitHub) GetRepos(user *model.User) ([]*model.Repo, error) {
|
||||||
Owner: *item.Owner.Login,
|
Owner: *item.Owner.Login,
|
||||||
Name: *item.Name,
|
Name: *item.Name,
|
||||||
Private: *item.Private,
|
Private: *item.Private,
|
||||||
URL: *item.URL,
|
URL: *item.HTMLURL,
|
||||||
CloneURL: *item.GitURL,
|
CloneURL: *item.GitURL,
|
||||||
GitURL: *item.GitURL,
|
GitURL: *item.GitURL,
|
||||||
SSHURL: *item.SSHURL,
|
SSHURL: *item.SSHURL,
|
||||||
|
|
|
@ -1,41 +1,54 @@
|
||||||
package github
|
package github
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"os"
|
"github.com/drone/config"
|
||||||
|
|
||||||
"github.com/drone/drone/plugin/remote"
|
"github.com/drone/drone/plugin/remote"
|
||||||
)
|
)
|
||||||
|
|
||||||
func init() {
|
var (
|
||||||
init_github()
|
// GitHub cloud configuration details
|
||||||
init_github_enterprise()
|
githubClient = config.String("github-client", "")
|
||||||
|
githubSecret = config.String("github-secret", "")
|
||||||
|
|
||||||
|
// GitHub Enterprise configuration details
|
||||||
|
githubEnterpriseURL = config.String("github-enterprise-url", "")
|
||||||
|
githubEnterpriseAPI = config.String("github-enterprise-api", "")
|
||||||
|
githubEnterpriseClient = config.String("github-enterprise-client", "")
|
||||||
|
githubEnterpriseSecret = config.String("github-enterprise-secret", "")
|
||||||
|
)
|
||||||
|
|
||||||
|
// Registers the GitHub plugins using the default
|
||||||
|
// settings from the config file or environment
|
||||||
|
// variables.
|
||||||
|
func Register() {
|
||||||
|
registerGitHub()
|
||||||
|
registerGitHubEnterprise()
|
||||||
}
|
}
|
||||||
|
|
||||||
// registers the GitHub (github.com) plugin
|
// registers the GitHub (github.com) plugin
|
||||||
func init_github() {
|
func registerGitHub() {
|
||||||
var cli = os.Getenv("GITHUB_CLIENT")
|
if len(*githubClient) == 0 || len(*githubSecret) == 0 {
|
||||||
var sec = os.Getenv("GITHUB_SECRET")
|
|
||||||
if len(cli) == 0 ||
|
|
||||||
len(sec) == 0 {
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
var github = NewDefault(cli, sec)
|
remote.Register(
|
||||||
remote.Register(github)
|
NewDefault(*githubClient, *githubSecret),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// registers the GitHub Enterprise plugin
|
// registers the GitHub Enterprise plugin
|
||||||
func init_github_enterprise() {
|
func registerGitHubEnterprise() {
|
||||||
var url = os.Getenv("GITHUB_ENTERPRISE_URL")
|
if len(*githubEnterpriseURL) == 0 ||
|
||||||
var api = os.Getenv("GITHUB_ENTERPRISE_API")
|
len(*githubEnterpriseAPI) == 0 ||
|
||||||
var cli = os.Getenv("GITHUB_ENTERPRISE_CLIENT")
|
len(*githubEnterpriseClient) == 0 ||
|
||||||
var sec = os.Getenv("GITHUB_ENTERPRISE_SECRET")
|
len(*githubEnterpriseSecret) == 0 {
|
||||||
|
|
||||||
if len(url) == 0 ||
|
|
||||||
len(api) == 0 ||
|
|
||||||
len(cli) == 0 ||
|
|
||||||
len(sec) == 0 {
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
var github = New(url, api, cli, sec)
|
remote.Register(
|
||||||
remote.Register(github)
|
New(
|
||||||
|
*githubEnterpriseURL,
|
||||||
|
*githubEnterpriseAPI,
|
||||||
|
*githubEnterpriseClient,
|
||||||
|
*githubEnterpriseSecret,
|
||||||
|
),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,16 +1,22 @@
|
||||||
package gitlab
|
package gitlab
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"os"
|
"github.com/drone/config"
|
||||||
|
|
||||||
"github.com/drone/drone/plugin/remote"
|
"github.com/drone/drone/plugin/remote"
|
||||||
)
|
)
|
||||||
|
|
||||||
// registers the Gitlab plugin
|
var (
|
||||||
func init() {
|
gitlabURL = config.String("gitlab-url", "")
|
||||||
var url = os.Getenv("GITLAB_URL")
|
)
|
||||||
if len(url) == 0 {
|
|
||||||
|
// Registers the Gitlab plugin using the default
|
||||||
|
// settings from the config file or environment
|
||||||
|
// variables.
|
||||||
|
func Register() {
|
||||||
|
if len(*gitlabURL) == 0 {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
remote.Register(New(url))
|
remote.Register(
|
||||||
|
New(*gitlabURL),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,7 @@ import (
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/drone/config"
|
||||||
"github.com/drone/drone/server/database"
|
"github.com/drone/drone/server/database"
|
||||||
"github.com/drone/drone/server/database/schema"
|
"github.com/drone/drone/server/database/schema"
|
||||||
"github.com/drone/drone/server/handler"
|
"github.com/drone/drone/server/handler"
|
||||||
|
@ -22,9 +23,9 @@ import (
|
||||||
_ "github.com/mattn/go-sqlite3"
|
_ "github.com/mattn/go-sqlite3"
|
||||||
"github.com/russross/meddler"
|
"github.com/russross/meddler"
|
||||||
|
|
||||||
_ "github.com/drone/drone/plugin/remote/bitbucket"
|
"github.com/drone/drone/plugin/remote/bitbucket"
|
||||||
_ "github.com/drone/drone/plugin/remote/github"
|
"github.com/drone/drone/plugin/remote/github"
|
||||||
_ "github.com/drone/drone/plugin/remote/gitlab"
|
"github.com/drone/drone/plugin/remote/gitlab"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -49,12 +50,16 @@ var (
|
||||||
// Number of concurrent build workers to run
|
// Number of concurrent build workers to run
|
||||||
// default to number of CPUs on machine
|
// default to number of CPUs on machine
|
||||||
workers int
|
workers int
|
||||||
|
|
||||||
|
conf string
|
||||||
|
prefix string
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
|
|
||||||
log.SetPriority(log.LOG_NOTICE)
|
log.SetPriority(log.LOG_NOTICE)
|
||||||
|
|
||||||
|
flag.StringVar(&conf, "config", "", "")
|
||||||
|
flag.StringVar(&prefix, "prefix", "DRONE_", "")
|
||||||
flag.StringVar(&port, "port", ":8080", "")
|
flag.StringVar(&port, "port", ":8080", "")
|
||||||
flag.StringVar(&driver, "driver", "sqlite3", "")
|
flag.StringVar(&driver, "driver", "sqlite3", "")
|
||||||
flag.StringVar(&datasource, "datasource", "drone.sqlite", "")
|
flag.StringVar(&datasource, "datasource", "drone.sqlite", "")
|
||||||
|
@ -63,6 +68,14 @@ func main() {
|
||||||
flag.IntVar(&workers, "workers", runtime.NumCPU(), "")
|
flag.IntVar(&workers, "workers", runtime.NumCPU(), "")
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
|
|
||||||
|
config.SetPrefix(prefix)
|
||||||
|
config.Parse(conf)
|
||||||
|
|
||||||
|
// setup the remote services
|
||||||
|
bitbucket.Register()
|
||||||
|
github.Register()
|
||||||
|
gitlab.Register()
|
||||||
|
|
||||||
// setup the database
|
// setup the database
|
||||||
meddler.Default = meddler.SQLite
|
meddler.Default = meddler.SQLite
|
||||||
db, _ := sql.Open(driver, datasource)
|
db, _ := sql.Open(driver, datasource)
|
||||||
|
@ -147,23 +160,3 @@ func main() {
|
||||||
panic(http.ListenAndServe(port, nil))
|
panic(http.ListenAndServe(port, nil))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func init() {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func init_flags() {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func init_database() {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func init_workers() {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func init_handlers() {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
Loading…
Reference in a new issue