diff --git a/.drone.sh b/.drone.sh index f82021cdd..c38654db6 100755 --- a/.drone.sh +++ b/.drone.sh @@ -1,27 +1,8 @@ #!/bin/sh -# only execute this script as part of the pipeline. -# [ -z "$CI" ] && echo "missing ci environment variable" && exit 2 - -# only execute the script when github token exists. -# [ -z "$SSH_KEY" ] && echo "missing ssh key" && exit 3 - -# write the ssh key. -# mkdir /root/.ssh -# echo -n "$SSH_KEY" > /root/.ssh/id_rsa -# chmod 600 /root/.ssh/id_rsa - -# add github.com to our known hosts. -# touch /root/.ssh/known_hosts -# chmod 600 /root/.ssh/known_hosts -# ssh-keyscan -H github.com > /etc/ssh/ssh_known_hosts 2> /dev/null - -# clone the extras project. set -e set -x -# git clone git@github.com:drone/drone-enterprise.git extras -# build a static binary with the build number and extra features. go build -ldflags '-extldflags "-static" -X github.com/laszlocph/drone-oss-08/version.VersionDev=build.'${DRONE_BUILD_NUMBER} -o release/drone-server github.com/laszlocph/drone-oss-08/cmd/drone-server GOOS=linux GOARCH=amd64 CGO_ENABLED=0 go build -ldflags '-X github.com/laszlocph/drone-oss-08/version.VersionDev=build.'${DRONE_BUILD_NUMBER} -o release/drone-agent github.com/laszlocph/drone-oss-08/cmd/drone-agent GOOS=linux GOARCH=arm64 CGO_ENABLED=0 go build -ldflags '-X github.com/laszlocph/drone-oss-08/version.VersionDev=build.'${DRONE_BUILD_NUMBER} -o release/linux/arm64/drone-agent github.com/laszlocph/drone-oss-08/cmd/drone-agent diff --git a/.drone.yml b/.drone.yml index ebda487a3..bc5005279 100644 --- a/.drone.yml +++ b/.drone.yml @@ -1,3 +1,7 @@ +clone: + git: + image: plugins/git:next + workspace: base: /go path: src/github.com/laszlocph/drone-oss-08 @@ -109,7 +113,7 @@ pipeline: repo: laszlocloud/drone-oss-08-server dockerfile: Dockerfile.alpine secrets: [ docker_username, docker_password ] - tag: [ 0.8.95-bitbucket-alpine ] + tag: "${DRONE_TAG}-alpine" when: event: tag @@ -118,7 +122,7 @@ pipeline: repo: laszlocloud/drone-oss-08-agent dockerfile: Dockerfile.agent.alpine secrets: [ docker_username, docker_password ] - tag: [ 0.8.95-bitbucket-alpine ] + tag: "${DRONE_TAG}-alpine" when: event: tag @@ -126,7 +130,7 @@ pipeline: image: plugins/docker repo: laszlocloud/drone-oss-08-server secrets: [ docker_username, docker_password ] - tag: [ 0.8.95-bitbucket ] + tag: ${DRONE_TAG} when: event: tag @@ -135,7 +139,7 @@ pipeline: repo: laszlocloud/drone-oss-08-agent dockerfile: Dockerfile.agent secrets: [ docker_username, docker_password ] - tag: [ 0.8.95-bitbucket ] + tag: ${DRONE_TAG} when: event: tag diff --git a/.gitignore b/.gitignore index 22527af4a..e2cebad11 100644 --- a/.gitignore +++ b/.gitignore @@ -8,4 +8,5 @@ release/ cli/release/ server/swagger/files/*.json +server/swagger/swagger_gen.go .idea/ diff --git a/BUILDING b/BUILDING index d7ef8996e..3e66d99f7 100644 --- a/BUILDING +++ b/BUILDING @@ -10,3 +10,20 @@ go install github.com/laszlocph/drone-oss-08/cmd/drone-agent go install github.com/laszlocph/drone-oss-08/cmd/drone-server + +--- + +0. To generate SQL files + +go get github.com/vektra/mockery/.../ + +export download_url=$(curl -s https://api.github.com/repos/go-swagger/go-swagger/releases/latest | \ + jq -r '.assets[] | select(.name | contains("'"$(uname | tr '[:upper:]' '[:lower:]')"'_amd64")) | .browser_download_url') +curl -o swagger -L'#' "$download_url" +chmod +x swagger +sudo mv swagger /usr/local/bin + +go get github.com/laszlocph/togo + +go generate + diff --git a/LICENSE b/LICENSE index 04b10d8b6..fed1248b2 100644 --- a/LICENSE +++ b/LICENSE @@ -1,80 +1,3 @@ -The Drone Community Edition (the "Community Edition") is licensed under the -Apache License, Version 2.0 (the "Apache License"). You may obtain a copy of -the Apache License at +Drone-OSS-08 is Apache 2.0 licensed with the source files in this repository having a header indicating which license they are under and what copyrights apply. - http://www.apache.org/licenses/LICENSE-2.0 - -The Drone Enterprise Edition (the "Enterprise Edition") is licensed under -the Drone Enterprise License, Version 1.1 (the "Enterprise License"). A copy -of the Enterprise License is provided below. - -The source files in this repository have a header indicating which license -they are under. The BUILDING file provides instructions for creating the -Community Edition distribution subject to the terms of the Apache License. - ------------------------------------------------------------------------------ - -Licensor: Drone.IO, Inc -Licensed Work: Drone Enterprise Edition - -Additional Use Grant: Usage of the software is free for entities with both: - (a) annual gross revenue under (USD) $1 million - (according to GAAP, or the equivalent in its country - of domicile); and (b) less than (USD) $5 million in - aggregate debt and equity funding. - -Change Date: 2022-01-01 - -Change License: Apache-2.0 - -Notice - -The Drone Enterprise License (this document, or the "License") is not an Open -Source license. However, the Licensed Work will eventually be made available -under an Open Source License, as stated in this License. - ------------------------------------------------------------------------------ - -Drone Enterprise License 1.1 - -Terms - -The Licensor hereby grants you the right to copy, modify, create derivative -works, redistribute, and make non-production use of the Licensed Work. The -Licensor may make an Additional Use Grant, above, permitting limited -production use. - -Effective on the Change Date, or the fourth anniversary of the first publicly -available distribution of a specific version of the Licensed Work under this -License, whichever comes first, the Licensor hereby grants you rights under -the terms of the Change License, and the rights granted in the paragraph -above terminate. - -If your use of the Licensed Work does not comply with the requirements -currently in effect as described in this License, you must purchase a -commercial license from the Licensor, its affiliated entities, or authorized -resellers, or you must refrain from using the Licensed Work. - -All copies of the original and modified Licensed Work, and derivative works -of the Licensed Work, are subject to this License. This License applies -separately for each version of the Licensed Work and the Change Date may vary -for each version of the Licensed Work released by Licensor. - -You must conspicuously display this License on each original or modified copy -of the Licensed Work. If you receive the Licensed Work in original or -modified form from a third party, the terms and conditions set forth in this -License apply to your use of that work. - -Any use of the Licensed Work in violation of this License will automatically -terminate your rights under this License for the current and all other -versions of the Licensed Work. - -This License does not grant you any right in any trademark or logo of -Licensor or its affiliates (provided that you may use a trademark or logo of -Licensor as expressly required by this License). - -TO THE EXTENT PERMITTED BY APPLICABLE LAW, THE LICENSED WORK IS PROVIDED ON -AN "AS IS" BASIS. LICENSOR HEREBY DISCLAIMS ALL WARRANTIES AND CONDITIONS, -EXPRESS OR IMPLIED, INCLUDING (WITHOUT LIMITATION) WARRANTIES OF -MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, NON-INFRINGEMENT, AND -TITLE. +Files under the `docs/` folder is licensed under Creative Commons Attribution-ShareAlike 4.0 International Public License. It is a derivative work of the https://github.com/drone/docs git repository. diff --git a/README.md b/README.md index 3581240c0..281e707b7 100644 --- a/README.md +++ b/README.md @@ -1,31 +1,326 @@ -## Yes, it's a fork +# Drone-OSS-08 -This repository is a hard fork of the Drone CI system. +An opinionated fork of the Drone CI system. -Forked at the `0.8.9` version https://github.com/drone/drone/commit/768ed784bd74b0e0c2d8d49c4c8b6dca99b25e96 +- Based on the v0.8 code tree +- Focused on developer experience. -## Why fork? +[![Build Status](https://cloud.drone.io/api/badges/laszlocph/drone-oss-08/status.svg)](https://cloud.drone.io/laszlocph/drone-oss-08) [![Go Report Card](https://goreportcard.com/badge/github.com/laszlocph/drone-oss-08)](https://goreportcard.com/report/github.com/laszlocph/drone-oss-08) [![License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](https://opensource.org/licenses/Apache-2.0) -Drone has been an open-core project since many prior versions. With each source file indicating whether it is part of the Apache 2.0 licensed or the propritary enterprise license. In the 0.8 line the enterprise features were limited to features like autoscaling and secret vaults. +![Drone-OSS-08](docs/drone.png) -However in the 1.0 line, databases other than SQLite, TLS support and agent based horizontal scaling were also moved under the enterprise license. Limiting the open source version to single node, hobbyist deployments. +## Table of contents -The above feature reductions and the lack of clear communication of what is part of the open-source version led to this fork. +- [About this fork](#about-this-fork) + - [Motivation](#motivation) + - [The focus of this fork](#the-focus-of-this-fork) + - [Who uses this fork](#who-uses-this-fork) +- [Pipelines](#pipelines) + - [Getting started](#getting-started) + - [Pipeline documentation](#pipeline-documentation) +- [Plugins](#plugins) + - [Custom plugins](#custom-plugins) +- [Server setup](#server-setup) + - [Quickstart](#quickstart) + - [Authentication](#authentication) + - [Database](#database) + - [SSL](#ssl) + - [Metrics](#metrics) + - [Behind a proxy](#behind-a-proxy) +- [Contributing](#contributing) +- [License](#license) -## The focus of this fork +## About this fork -The focus of this fork is +#### Motivation -- Github -- Kubernetes and VM based backends -- Linux/amd64 -- Some really good features that Drone 1.0 introduced: multiple pipelines, cron triggers +Why fork? See my [motivation](docs/motivation.md) -## Why should you use this fork? +#### The focus of this fork -you shouldn't necessarily. Paying for Drone 1.0 is a fine choice. +This fork is not meant to compete with Drone or reimplement its enterprise features in the open. -Check the issues and releases of this project if you are evaluating this project. -Also you can check the devlog to get the nuances: https://laszlo.cloud/drone-oss-08-devlog-1 +Instead, I'm taking a proven CI system - that Drone 0.8 is - and applying a distinct set of product ideas focusing on: -The project is currently used by one user, with 50+ repos and 500+ builds a week. \ No newline at end of file +- UI experience +- the developer feedback loop +- documentation and best practices +- tighter Github integration +- Kubernetes backend + +with less focus on: + +- niche git systems like gitea, gogs +- computing architectures like arm64 +- new pipeline formats like jsonnet + +#### Who uses this fork + +Currently I know of one organization using this fork. With 50+ users, 130+ repos and more than 300 builds a week. + +## Pipelines + +#### Getting started + +Place this snippet into a file called `.drone.yml` + +```yaml +pipeline: + build: + image: debian:stable-slim + commands: + - echo "This is the build step" + a-test-step: + image: debian:stable-slim + commands: + - echo "Testing.." +``` + +The pipeline runs on the Drone CI server and typically triggered by webhooks. One benefit of the container architecture is that it runs on your laptop too: + +```sh +$ drone exec --local +stable-slim: Pulling from library/debian +a94641239323: Pull complete +Digest: sha256:d846d80f98c8aca7d3db0fadd14a0a4c51a2ce1eb2e9e14a550b3bd0c45ba941 +Status: Downloaded newer image for debian:stable-slim +[build:L0:0s] + echo "This is the build step" +[build:L1:0s] This is the build step +[a-test-step:L0:0s] + echo "Testing.." +[a-test-step:L1:0s] Testing.. +``` + +Pipeline steps are commands running in container images. +These containers are wired together and they share a volume with the source code on it. + +#### Pipeline documentation + +See all [pipeline features](docs/usage/pipeline.md). + +## Plugins + +Plugins are Docker containers that perform pre-defined tasks and are configured as steps in your pipeline. Plugins can be used to deploy code, publish artifacts, send notification, and more. + +Example pipeline using the Docker and Slack plugins: + +```yaml +pipeline: + backend: + image: golang + commands: + - go get + - go build + - go test + + docker: + image: plugins/docker + username: kevinbacon + password: pa55word + repo: foo/bar + tags: latest + + notify: + image: plugins/slack + channel: developers + username: drone +``` + +#### Custom plugins + +Plugins are Docker containers with their entrypoint set to a predefined script. + +[See how an example plugin can be implemented in a bash script](docs/usage/bash_plugin.md). + +## Server setup + +#### Quickstart + +The below [docker-compose](https://docs.docker.com/compose/) configuration can be used to start the Drone server with a single agent. It relies on a number of environment variables that you must set before running `docker-compose up`. The variables are described below. + +Each agent is able to process one build by default. If you have 4 agents installed and connected to the Drone server, your system will process 4 builds in parallel. You can add more agents to increase the number of parallel builds or set the agent's `DRONE_MAX_PROCS=1` environment variable to increase the number of parallel builds for that agent. + +```yaml +version: '2' + +services: + drone-server: + image: drone/drone:{{% version %}} + ports: + - 80:8000 + - 9000 + volumes: + - drone-server-data:/var/lib/drone/ + restart: always + environment: + - DRONE_OPEN=true + - DRONE_HOST=${DRONE_HOST} + - DRONE_GITHUB=true + - DRONE_GITHUB_CLIENT=${DRONE_GITHUB_CLIENT} + - DRONE_GITHUB_SECRET=${DRONE_GITHUB_SECRET} + - DRONE_SECRET=${DRONE_SECRET} + + drone-agent: + image: drone/agent:{{% version %}} + command: agent + restart: always + depends_on: + - drone-server + volumes: + - /var/run/docker.sock:/var/run/docker.sock + environment: + - DRONE_SERVER=drone-server:9000 + - DRONE_SECRET=${DRONE_SECRET} + +volumes: + drone-server-data: +``` + +Drone needs to know its own address. You must therefore provide the address in `://` format. Please omit trailing slashes. + +```diff +services: + drone-server: + image: drone/drone:{{% version %}} + environment: + - DRONE_OPEN=true ++ - DRONE_HOST=${DRONE_HOST} + - DRONE_GITHUB=true + - DRONE_GITHUB_CLIENT=${DRONE_GITHUB_CLIENT} + - DRONE_GITHUB_SECRET=${DRONE_GITHUB_SECRET} + - DRONE_SECRET=${DRONE_SECRET} +``` + +Drone agents require access to the host machine Docker daemon. + +```diff +services: + drone-agent: + image: drone/agent:{{% version %}} + command: agent + restart: always + depends_on: [ drone-server ] ++ volumes: ++ - /var/run/docker.sock:/var/run/docker.sock +``` + +Drone agents require the server address for agent-to-server communication. + +```diff +services: + drone-agent: + image: drone/agent:{{% version %}} + command: agent + restart: always + depends_on: [ drone-server ] + volumes: + - /var/run/docker.sock:/var/run/docker.sock + environment: ++ - DRONE_SERVER=drone-server:9000 + - DRONE_SECRET=${DRONE_SECRET} +``` + +Drone server and agents use a shared secret to authenticate communication. This should be a random string of your choosing and should be kept private. You can generate such string with `openssl rand -hex 32`. + +```diff +services: + drone-server: + image: drone/drone:{{% version %}} + environment: + - DRONE_OPEN=true + - DRONE_HOST=${DRONE_HOST} + - DRONE_GITHUB=true + - DRONE_GITHUB_CLIENT=${DRONE_GITHUB_CLIENT} + - DRONE_GITHUB_SECRET=${DRONE_GITHUB_SECRET} ++ - DRONE_SECRET=${DRONE_SECRET} + drone-agent: + image: drone/agent:{{% version %}} + environment: + - DRONE_SERVER=drone-server:9000 + - DRONE_DEBUG=true ++ - DRONE_SECRET=${DRONE_SECRET} +``` + +Drone registration is closed by default. This example enables open registration for users that are members of approved GitHub organizations. + +```diff +services: + drone-server: + image: drone/drone:{{% version %}} + environment: ++ - DRONE_OPEN=true ++ - DRONE_ORGS=dolores,dogpatch + - DRONE_HOST=${DRONE_HOST} + - DRONE_GITHUB=true + - DRONE_GITHUB_CLIENT=${DRONE_GITHUB_CLIENT} + - DRONE_GITHUB_SECRET=${DRONE_GITHUB_SECRET} + - DRONE_SECRET=${DRONE_SECRET} +``` + +Drone administrators should also be enumerated in your configuration. + +```diff +services: + drone-server: + image: drone/drone:{{% version %}} + environment: + - DRONE_OPEN=true + - DRONE_ORGS=dolores,dogpatch ++ - DRONE_ADMIN=johnsmith,janedoe + - DRONE_HOST=${DRONE_HOST} + - DRONE_GITHUB=true + - DRONE_GITHUB_CLIENT=${DRONE_GITHUB_CLIENT} + - DRONE_GITHUB_SECRET=${DRONE_GITHUB_SECRET} + - DRONE_SECRET=${DRONE_SECRET} +``` + +#### Authentication + +Authentication is done using OAuth and is delegated to one of multiple version control providers, configured using environment variables. The example above demonstrates basic GitHub integration. + +See the complete reference for [Github](docs/administration/github.md), [Bitbucket Cloud](docs/administration/bitbucket.md), [Bitbucket Server](docs/administration/bitbucket_server.md) and [Gitlab](docs/administration/gitlab.md). + +#### Database + +Drone mounts a [data volume](https://docs.docker.com/storage/volumes/#create-and-manage-volumes) to persist the sqlite database. + +See the [database settings](docs/administration/database.md) page to configure Postgresql or MySQL as database. + +```diff +services: + drone-server: + image: drone/drone:{{% version %}} + ports: + - 80:8000 + - 9000 ++ volumes: ++ - drone-server-data:/var/lib/drone/ + restart: always +``` + +#### SSL + +Drone supports ssl configuration by mounting certificates into your container. + +See the [SSL guide](docs/administration/ssl.md). + +Automated [Lets Encrypt](docs/administration/lets_encrypt.md) is also supported. + +#### Metrics + +A [Prometheus endpoint](docs/administration/lets_encrypt.md) is exposed. + +#### Behind a proxy + +See the [proxy guide](docs/administration/proxy.md) if you want to see a setup behind Apache, Nginx, Caddy or ngrok. + +## Contributing + +Drone-OSS-08 is Apache 2.0 licensed and accepts contributions via GitHub pull requests. + +[How to build the project]() + +## License + +Drone-OSS-08 is Apache 2.0 licensed with the source files in this repository having a header indicating which license they are under and what copyrights apply. + +Files under the `docs/` folder is licensed under Creative Commons Attribution-ShareAlike 4.0 International Public License. It is a derivative work of the https://github.com/drone/docs git repository. diff --git a/cli/drone/deploy/deploy.go b/cli/drone/deploy/deploy.go index c97a254bd..27bec18c1 100644 --- a/cli/drone/deploy/deploy.go +++ b/cli/drone/deploy/deploy.go @@ -7,7 +7,7 @@ import ( "strconv" "github.com/laszlocph/drone-oss-08/cli/drone/internal" - "github.com/drone/drone-go/drone" + "github.com/laszlocph/drone-oss-08/drone-go/drone" "github.com/urfave/cli" ) diff --git a/cli/drone/internal/util.go b/cli/drone/internal/util.go index a25181d4d..8f9d8bba3 100644 --- a/cli/drone/internal/util.go +++ b/cli/drone/internal/util.go @@ -11,7 +11,7 @@ import ( "golang.org/x/net/proxy" "golang.org/x/oauth2" - "github.com/drone/drone-go/drone" + "github.com/laszlocph/drone-oss-08/drone-go/drone" ) // NewClient returns a new client from the CLI context. diff --git a/cli/drone/registry/registry_add.go b/cli/drone/registry/registry_add.go index d83e1f317..0378432c0 100644 --- a/cli/drone/registry/registry_add.go +++ b/cli/drone/registry/registry_add.go @@ -5,7 +5,7 @@ import ( "strings" "github.com/laszlocph/drone-oss-08/cli/drone/internal" - "github.com/drone/drone-go/drone" + "github.com/laszlocph/drone-oss-08/drone-go/drone" "github.com/urfave/cli" ) diff --git a/cli/drone/registry/registry_set.go b/cli/drone/registry/registry_set.go index 98a47846f..25ce312f7 100644 --- a/cli/drone/registry/registry_set.go +++ b/cli/drone/registry/registry_set.go @@ -5,7 +5,7 @@ import ( "strings" "github.com/laszlocph/drone-oss-08/cli/drone/internal" - "github.com/drone/drone-go/drone" + "github.com/laszlocph/drone-oss-08/drone-go/drone" "github.com/urfave/cli" ) diff --git a/cli/drone/repo/repo_update.go b/cli/drone/repo/repo_update.go index 53f2253b5..d773d237c 100644 --- a/cli/drone/repo/repo_update.go +++ b/cli/drone/repo/repo_update.go @@ -5,7 +5,7 @@ import ( "time" "github.com/laszlocph/drone-oss-08/cli/drone/internal" - "github.com/drone/drone-go/drone" + "github.com/laszlocph/drone-oss-08/drone-go/drone" "github.com/urfave/cli" ) diff --git a/cli/drone/secret/secret_add.go b/cli/drone/secret/secret_add.go index ee931e14a..20879135e 100644 --- a/cli/drone/secret/secret_add.go +++ b/cli/drone/secret/secret_add.go @@ -5,7 +5,7 @@ import ( "strings" "github.com/laszlocph/drone-oss-08/cli/drone/internal" - "github.com/drone/drone-go/drone" + "github.com/laszlocph/drone-oss-08/drone-go/drone" "github.com/urfave/cli" ) diff --git a/cli/drone/secret/secret_set.go b/cli/drone/secret/secret_set.go index 656eb3ea8..080f2dfd6 100644 --- a/cli/drone/secret/secret_set.go +++ b/cli/drone/secret/secret_set.go @@ -5,7 +5,7 @@ import ( "strings" "github.com/laszlocph/drone-oss-08/cli/drone/internal" - "github.com/drone/drone-go/drone" + "github.com/laszlocph/drone-oss-08/drone-go/drone" "github.com/urfave/cli" ) diff --git a/cli/drone/user/user_add.go b/cli/drone/user/user_add.go index 67c0fee28..836be8e7e 100644 --- a/cli/drone/user/user_add.go +++ b/cli/drone/user/user_add.go @@ -3,7 +3,7 @@ package user import ( "fmt" - "github.com/drone/drone-go/drone" + "github.com/laszlocph/drone-oss-08/drone-go/drone" "github.com/urfave/cli" "github.com/laszlocph/drone-oss-08/cli/drone/internal" diff --git a/cmd/drone-server/server.go b/cmd/drone-server/server.go index be82d6fb5..18393bdfb 100644 --- a/cmd/drone-server/server.go +++ b/cmd/drone-server/server.go @@ -563,23 +563,17 @@ func server(c *cli.Context) error { auther := &authorizer{ password: c.String("agent-secret"), } - s := grpc.NewServer( + grpcServer := grpc.NewServer( grpc.StreamInterceptor(auther.streamInterceptor), grpc.UnaryInterceptor(auther.unaryIntercaptor), grpc.KeepaliveEnforcementPolicy(keepalive.EnforcementPolicy{ MinTime: c.Duration("keepalive-min-time"), }), ) - ss := new(droneserver.DroneServer) - ss.Queue = droneserver.Config.Services.Queue - ss.Logger = droneserver.Config.Services.Logs - ss.Pubsub = droneserver.Config.Services.Pubsub - ss.Remote = remote_ - ss.Store = store_ - ss.Host = droneserver.Config.Server.Host - proto.RegisterDroneServer(s, ss) + droneServer := droneserver.NewDroneServer(remote_, droneserver.Config.Services.Queue, droneserver.Config.Services.Logs, droneserver.Config.Services.Pubsub, store_, droneserver.Config.Server.Host) + proto.RegisterDroneServer(grpcServer, droneServer) - err = s.Serve(lis) + err = grpcServer.Serve(lis) if err != nil { logrus.Error(err) return err @@ -651,11 +645,6 @@ func server(c *cli.Context) error { return g.Wait() } -// HACK please excuse the message during this period of heavy refactoring. -// We are currently transitioning from storing services (ie database, queue) -// in the gin.Context to storing them in a struct. We are also moving away -// from gin to gorilla. We will temporarily use global during our refactoring -// which will be removing in the final implementation. func setupEvilGlobals(c *cli.Context, v store.Store, r remote.Remote) { // storage diff --git a/cmd/drone-server/setup.go b/cmd/drone-server/setup.go index 16f5934db..ababab63a 100644 --- a/cmd/drone-server/setup.go +++ b/cmd/drone-server/setup.go @@ -219,7 +219,7 @@ func setupMetrics(g *errgroup.Group, store_ store.Store) { }) builds := promauto.NewGauge(prometheus.GaugeOpts{ Namespace: "drone", - Name: "build_count", + Name: "build_total_count", Help: "Total number of builds.", }) users := promauto.NewGauge(prometheus.GaugeOpts{ diff --git a/cncd/logging/log_test.go b/cncd/logging/log_test.go index 20d8e3d55..2c235297e 100644 --- a/cncd/logging/log_test.go +++ b/cncd/logging/log_test.go @@ -30,7 +30,7 @@ func TestLogging(t *testing.T) { logger.Tail(ctx, testPath, func(entry ...*Entry) { wg.Done() }) }() - <-time.After(time.Millisecond) + <-time.After(500 * time.Millisecond) wg.Add(4) go func() { @@ -45,7 +45,7 @@ func TestLogging(t *testing.T) { logger.Tail(ctx, testPath, func(entry ...*Entry) { wg.Done() }) }() - <-time.After(time.Millisecond) + <-time.After(500 * time.Millisecond) wg.Wait() cancel() diff --git a/cncd/pipeline/pipeline/frontend/metadata.go b/cncd/pipeline/pipeline/frontend/metadata.go index edd4b0b62..c46791add 100644 --- a/cncd/pipeline/pipeline/frontend/metadata.go +++ b/cncd/pipeline/pipeline/frontend/metadata.go @@ -222,3 +222,10 @@ func (m *Metadata) EnvironDrone() map[string]string { } var pullRegexp = regexp.MustCompile("\\d+") + +func (m *Metadata) SetPlatform(platform string) { + if platform == "" { + platform = "linux/amd64" + } + m.Sys.Arch = platform +} diff --git a/cncd/pipeline/pipeline/frontend/yaml/compiler/compiler.go b/cncd/pipeline/pipeline/frontend/yaml/compiler/compiler.go index d21311d36..4049a10f9 100644 --- a/cncd/pipeline/pipeline/frontend/yaml/compiler/compiler.go +++ b/cncd/pipeline/pipeline/frontend/yaml/compiler/compiler.go @@ -97,7 +97,7 @@ func (c *Compiler) Compile(conf *yaml.Config) *backend.Config { } // add default clone step - if c.local == false && len(conf.Clone.Containers) == 0 { + if c.local == false && len(conf.Clone.Containers) == 0 && !conf.SkipClone { container := &yaml.Container{ Name: "clone", Image: "plugins/git:latest", @@ -118,7 +118,7 @@ func (c *Compiler) Compile(conf *yaml.Config) *backend.Config { stage.Steps = append(stage.Steps, step) config.Stages = append(config.Stages, stage) - } else if c.local == false { + } else if c.local == false && !conf.SkipClone { for i, container := range conf.Clone.Containers { if !container.Constraints.Match(c.metadata) { continue diff --git a/cncd/pipeline/pipeline/frontend/yaml/config.go b/cncd/pipeline/pipeline/frontend/yaml/config.go index e5d52f8c2..9893dbf26 100644 --- a/cncd/pipeline/pipeline/frontend/yaml/config.go +++ b/cncd/pipeline/pipeline/frontend/yaml/config.go @@ -22,6 +22,9 @@ type ( Networks Networks Volumes Volumes Labels libcompose.SliceorMap + DependsOn []string `yaml:"depends_on,omitempty"` + RunsOn []string `yaml:"runs_on,omitempty"` + SkipClone bool `yaml:"skip_clone"` } // Workspace defines a pipeline workspace. diff --git a/cncd/pipeline/pipeline/frontend/yaml/config_test.go b/cncd/pipeline/pipeline/frontend/yaml/config_test.go index 54549ffda..79c360d19 100644 --- a/cncd/pipeline/pipeline/frontend/yaml/config_test.go +++ b/cncd/pipeline/pipeline/frontend/yaml/config_test.go @@ -7,7 +7,7 @@ import ( "github.com/franela/goblin" ) -func xTestParse(t *testing.T) { +func TestParse(t *testing.T) { g := goblin.Goblin(t) g.Describe("Parser", func() { @@ -35,9 +35,14 @@ func xTestParse(t *testing.T) { g.Assert(out.Pipeline.Containers[1].Commands).Equal(yaml.Stringorslice{"go build"}) g.Assert(out.Pipeline.Containers[2].Name).Equal("notify") g.Assert(out.Pipeline.Containers[2].Image).Equal("slack") - g.Assert(out.Pipeline.Containers[2].NetworkMode).Equal("container:name") + // g.Assert(out.Pipeline.Containers[2].NetworkMode).Equal("container:name") g.Assert(out.Labels["com.example.team"]).Equal("frontend") g.Assert(out.Labels["com.example.type"]).Equal("build") + g.Assert(out.DependsOn[0]).Equal("lint") + g.Assert(out.DependsOn[1]).Equal("test") + g.Assert(out.RunsOn[0]).Equal("success") + g.Assert(out.RunsOn[1]).Equal("failure") + g.Assert(out.SkipClone).Equal(false) }) // Check to make sure variable expansion works in yaml.MapSlice // g.It("Should unmarshal variables", func() { @@ -94,6 +99,12 @@ volumes: labels: com.example.type: "build" com.example.team: "frontend" +depends_on: + - lint + - test +runs_on: + - success + - failure ` var sampleVarYaml = ` diff --git a/cncd/pipeline/pipeline/frontend/yaml/matrix/matrix.go b/cncd/pipeline/pipeline/frontend/yaml/matrix/matrix.go index 76a0f6bed..7177e1a2f 100644 --- a/cncd/pipeline/pipeline/frontend/yaml/matrix/matrix.go +++ b/cncd/pipeline/pipeline/frontend/yaml/matrix/matrix.go @@ -40,9 +40,8 @@ func Parse(data []byte) ([]Axis, error) { return nil, err } - // if not a matrix build return an array with just the single axis. if len(matrix) == 0 { - return nil, nil + return []Axis{}, nil } return calc(matrix), nil diff --git a/cncd/pipeline/pipeline/frontend/yaml/matrix/matrix_test.go b/cncd/pipeline/pipeline/frontend/yaml/matrix/matrix_test.go index 0c4eef50a..690d71082 100644 --- a/cncd/pipeline/pipeline/frontend/yaml/matrix/matrix_test.go +++ b/cncd/pipeline/pipeline/frontend/yaml/matrix/matrix_test.go @@ -25,10 +25,10 @@ func TestMatrix(t *testing.T) { g.Assert(len(set)).Equal(24) }) - g.It("Should return nil if no matrix", func() { + g.It("Should return empty array if no matrix", func() { axis, err := ParseString("") g.Assert(err == nil).IsTrue() - g.Assert(axis == nil).IsTrue() + g.Assert(len(axis) == 0).IsTrue() }) g.It("Should return included axis", func() { diff --git a/cncd/pubsub/pub_test.go b/cncd/pubsub/pub_test.go index 574c0e673..007f3192d 100644 --- a/cncd/pubsub/pub_test.go +++ b/cncd/pubsub/pub_test.go @@ -30,7 +30,7 @@ func TestPubsub(t *testing.T) { broker.Subscribe(ctx, testTopic, func(message Message) { wg.Done() }) }() - <-time.After(time.Millisecond) + <-time.After(500 * time.Millisecond) if _, ok := broker.(*publisher).topics[testTopic]; !ok { t.Errorf("Expect topic registered with publisher") @@ -86,7 +86,7 @@ func TestSubscriptionClosed(t *testing.T) { wg.Done() }() - <-time.After(time.Millisecond) + <-time.After(500 * time.Millisecond) if _, ok := broker.(*publisher).topics[testTopic]; !ok { t.Errorf("Expect topic registered with publisher") diff --git a/cncd/queue/fifo.go b/cncd/queue/fifo.go index ced613d64..bc8936ba6 100644 --- a/cncd/queue/fifo.go +++ b/cncd/queue/fifo.go @@ -7,6 +7,8 @@ import ( "runtime" "sync" "time" + + "github.com/Sirupsen/logrus" ) type entry struct { @@ -29,6 +31,7 @@ type fifo struct { running map[string]*entry pending *list.List extension time.Duration + paused bool } // New returns a new fifo queue. @@ -38,6 +41,7 @@ func New() Queue { running: map[string]*entry{}, pending: list.New(), extension: time.Minute * 10, + paused: false, } } @@ -50,6 +54,17 @@ func (q *fifo) Push(c context.Context, task *Task) error { return nil } +// Push pushes an item to the tail of this queue. +func (q *fifo) PushAtOnce(c context.Context, tasks []*Task) error { + q.Lock() + for _, task := range tasks { + q.pending.PushBack(task) + } + q.Unlock() + go q.process() + return nil +} + // Poll retrieves and removes the head of this queue. func (q *fifo) Poll(c context.Context, f Filter) (*Task, error) { q.Lock() @@ -82,11 +97,14 @@ func (q *fifo) Done(c context.Context, id string) error { // Error signals that the item is done executing with error. func (q *fifo) Error(c context.Context, id string, err error) error { q.Lock() - state, ok := q.running[id] + taskEntry, ok := q.running[id] if ok { - state.error = err - close(state.done) + q.updateDepStatusInQueue(id, err == nil) + taskEntry.error = err + close(taskEntry.done) delete(q.running, id) + } else { + q.removeFromPending(id) } q.Unlock() return nil @@ -151,14 +169,32 @@ func (q *fifo) Info(c context.Context) InfoT { for _, entry := range q.running { stats.Running = append(stats.Running, entry.item) } + stats.Paused = q.paused q.Unlock() return stats } +func (q *fifo) Pause() { + q.Lock() + q.paused = true + q.Unlock() +} + +func (q *fifo) Resume() { + q.Lock() + q.paused = false + q.Unlock() + go q.process() +} + // helper function that loops through the queue and attempts to // match the item to a single subscriber. func (q *fifo) process() { + if q.paused { + return + } + defer func() { // the risk of panic is low. This code can probably be removed // once the code has been used in real world installs without issue. @@ -173,8 +209,44 @@ func (q *fifo) process() { q.Lock() defer q.Unlock() - // TODO(bradrydzewski) move this to a helper function - // push items to the front of the queue if the item expires. + q.resubmitExpiredBuilds() + + for pending, worker := q.assignToWorker(); pending != nil && worker != nil; pending, worker = q.assignToWorker() { + task := pending.Value.(*Task) + delete(q.workers, worker) + q.pending.Remove(pending) + q.running[task.ID] = &entry{ + item: task, + done: make(chan bool), + deadline: time.Now().Add(q.extension), + } + worker.channel <- task + } +} + +func (q *fifo) assignToWorker() (*list.Element, *worker) { + var next *list.Element + for e := q.pending.Front(); e != nil; e = next { + next = e.Next() + task := e.Value.(*Task) + logrus.Debugf("queue: trying to assign task: %v with deps %v", task.ID, task.Dependencies) + if q.depsInQueue(task) { + logrus.Debugf("queue: skipping due to unmet dependencies %v", task.ID) + continue + } + + for w := range q.workers { + if w.filter(task) { + logrus.Debugf("queue: assigned task: %v with deps %v", task.ID, task.Dependencies) + return e, w + } + } + } + + return nil, nil +} + +func (q *fifo) resubmitExpiredBuilds() { for id, state := range q.running { if time.Now().After(state.deadline) { q.pending.PushFront(state.item) @@ -182,26 +254,61 @@ func (q *fifo) process() { close(state.done) } } +} +func (q *fifo) depsInQueue(task *Task) bool { var next *list.Element -loop: for e := q.pending.Front(); e != nil; e = next { next = e.Next() - item := e.Value.(*Task) - for w := range q.workers { - if w.filter(item) { - delete(q.workers, w) - q.pending.Remove(e) + possibleDep, ok := e.Value.(*Task) + logrus.Debugf("queue: pending right now: %v", possibleDep.ID) + for _, dep := range task.Dependencies { + if ok && possibleDep.ID == dep { + return true + } + } + } + for possibleDepID := range q.running { + logrus.Debugf("queue: running right now: %v", possibleDepID) + for _, dep := range task.Dependencies { + if possibleDepID == dep { + return true + } + } + } + return false +} - q.running[item.ID] = &entry{ - item: item, - done: make(chan bool), - deadline: time.Now().Add(q.extension), - } - - w.channel <- item - break loop +func (q *fifo) updateDepStatusInQueue(taskID string, success bool) { + var next *list.Element + for e := q.pending.Front(); e != nil; e = next { + next = e.Next() + pending, ok := e.Value.(*Task) + for _, dep := range pending.Dependencies { + if ok && taskID == dep { + pending.DepStatus[dep] = success + } + } + } + for _, running := range q.running { + for _, dep := range running.item.Dependencies { + if taskID == dep { + running.item.DepStatus[dep] = success } } } } + +func (q *fifo) removeFromPending(taskID string) { + logrus.Debugf("queue: trying to remove %s", taskID) + var next *list.Element + for e := q.pending.Front(); e != nil; e = next { + next = e.Next() + task := e.Value.(*Task) + if task.ID == taskID { + logrus.Debugf("queue: %s is removed from pending", taskID) + q.pending.Remove(e) + return + } + } +} diff --git a/cncd/queue/fifo_test.go b/cncd/queue/fifo_test.go index 5123671bd..2252bbb7f 100644 --- a/cncd/queue/fifo_test.go +++ b/cncd/queue/fifo_test.go @@ -2,6 +2,7 @@ package queue import ( "context" + "fmt" "sync" "testing" "time" @@ -117,3 +118,234 @@ func TestFifoEvict(t *testing.T) { t.Errorf("expect not found error when evicting item not in queue, got %s", err) } } + +func TestFifoDependencies(t *testing.T) { + task1 := &Task{ + ID: "1", + } + + task2 := &Task{ + ID: "2", + Dependencies: []string{"1"}, + DepStatus: make(map[string]bool), + } + + q := New().(*fifo) + q.Push(noContext, task2) + q.Push(noContext, task1) + + got, _ := q.Poll(noContext, func(*Task) bool { return true }) + if got != task1 { + t.Errorf("expect task1 returned from queue as task2 depends on it") + return + } + + q.Done(noContext, got.ID) + + got, _ = q.Poll(noContext, func(*Task) bool { return true }) + if got != task2 { + t.Errorf("expect task2 returned from queue") + return + } +} + +func TestFifoErrors(t *testing.T) { + task1 := &Task{ + ID: "1", + } + + task2 := &Task{ + ID: "2", + Dependencies: []string{"1"}, + DepStatus: make(map[string]bool), + } + + task3 := &Task{ + ID: "3", + Dependencies: []string{"1"}, + DepStatus: make(map[string]bool), + RunOn: []string{"success", "failure"}, + } + + q := New().(*fifo) + q.Push(noContext, task2) + q.Push(noContext, task3) + q.Push(noContext, task1) + + got, _ := q.Poll(noContext, func(*Task) bool { return true }) + if got != task1 { + t.Errorf("expect task1 returned from queue as task2 depends on it") + return + } + + q.Error(noContext, got.ID, fmt.Errorf("exitcode 1, there was an error")) + + got, _ = q.Poll(noContext, func(*Task) bool { return true }) + if got != task2 { + t.Errorf("expect task2 returned from queue") + return + } + + if got.ShouldRun() { + t.Errorf("expect task2 should not run, since task1 failed") + return + } + + got, _ = q.Poll(noContext, func(*Task) bool { return true }) + if got != task3 { + t.Errorf("expect task3 returned from queue") + return + } + + if !got.ShouldRun() { + t.Errorf("expect task3 should run, task1 failed, but task3 runs on failure too") + return + } +} + +func TestFifoCancel(t *testing.T) { + task1 := &Task{ + ID: "1", + } + + task2 := &Task{ + ID: "2", + Dependencies: []string{"1"}, + DepStatus: make(map[string]bool), + } + + task3 := &Task{ + ID: "3", + Dependencies: []string{"1"}, + DepStatus: make(map[string]bool), + RunOn: []string{"success", "failure"}, + } + + q := New().(*fifo) + q.Push(noContext, task2) + q.Push(noContext, task3) + q.Push(noContext, task1) + + _, _ = q.Poll(noContext, func(*Task) bool { return true }) + q.Error(noContext, task1.ID, fmt.Errorf("cancelled")) + q.Error(noContext, task2.ID, fmt.Errorf("cancelled")) + q.Error(noContext, task3.ID, fmt.Errorf("cancelled")) + + info := q.Info(noContext) + if len(info.Pending) != 0 { + t.Errorf("All pipelines should be cancelled") + return + } +} + +func TestFifoPause(t *testing.T) { + task1 := &Task{ + ID: "1", + } + + q := New().(*fifo) + var wg sync.WaitGroup + wg.Add(1) + go func() { + _, _ = q.Poll(noContext, func(*Task) bool { return true }) + wg.Done() + }() + + + q.Pause() + t0 := time.Now() + q.Push(noContext, task1) + time.Sleep(20 * time.Millisecond) + q.Resume() + + wg.Wait() + t1 := time.Now() + + if t1.Sub(t0) < 20 * time.Millisecond { + t.Errorf("Should have waited til resume") + } + + q.Pause() + q.Push(noContext, task1) + q.Resume() + _, _ = q.Poll(noContext, func(*Task) bool { return true }) +} + +func TestFifoPauseResume(t *testing.T) { + task1 := &Task{ + ID: "1", + } + + q := New().(*fifo) + q.Pause() + q.Push(noContext, task1) + q.Resume() + + _, _ = q.Poll(noContext, func(*Task) bool { return true }) +} + +func TestShouldRun(t *testing.T) { + task := &Task{ + ID: "2", + Dependencies: []string{"1"}, + DepStatus: map[string]bool{ + "1": true, + }, + RunOn: []string{"failure"}, + } + if task.ShouldRun() { + t.Errorf("expect task to not run, it runs on failure only") + return + } + + task = &Task{ + ID: "2", + Dependencies: []string{"1"}, + DepStatus: map[string]bool{ + "1": true, + }, + RunOn: []string{"failure", "success"}, + } + if !task.ShouldRun() { + t.Errorf("expect task to run") + return + } + + task = &Task{ + ID: "2", + Dependencies: []string{"1"}, + DepStatus: map[string]bool{ + "1": false, + }, + } + if task.ShouldRun() { + t.Errorf("expect task to not run") + return + } + + task = &Task{ + ID: "2", + Dependencies: []string{"1"}, + DepStatus: map[string]bool{ + "1": true, + }, + RunOn: []string{"success"}, + } + if !task.ShouldRun() { + t.Errorf("expect task to run") + return + } + + task = &Task{ + ID: "2", + Dependencies: []string{"1"}, + DepStatus: map[string]bool{ + "1": false, + }, + RunOn: []string{"failure"}, + } + if !task.ShouldRun() { + t.Errorf("expect task to run") + return + } +} diff --git a/cncd/queue/queue.go b/cncd/queue/queue.go index fa25a798a..fc1641c36 100644 --- a/cncd/queue/queue.go +++ b/cncd/queue/queue.go @@ -23,6 +23,64 @@ type Task struct { // Labels represents the key-value pairs the entry is lebeled with. Labels map[string]string `json:"labels,omitempty"` + + // Task IDs this task depend + Dependencies []string + + // If dep finished sucessfully + DepStatus map[string]bool + + // RunOn failure or success + RunOn []string +} + +// ShouldRun tells if a task should be run or skipped, based on dependencies +func (t *Task) ShouldRun() bool { + if runsOnFailure(t.RunOn) && runsOnSuccess(t.RunOn) { + return true + } + + if !runsOnFailure(t.RunOn) && runsOnSuccess(t.RunOn) { + for _, success := range t.DepStatus { + if !success { + return false + } + } + return true + } + + if runsOnFailure(t.RunOn) && !runsOnSuccess(t.RunOn) { + for _, success := range t.DepStatus { + if success { + return false + } + } + return true + } + + return false +} + +func runsOnFailure(runsOn []string) bool { + for _, status := range runsOn { + if status == "failure" { + return true + } + } + return false +} + +func runsOnSuccess(runsOn []string) bool { + if len(runsOn) == 0 { + return true + } + + for _, status := range runsOn { + if status == "success" { + return true + } + } + return false } // InfoT provides runtime information. @@ -35,6 +93,7 @@ type InfoT struct { Running int `json:"running_count"` Complete int `json:"completed_count"` } `json:"stats"` + Paused bool } // Filter filters tasks in the queue. If the Filter returns false, @@ -44,9 +103,12 @@ type Filter func(*Task) bool // Queue defines a task queue for scheduling tasks among // a pool of workers. type Queue interface { - // Push pushes an task to the tail of this queue. + // Push pushes a task to the tail of this queue. Push(c context.Context, task *Task) error + // Push pushes a task to the tail of this queue. + PushAtOnce(c context.Context, tasks []*Task) error + // Poll retrieves and removes a task head of this queue. Poll(c context.Context, f Filter) (*Task, error) @@ -67,47 +129,10 @@ type Queue interface { // Info returns internal queue information. Info(c context.Context) InfoT -} -// // global instance of the queue. -// var global = New() -// -// // Set sets the global queue. -// func Set(queue Queue) { -// global = queue -// } -// -// // Push pushes an task to the tail of the global queue. -// func Push(c context.Context, task *Task) error { -// return global.Push(c, task) -// } -// -// // Poll retrieves and removes a task head of the global queue. -// func Poll(c context.Context, f Filter) (*Task, error) { -// return global.Poll(c, f) -// } -// -// // Extend extends the deadline for a task. -// func Extend(c context.Context, id string) error { -// return global.Extend(c, id) -// } -// -// // Done signals the task is complete. -// func Done(c context.Context, id string) error { -// return global.Done(c, id) -// } -// -// // Error signals the task is complete with errors. -// func Error(c context.Context, id string, err error) { -// global.Error(c, id, err) -// } -// -// // Wait waits until the task is complete. -// func Wait(c context.Context, id string) error { -// return global.Wait(c, id) -// } -// -// // Info returns internal queue information. -// func Info(c context.Context) InfoT { -// return global.Info(c) -// } + // Stops the queue from handing out new work items in Poll + Pause() + + // Starts the queue again, Poll returns new items + Resume() +} diff --git a/docker-compose.yml b/docker-compose.yml index 4a2594fb6..e90443644 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -28,4 +28,4 @@ services: environment: - DRONE_SERVER=drone-server:9000 - DRONE_SECRET=${DRONE_SECRET} - - DRONE_MAX_PROCS=1 \ No newline at end of file + - DRONE_MAX_PROCS=2 \ No newline at end of file diff --git a/docs/LICENSE b/docs/LICENSE new file mode 100644 index 000000000..53c093f79 --- /dev/null +++ b/docs/LICENSE @@ -0,0 +1 @@ +Files in this folder are licensed under Creative Commons Attribution-ShareAlike 4.0 International Public License. It is a derivative work of the https://github.com/drone/docs git repository. diff --git a/docs/administration/bitbucket.md b/docs/administration/bitbucket.md new file mode 100644 index 000000000..020e1686d --- /dev/null +++ b/docs/administration/bitbucket.md @@ -0,0 +1,70 @@ +Drone comes with built-in support for Bitbucket Cloud. To enable Bitbucket Cloud you should configure the Drone container using the following environment variables: + +```diff +version: '2' + +services: + drone-server: + image: drone/drone:{{% version %}} + ports: + - 80:8000 + - 9000 + volumes: + - /var/lib/drone:/var/lib/drone/ + restart: always + environment: + - DRONE_OPEN=true + - DRONE_HOST=${DRONE_HOST} ++ - DRONE_BITBUCKET=true ++ - DRONE_BITBUCKET_CLIENT=95c0282573633eb25e82 ++ - DRONE_BITBUCKET_SECRET=30f5064039e6b359e075 + - DRONE_SECRET=${DRONE_SECRET} + + drone-agent: + image: drone/agent:{{% version %}} + restart: always + depends_on: + - drone-server + volumes: + - /var/run/docker.sock:/var/run/docker.sock + environment: + - DRONE_SERVER=drone-server:9000 + - DRONE_SECRET=${DRONE_SECRET} +``` + +# Configuration + +This is a full list of configuration options. Please note that many of these options use default configuration values that should work for the majority of installations. + +DRONE_BITBUCKET=true +: Set to true to enable the Bitbucket driver. + +DRONE_BITBUCKET_CLIENT +: Bitbucket oauth2 client id + +DRONE_BITBUCKET_SECRET +: Bitbucket oauth2 client secret + +# Registration + +You must register your application with Bitbucket in order to generate a client and secret. Navigate to your account settings and choose OAuth from the menu, and click Add Consumer. + +Please use the Authorization callback URL: + +```nohighlight +http://drone.mycompany.com/authorize +``` + +Please also be sure to check the following permissions: + +```nohighlight +Account:Email +Account:Read +Team Membership:Read +Repositories:Read +Webhooks:Read and Write +``` + +# Missing Features + +Merge requests are not currently supported. We are interested in patches to include this functionality. If you are interested in contributing to Drone and submitting a patch please [contact us](https://discourse.drone.io). diff --git a/docs/administration/bitbucket_server.md b/docs/administration/bitbucket_server.md new file mode 100644 index 000000000..46595df25 --- /dev/null +++ b/docs/administration/bitbucket_server.md @@ -0,0 +1,133 @@ +Drone comes with experimental support for Bitbucket Server, formerly known as Atlassian Stash. To enable Bitbucket Server you should configure the Drone container using the following environment variables: + +```diff +version: '2' + +services: + drone-server: + image: drone/drone:{{% version %}} + ports: + - 80:8000 + - 9000 + volumes: + - /var/lib/drone:/var/lib/drone/ + restart: always + environment: + - DRONE_OPEN=true + - DRONE_HOST=${DRONE_HOST} ++ - DRONE_STASH=true ++ - DRONE_STASH_GIT_USERNAME=foo ++ - DRONE_STASH_GIT_PASSWORD=bar ++ - DRONE_STASH_CONSUMER_KEY=95c0282573633eb25e82 ++ - DRONE_STASH_CONSUMER_RSA=/etc/bitbucket/key.pem ++ - DRONE_STASH_URL=http://stash.mycompany.com + - DRONE_SECRET=${DRONE_SECRET} + volumes: ++ - /path/to/key.pem:/path/to/key.pem + + drone-agent: + image: drone/agent:{{% version %}} + restart: always + depends_on: + - drone-server + volumes: + - /var/run/docker.sock:/var/run/docker.sock + environment: + - DRONE_SERVER=drone-server:9000 + - DRONE_SECRET=${DRONE_SECRET} +``` + +# Private Key File + +The OAuth process in Bitbucket server requires a private and a public RSA certificate. This is how you create the private RSA certificate. + +```nohighlight +openssl genrsa -out /etc/bitbucket/key.pem 1024 +``` + +This stores the private RSA certificate in `key.pem`. The next command generates the public RSA certificate and stores it in `key.pub`. + +```nohighlight +openssl rsa -in /etc/bitbucket/key.pem -pubout >> /etc/bitbucket/key.pub +``` + +Please note that the private key file can be mounted into your Drone conatiner at runtime or as an environment variable + +Private key file mounted into your Drone container at runtime as a volume. + +```diff +version: '2' + +services: + drone-server: + image: drone/drone:{{% version %}} + environment: + - DRONE_OPEN=true + - DRONE_HOST=${DRONE_HOST} + - DRONE_STASH=true + - DRONE_STASH_GIT_USERNAME=foo + - DRONE_STASH_GIT_PASSWORD=bar + - DRONE_STASH_CONSUMER_KEY=95c0282573633eb25e82 ++ - DRONE_STASH_CONSUMER_RSA=/etc/bitbucket/key.pem + - DRONE_STASH_URL=http://stash.mycompany.com + - DRONE_SECRET=${DRONE_SECRET} ++ volumes: ++ - /etc/bitbucket/key.pem:/etc/bitbucket/key.pem +``` + +Private key as environment variable + +```diff +version: '2' + +services: + drone-server: + image: drone/drone:{{% version %}} + environment: + - DRONE_OPEN=true + - DRONE_HOST=${DRONE_HOST} + - DRONE_STASH=true + - DRONE_STASH_GIT_USERNAME=foo + - DRONE_STASH_GIT_PASSWORD=bar + - DRONE_STASH_CONSUMER_KEY=95c0282573633eb25e82 ++ - DRONE_STASH_CONSUMER_RSA_STRING=contentOfPemKeyAsString + - DRONE_STASH_URL=http://stash.mycompany.com + - DRONE_SECRET=${DRONE_SECRET} +``` + +# Service Account + +Drone uses `git+https` to clone repositories, however, Bitbucket Server does not currently support cloning repositories with oauth token. To work around this limitation, you must create a service account and provide the username and password to Drone. This service account will be used to authenticate and clone private repositories. + +# Registration + +You must register your application with Bitbucket Server in order to generate a consumer key. Navigate to your account settings and choose Applications from the menu, and click Register new application. Now copy & paste the text value from `/etc/bitbucket/key.pub` into the `Public Key` in the incoming link part of the application registration. + +Please use http://drone.mycompany.com/authorize as the Authorization callback URL. + + +# Configuration + +This is a full list of configuration options. Please note that many of these options use default configuration values that should work for the majority of installations. + + +DRONE_STASH=true +: Set to true to enable the Bitbucket Server (Stash) driver. + +DRONE_STASH_URL +: Bitbucket Server address. + +DRONE_STASH_CONSUMER_KEY +: Bitbucket Server oauth1 consumer key + +DRONE_STASH_CONSUMER_RSA +: Bitbucket Server oauth1 private key file + +DRONE_STASH_CONSUMER_RSA_STRING +: Bibucket Server oauth1 private key as a string + +DRONE_STASH_GIT_USERNAME +: Machine account username used to clone repositories. + +DRONE_STASH_GIT_PASSWORD +: Machine account password used to clone repositories. diff --git a/docs/administration/database.md b/docs/administration/database.md new file mode 100644 index 000000000..0bc1d9a21 --- /dev/null +++ b/docs/administration/database.md @@ -0,0 +1,48 @@ + +This guide provides instructions for using alternate storage engines. Please note this is optional. The default storage engine is an embedded SQLite database which requires zero installation or configuration. + +# Configure MySQL + +The below example demonstrates mysql database configuration. See the official driver [documentation](https://github.com/go-sql-driver/mysql#dsn-data-source-name) for configuration options and examples. + +```diff +version: '2' + +services: + drone-server: + image: drone/drone:{{% version %}} + environment: ++ DRONE_DATABASE_DRIVER: mysql ++ DRONE_DATABASE_DATASOURCE: root:password@tcp(1.2.3.4:3306)/drone?parseTime=true +``` + +# Configure Postgres + +The below example demonstrates postgres database configuration. See the official driver [documentation](https://www.postgresql.org/docs/current/static/libpq-connect.html#LIBPQ-CONNSTRING) for configuration options and examples. + +```diff +version: '2' + +services: + drone-server: + image: drone/drone:{{% version %}} + environment: ++ DRONE_DATABASE_DRIVER: postgres ++ DRONE_DATABASE_DATASOURCE: postgres://root:password@1.2.3.4:5432/postgres?sslmode=disable +``` + +# Database Creation + +Drone does not create your database automatically. If you are using the mysql or postgres driver you will need to manually create your database using `CREATE DATABASE` + +# Database Migration + +Drone automatically handles database migration, including the initial creation of tables and indexes. New versions of Drone will automatically upgrade the database unless otherwise specified in the release notes. + +# Database Backups + +Drone does not perform database backups. This should be handled by separate third party tools provided by your database vendor of choice. + +# Database Archiving + +Drone does not perform data archival; it considered out-of-scope for the project. Drone is rather conservative with the amount of data it stores, however, you should expect the database logs to grow the size of your database considerably. diff --git a/docs/administration/github.md b/docs/administration/github.md new file mode 100644 index 000000000..929f1131d --- /dev/null +++ b/docs/administration/github.md @@ -0,0 +1,78 @@ +Drone comes with built-in support for GitHub and GitHub Enterprise. To enable GitHub you should configure the Drone container using the following environment variables: + +```diff +version: '2' + +services: + drone-server: + image: drone/drone:{{% version %}} + ports: + - 80:8000 + - 9000 + volumes: + - /var/lib/drone:/var/lib/drone/ + restart: always + environment: + - DRONE_OPEN=true + - DRONE_HOST=${DRONE_HOST} ++ - DRONE_GITHUB=true ++ - DRONE_GITHUB_CLIENT=${DRONE_GITHUB_CLIENT} ++ - DRONE_GITHUB_SECRET=${DRONE_GITHUB_SECRET} + - DRONE_SECRET=${DRONE_SECRET} + + drone-agent: + image: drone/agent:{{% version %}} + restart: always + depends_on: + - drone-server + volumes: + - /var/run/docker.sock:/var/run/docker.sock + environment: + - DRONE_SERVER=drone-server:9000 + - DRONE_SECRET=${DRONE_SECRET} +``` + +# Registration + +Register your application with GitHub to create your client id and secret. It is very import the authorization callback URL matches your http(s) scheme and hostname exactly with `:///authorize` as the path. + +Please use this screenshot for reference: + +![github oauth setup](github_oauth.png) + +# Configuration + +This is a full list of configuration options. Please note that many of these options use default configuration values that should work for the majority of installations. + +DRONE_GITHUB=true +: Set to true to enable the GitHub driver. + +DRONE_GITHUB_URL=`https://github.com` +: GitHub server address. + +DRONE_GITHUB_CLIENT +: Github oauth2 client id. + +DRONE_GITHUB_SECRET +: Github oauth2 client secret. + +DRONE_GITHUB_SCOPE=repo,repo:status,user:email,read:org +: Comma-separated Github oauth scope. + +DRONE_GITHUB_GIT_USERNAME +: Optional. Use a single machine account username to clone all repositories. + +DRONE_GITHUB_GIT_PASSWORD +: Optional. Use a single machine account password to clone all repositories. + +DRONE_GITHUB_PRIVATE_MODE=false +: Set to true if Github is running in private mode. + +DRONE_GITHUB_MERGE_REF=true +: Set to true to use the `refs/pulls/%d/merge` vs `refs/pulls/%d/head` + +DRONE_GITHUB_CONTEXT=continuous-integration/drone +: Customize the GitHub status message context + +DRONE_GITHUB_SKIP_VERIFY=false +: Set to true to disable SSL verification. diff --git a/docs/administration/github_oauth.png b/docs/administration/github_oauth.png new file mode 100644 index 000000000..f37b19a74 Binary files /dev/null and b/docs/administration/github_oauth.png differ diff --git a/docs/administration/gitlab.md b/docs/administration/gitlab.md new file mode 100644 index 000000000..553c08f98 --- /dev/null +++ b/docs/administration/gitlab.md @@ -0,0 +1,66 @@ +Drone comes with built-in support for the GitLab version 8.2 and higher. To enable GitLab you should configure the Drone container using the following environment variables: + +```diff +version: '2' + +services: + drone-server: + image: drone/drone:{{% version %}} + ports: + - 80:8000 + - 9000 + volumes: + - /var/lib/drone:/var/lib/drone/ + restart: always + environment: ++ - DRONE_GITLAB=true ++ - DRONE_GITLAB_CLIENT=95c0282573633eb25e82 ++ - DRONE_GITLAB_SECRET=30f5064039e6b359e075 ++ - DRONE_GITLAB_URL=http://gitlab.mycompany.com + - DRONE_SECRET=${DRONE_SECRET} + + drone-agent: + image: drone/agent:{{% version %}} + restart: always + depends_on: + - drone-server + volumes: + - /var/run/docker.sock:/var/run/docker.sock + environment: + - DRONE_SERVER=drone-server:9000 + - DRONE_SECRET=${DRONE_SECRET} +``` + +# Configuration + +This is a full list of configuration options. Please note that many of these options use default configuration values that should work for the majority of installations. + +DRONE_GITLAB=true +: Set to true to enable the GitLab driver. + +DRONE_GITLAB_URL=`https://gitlab.com` +: GitLab Server address. + +DRONE_GITLAB_CLIENT +: GitLab oauth2 client id. + +DRONE_GITLAB_SECRET +: GitLab oauth2 client secret. + +DRONE_GITLAB_GIT_USERNAME +: Optional. Use a single machine account username to clone all repositories. + +DRONE_GITLAB_GIT_PASSWORD +: Optional. Use a single machine account password to clone all repositories. + +DRONE_GITLAB_SKIP_VERIFY=false +: Set to true to disable SSL verification. + +DRONE_GITLAB_PRIVATE_MODE=false +: Set to true if GitLab is running in private mode. + +# Registration + +You must register your application with GitLab in order to generate a Client and Secret. Navigate to your account settings and choose Applications from the menu, and click New Application. + +Please use `http://drone.mycompany.com/authorize` as the Authorization callback URL. Grant `api` scope to the application. diff --git a/docs/administration/lets_encrypt.md b/docs/administration/lets_encrypt.md new file mode 100644 index 000000000..f608e5f8b --- /dev/null +++ b/docs/administration/lets_encrypt.md @@ -0,0 +1,38 @@ +Drone supports automated ssl configuration and updates using let's encrypt. You can enable let's encrypt by making the following modifications to your server configuration: + +```diff +services: + drone-server: + image: drone/drone:{{% version %}} + ports: ++ - 80:80 ++ - 443:443 + - 9000:9000 + volumes: + - /var/lib/drone:/var/lib/drone/ + restart: always + environment: + - DRONE_OPEN=true + - DRONE_HOST=${DRONE_HOST} + - DRONE_GITHUB=true + - DRONE_GITHUB_CLIENT=${DRONE_GITHUB_CLIENT} + - DRONE_GITHUB_SECRET=${DRONE_GITHUB_SECRET} + - DRONE_SECRET=${DRONE_SECRET} ++ - DRONE_LETS_ENCRYPT=true +``` + +Note that Drone uses the hostname from the `DRONE_HOST` environment variable when requesting certificates. For example, if `DRONE_HOST=https://foo.com` the certificate is requested for `foo.com`. + +>Once enabled you can visit your website at both the http and the https address + +# Certificate Cache + +Drone writes the certificates to the below directory: + +``` +/var/lib/drone/golang-autocert +``` + +# Certificate Updates + +Drone uses the official Go acme library which will handle certificate upgrades. There should be no addition configuration or management required. diff --git a/docs/administration/prometheus.md b/docs/administration/prometheus.md new file mode 100644 index 000000000..4e2c2cc46 --- /dev/null +++ b/docs/administration/prometheus.md @@ -0,0 +1,162 @@ + +Drone is compatible with Prometheus and exposes a `/metrics` endpoint. Please note that access to the metrics endpoint is restricted and requires an authorization token with administrative privileges. + +```nohighlight +global: + scrape_interval: 60s + +scrape_configs: + - job_name: 'drone' + bearer_token: eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9... + + static_configs: + - targets: ['drone.domain.com'] +``` + +# Authorization + +An administrator will need to generate a user api token and configure in the prometheus configuration file as a bearer token. Please see the following example: + +```diff +global: + scrape_interval: 60s + +scrape_configs: + - job_name: 'drone' ++ bearer_token: eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9... + + static_configs: + - targets: ['drone.domain.com'] +``` + +# Metric Reference + +List of prometheus metrics specific to Drone: + +``` +# HELP drone_build_count Total number of builds. +# TYPE drone_build_count gauge +drone_build_count 7275 +# HELP drone_pending_jobs Total number of pending build processes. +# TYPE drone_pending_jobs gauge +drone_pending_jobs 0 +# HELP drone_repo_count Total number of registered repositories. +# TYPE drone_repo_count gauge +drone_repo_count 133 +# HELP drone_running_jobs Total number of running build processes. +# TYPE drone_running_jobs gauge +drone_running_jobs 0 +# HELP drone_user_count Total number of active users. +# TYPE drone_user_count gauge +drone_user_count 15 +``` + +List of prometheus metrics for server resource usage: + +``` +# HELP go_gc_duration_seconds A summary of the GC invocation durations. +# TYPE go_gc_duration_seconds summary +go_gc_duration_seconds{quantile="0"} 0.000189189 +go_gc_duration_seconds{quantile="0.25"} 0.000391444 +go_gc_duration_seconds{quantile="0.5"} 0.001895967 +go_gc_duration_seconds{quantile="0.75"} 0.003075854 +go_gc_duration_seconds{quantile="1"} 0.004224575 +go_gc_duration_seconds_sum 0.019922696 +go_gc_duration_seconds_count 10 +# HELP go_goroutines Number of goroutines that currently exist. +# TYPE go_goroutines gauge +go_goroutines 24 +# HELP go_memstats_alloc_bytes Number of bytes allocated and still in use. +# TYPE go_memstats_alloc_bytes gauge +go_memstats_alloc_bytes 2.556344e+06 +# HELP go_memstats_alloc_bytes_total Total number of bytes allocated, even if freed. +# TYPE go_memstats_alloc_bytes_total counter +go_memstats_alloc_bytes_total 2.0479656e+07 +# HELP go_memstats_buck_hash_sys_bytes Number of bytes used by the profiling bucket hash table. +# TYPE go_memstats_buck_hash_sys_bytes gauge +go_memstats_buck_hash_sys_bytes 1.45144e+06 +# HELP go_memstats_frees_total Total number of frees. +# TYPE go_memstats_frees_total counter +go_memstats_frees_total 200332 +# HELP go_memstats_gc_cpu_fraction The fraction of this program's available CPU time used by the GC since the program started. +# TYPE go_memstats_gc_cpu_fraction gauge +go_memstats_gc_cpu_fraction 8.821705133777562e-05 +# HELP go_memstats_gc_sys_bytes Number of bytes used for garbage collection system metadata. +# TYPE go_memstats_gc_sys_bytes gauge +go_memstats_gc_sys_bytes 557056 +# HELP go_memstats_heap_alloc_bytes Number of heap bytes allocated and still in use. +# TYPE go_memstats_heap_alloc_bytes gauge +go_memstats_heap_alloc_bytes 2.556344e+06 +# HELP go_memstats_heap_idle_bytes Number of heap bytes waiting to be used. +# TYPE go_memstats_heap_idle_bytes gauge +go_memstats_heap_idle_bytes 3.842048e+06 +# HELP go_memstats_heap_inuse_bytes Number of heap bytes that are in use. +# TYPE go_memstats_heap_inuse_bytes gauge +go_memstats_heap_inuse_bytes 4.972544e+06 +# HELP go_memstats_heap_objects Number of allocated objects. +# TYPE go_memstats_heap_objects gauge +go_memstats_heap_objects 19986 +# HELP go_memstats_heap_released_bytes Number of heap bytes released to OS. +# TYPE go_memstats_heap_released_bytes gauge +go_memstats_heap_released_bytes 0 +# HELP go_memstats_heap_sys_bytes Number of heap bytes obtained from system. +# TYPE go_memstats_heap_sys_bytes gauge +go_memstats_heap_sys_bytes 8.814592e+06 +# HELP go_memstats_last_gc_time_seconds Number of seconds since 1970 of last garbage collection. +# TYPE go_memstats_last_gc_time_seconds gauge +go_memstats_last_gc_time_seconds 1.4941783810383117e+09 +# HELP go_memstats_lookups_total Total number of pointer lookups. +# TYPE go_memstats_lookups_total counter +go_memstats_lookups_total 325 +# HELP go_memstats_mallocs_total Total number of mallocs. +# TYPE go_memstats_mallocs_total counter +go_memstats_mallocs_total 220318 +# HELP go_memstats_mcache_inuse_bytes Number of bytes in use by mcache structures. +# TYPE go_memstats_mcache_inuse_bytes gauge +go_memstats_mcache_inuse_bytes 2400 +# HELP go_memstats_mcache_sys_bytes Number of bytes used for mcache structures obtained from system. +# TYPE go_memstats_mcache_sys_bytes gauge +go_memstats_mcache_sys_bytes 16384 +# HELP go_memstats_mspan_inuse_bytes Number of bytes in use by mspan structures. +# TYPE go_memstats_mspan_inuse_bytes gauge +go_memstats_mspan_inuse_bytes 81016 +# HELP go_memstats_mspan_sys_bytes Number of bytes used for mspan structures obtained from system. +# TYPE go_memstats_mspan_sys_bytes gauge +go_memstats_mspan_sys_bytes 98304 +# HELP go_memstats_next_gc_bytes Number of heap bytes when next garbage collection will take place. +# TYPE go_memstats_next_gc_bytes gauge +go_memstats_next_gc_bytes 4.819216e+06 +# HELP go_memstats_other_sys_bytes Number of bytes used for other system allocations. +# TYPE go_memstats_other_sys_bytes gauge +go_memstats_other_sys_bytes 672584 +# HELP go_memstats_stack_inuse_bytes Number of bytes in use by the stack allocator. +# TYPE go_memstats_stack_inuse_bytes gauge +go_memstats_stack_inuse_bytes 622592 +# HELP go_memstats_stack_sys_bytes Number of bytes obtained from system for stack allocator. +# TYPE go_memstats_stack_sys_bytes gauge +go_memstats_stack_sys_bytes 622592 +# HELP go_memstats_sys_bytes Number of bytes obtained from system. +# TYPE go_memstats_sys_bytes gauge +go_memstats_sys_bytes 1.2232952e+07 +# HELP go_threads Number of OS threads created +# TYPE go_threads gauge +go_threads 9 +# HELP process_cpu_seconds_total Total user and system CPU time spent in seconds. +# TYPE process_cpu_seconds_total counter +process_cpu_seconds_total 0.9 +# HELP process_max_fds Maximum number of open file descriptors. +# TYPE process_max_fds gauge +process_max_fds 524288 +# HELP process_open_fds Number of open file descriptors. +# TYPE process_open_fds gauge +process_open_fds 17 +# HELP process_resident_memory_bytes Resident memory size in bytes. +# TYPE process_resident_memory_bytes gauge +process_resident_memory_bytes 2.5296896e+07 +# HELP process_start_time_seconds Start time of the process since unix epoch in seconds. +# TYPE process_start_time_seconds gauge +process_start_time_seconds 1.494177893e+09 +# HELP process_virtual_memory_bytes Virtual memory size in bytes. +# TYPE process_virtual_memory_bytes gauge +process_virtual_memory_bytes 4.23243776e+08 +``` diff --git a/docs/administration/proxy.md b/docs/administration/proxy.md new file mode 100644 index 000000000..968aa7c26 --- /dev/null +++ b/docs/administration/proxy.md @@ -0,0 +1,148 @@ +**Table of Contents** +- [Apache](#apache) +- [Nginx](#nginx) +- [Caddy](#caddy) +- [Ngrok](#ngrok) + +# Apache +This guide provides a brief overview for installing Drone server behind the Apache2 webserver. This is an example configuration: + +```nohighlight +ProxyPreserveHost On + +RequestHeader set X-Forwarded-Proto "https" + +ProxyPass / http://127.0.0.1:8000/ +ProxyPassReverse / http://127.0.0.1:8000/ +``` + +You must have the below Apache modules installed. + +```nohighlight +a2enmod proxy +a2enmod proxy_http +``` + +You must configure Apache to set `X-Forwarded-Proto` when using https. + +```diff +ProxyPreserveHost On + ++RequestHeader set X-Forwarded-Proto "https" + +ProxyPass / http://127.0.0.1:8000/ +ProxyPassReverse / http://127.0.0.1:8000/ +``` + +# Nginx + +This guide provides a basic overview for installing Drone server behind the nginx webserver. For more advanced configuration options please consult the official nginx [documentation](https://www.nginx.com/resources/admin-guide/). + +Example configuration: + +```nginx +server { + listen 80; + server_name drone.example.com; + + location / { + proxy_set_header X-Forwarded-For $remote_addr; + proxy_set_header X-Forwarded-Proto $scheme; + proxy_set_header Host $http_host; + + proxy_pass http://127.0.0.1:8000; + proxy_redirect off; + proxy_http_version 1.1; + proxy_buffering off; + + chunked_transfer_encoding off; + } +} +``` + +You must configure the proxy to set `X-Forwarded` proxy headers: + +```diff +server { + listen 80; + server_name drone.example.com; + + location / { ++ proxy_set_header X-Forwarded-For $remote_addr; ++ proxy_set_header X-Forwarded-Proto $scheme; + + proxy_pass http://127.0.0.1:8000; + proxy_redirect off; + proxy_http_version 1.1; + proxy_buffering off; + + chunked_transfer_encoding off; + } +} +``` + +# Caddy + +This guide provides a brief overview for installing Drone server behind the [Caddy webserver](https://caddyserver.com/). This is an example caddyfile proxy configuration: + +```nohighlight +drone.mycompany.com { + gzip { + not /stream/ + } + proxy / localhost:8000 { + websocket + transparent + } +} +``` +You must disable gzip compression for streamed data otherwise the live updates won't be instant: + +```diff +drone.mycompany.com { ++ gzip { ++ not /stream/ ++ } + proxy / localhost:8000 { + websocket + transparent + } +} +``` + +You must configure the proxy to enable websocket upgrades: + +```diff +drone.mycompany.com { + gzip { + not /stream/ + } + proxy / localhost:8000 { ++ websocket + transparent + } +} +``` + +You must configure the proxy to include `X-Forwarded` headers using the `transparent` directive: + +```diff +drone.mycompany.com { + gzip { + not /stream/ + } + proxy / localhost:8000 { + websocket ++ transparent + } +} +``` + +# Ngrok +After installing [ngrok](https://ngrok.com/), open a new console and run: + +``` +ngrok http 80 +``` + +Set `DRONE_HOST` (for example in `docker-compose.yml`) to the ngrok url (usually xxx.ngrok.io) and start the server. diff --git a/docs/administration/ssl.md b/docs/administration/ssl.md new file mode 100644 index 000000000..bdb7c0b73 --- /dev/null +++ b/docs/administration/ssl.md @@ -0,0 +1,77 @@ +Drone supports ssl configuration by mounting certificates into your container. + +```diff +services: + drone-server: + image: drone/drone:{{% version %}} + ports: ++ - 80:80 ++ - 443:443 + - 9000:9000 + volumes: + - /var/lib/drone:/var/lib/drone/ ++ - /etc/certs/drone.foo.com/server.crt:/etc/certs/drone.foo.com/server.crt ++ - /etc/certs/drone.foo.com/server.key:/etc/certs/drone.foo.com/server.key + restart: always + environment: ++ - DRONE_SERVER_CERT=/etc/certs/drone.foo.com/server.crt ++ - DRONE_SERVER_KEY=/etc/certs/drone.foo.com/server.key +``` + +Update your configuration to expose the following ports: + +```diff +services: + drone-server: + image: drone/drone:{{% version %}} + ports: ++ - 80:80 ++ - 443:443 + - 9000:9000 +``` + +Update your configuration to mount your certificate and key: + +```diff +services: + drone-server: + image: drone/drone:{{% version %}} + ports: + - 80:80 + - 443:443 + - 9000:9000 + volumes: + - /var/lib/drone:/var/lib/drone/ ++ - /etc/certs/drone.foo.com/server.crt:/etc/certs/drone.foo.com/server.crt ++ - /etc/certs/drone.foo.com/server.key:/etc/certs/drone.foo.com/server.key +``` + +Update your configuration to provide the paths of your certificate and key: + +```diff +services: + drone-server: + image: drone/drone:{{% version %}} + ports: + - 80:80 + - 443:443 + - 9000:9000 + volumes: + - /var/lib/drone:/var/lib/drone/ + - /etc/certs/drone.foo.com/server.crt:/etc/certs/drone.foo.com/server.crt + - /etc/certs/drone.foo.com/server.key:/etc/certs/drone.foo.com/server.key + restart: always + environment: ++ - DRONE_SERVER_CERT=/etc/certs/drone.foo.com/server.crt ++ - DRONE_SERVER_KEY=/etc/certs/drone.foo.com/server.key +``` + +# Certificate Chain + +The most common problem encountered is providing a certificate file without the intermediate chain. + +> LoadX509KeyPair reads and parses a public/private key pair from a pair of files. The files must contain PEM encoded data. The certificate file may contain intermediate certificates following the leaf certificate to form a certificate chain. + +# Certificate Errors + +SSL support is provided using the [ListenAndServeTLS](https://golang.org/pkg/net/http/#ListenAndServeTLS) function from the Go standard library. If you receive certificate errors or warnings please examine your configuration more closely. Please do not create issues claiming SSL is broken. diff --git a/docs/drone.png b/docs/drone.png new file mode 100644 index 000000000..28a162463 Binary files /dev/null and b/docs/drone.png differ diff --git a/docs/motivation.md b/docs/motivation.md new file mode 100644 index 000000000..aa3441756 --- /dev/null +++ b/docs/motivation.md @@ -0,0 +1,18 @@ +# Motivation + +I was using Drone for two years with great satisfaction. The container architecture, the speedy backend and UI, the simple plugin system made it a flexible and simple platform. Kudos for the author, Brad to make it such a joy to use. + +It wasn't without flaws +- inconsistencies in variables and CLI features +- lack of documentation +- lack of published best practices +- UI/UX issues +- stuck builds + +Things that could be circumvented by reading the codebase. Over time however these started to annoy me, also PRs that tried to address these were not merged. Instead the development of Drone headed towards a 1.0 release with features less interesting to me. + +1.0 landed and it came with a licence change. Drone has been an open-core project since many prior versions, but the enterprise features were limited to features like autoscaling and secret vaults. + +In the 1.0 line however, Postgresql, Mysql and TLS support along with agent based horizontal scaling were also moved under the enterprise license. Limiting the open source version to single node, hobbyist deployments. + +These feature reductions and my long time UX annoyance and general dissatisfaction of the CI space lead to this fork. diff --git a/docs/usage/bash_plugin.md b/docs/usage/bash_plugin.md new file mode 100644 index 000000000..635854d64 --- /dev/null +++ b/docs/usage/bash_plugin.md @@ -0,0 +1,49 @@ +This provides a brief tutorial for creating a Drone webhook plugin, using simple shell scripting, to make an http requests during the build pipeline. The below example demonstrates how we might configure a webhook plugin in the Yaml file: + +```yaml +pipeline: + webhook: + image: foo/webhook + url: http://foo.com + method: post + body: | + hello world +``` + +Create a simple shell script that invokes curl using the Yaml configuration parameters, which are passed to the script as environment variables in uppercase and prefixed with `PLUGIN_`. + +```bash +#!/bin/sh + +curl \ + -X ${PLUGIN_METHOD} \ + -d ${PLUGIN_BODY} \ + ${PLUGIN_URL} + +``` +Create a Dockerfile that adds your shell script to the image, and configures the image to execute your shell script as the main entrypoint. + +```dockerfile +FROM alpine +ADD script.sh /bin/ +RUN chmod +x /bin/script.sh +RUN apk -Uuv add curl ca-certificates +ENTRYPOINT /bin/script.sh +``` + +Build and publish your plugin to the Docker registry. Once published your plugin can be shared with the broader Drone community. + +```nohighlight +docker build -t foo/webhook . +docker push foo/webhook +``` + +Execute your plugin locally from the command line to verify it is working: + +```nohighlight +docker run --rm \ + -e PLUGIN_METHOD=post \ + -e PLUGIN_URL=http://foo.com \ + -e PLUGIN_BODY="hello world" \ + foo/webhook +``` diff --git a/docs/usage/pipeline.md b/docs/usage/pipeline.md new file mode 100644 index 000000000..e2f7c0f51 --- /dev/null +++ b/docs/usage/pipeline.md @@ -0,0 +1,1400 @@ + + + +**Table of Contents** _generated with [DocToc](https://github.com/thlorenz/doctoc)_ + +- [Pipeline basics](#pipeline-basics) + - [Activation](#activation) + - [Configuration](#configuration) + - [Execution](#execution) +- [Pipelines](#pipelines) + - [Build Steps](#build-steps) + - [Images](#images) + - [Images from private registries](#images-from-private-registries) + - [GCR Registry Support](#gcr-registry-support) + - [Parallel Execution](#parallel-execution) + - [Conditional Pipeline Execution](#conditional-pipeline-execution) + - [Conditional Step Execution](#conditional-step-execution) + - [Failure Execution](#failure-execution) +- [Services](#services) + - [Configuration](#configuration-1) + - [Detachment](#detachment) + - [Initialization](#initialization) +- [Plugins](#plugins) + - [Plugin Isolation](#plugin-isolation) + - [Plugin Marketplace](#plugin-marketplace) +- [Environment variables](#environment-variables) + - [Built-in environment variables](#built-in-environment-variables) + - [String Substitution](#string-substitution) + - [String Operations](#string-operations) +- [Secrets](#secrets) + - [Adding Secrets](#adding-secrets) + - [Alternate Names](#alternate-names) + - [Pull Requests](#pull-requests) + - [Examples](#examples) +- [Volumes](#volumes) +- [Webhooks](#webhooks) + - [Required Permissions](#required-permissions) + - [Skip Commits](#skip-commits) + - [Skip Branches](#skip-branches) +- [Workspace](#workspace) +- [Cloning](#cloning) + - [Git Submodules](#git-submodules) +- [Privileged mode](#privileged-mode) +- [Promoting](#promoting) + - [Triggering Deployments](#triggering-deployments) +- [Matrix builds](#matrix-builds) + - [Interpolation](#interpolation) + - [Examples](#examples-1) +- [Multi-pipeline builds](#multi-pipeline-builds) + - [Example multi-pipeline definition](#example-multi-pipeline-definition) + - [Flow control](#flow-control) + - [Status lines](#status-lines) + - [Rational](#rational) +- [Badges](#badges) + + + +This document explains the process for activating and configuring a continuous delivery pipeline. + +# Pipeline basics + +## Activation + +To activate your project navigate to your account settings. You will see a list of repositories which can be activated with a simple toggle. When you activate your repository, Drone automatically adds webhooks to your version control system (e.g. GitHub). + +Webhooks are used to trigger pipeline executions. When you push code to your repository, open a pull request, or create a tag, your version control system will automatically send a webhook to Drone which will in turn trigger pipeline execution. + +![repository list](`repo_list.png) + +## Configuration + +To configure you pipeline you should place a `.drone.yml` file in the root of your repository. The .drone.yml file is used to define your pipeline steps. It is a superset of the widely used docker-compose file format. + +Example pipeline configuration: + +```yaml +pipeline: + build: + image: golang + commands: + - go get + - go build + - go test + +services: + postgres: + image: postgres:9.4.5 + environment: + - POSTGRES_USER=myapp +``` + +Example pipeline configuration with multiple, serial steps: + +```yaml +pipeline: + backend: + image: golang + commands: + - go get + - go build + - go test + + frontend: + image: node:6 + commands: + - npm install + - npm test + + notify: + image: plugins/slack + channel: developers + username: drone +``` + +## Execution + +To trigger your first pipeline execution you can push code to your repository, open a pull request, or push a tag. Any of these events triggers a webhook from your version control system and execute your pipeline. + +# Pipelines + +The pipeline section defines a list of steps to build, test and deploy your code. Pipeline steps are executed serially, in the order in which they are defined. If a step returns a non-zero exit code, the pipeline immediately aborts and returns a failure status. + +Example pipeline: + +```yaml +pipeline: + backend: + image: golang + commands: + - go build + - go test + frontend: + image: node + commands: + - npm install + - npm run test + - npm run build +``` + +In the above example we define two pipeline steps, `frontend` and `backend`. The names of these steps are completely arbitrary. + +## Build Steps + +Build steps are steps in your pipeline that execute arbitrary commands inside the specified docker container. The commands are executed using the workspace as the working directory. + +```diff +pipeline: + backend: + image: golang + commands: ++ - go build ++ - go test +``` + +There is no magic here. The above commands are converted to a simple shell script. The commands in the above example are roughly converted to the below script: + +```diff +#!/bin/sh +set -e + +go build +go test +``` + +The above shell script is then executed as the docker entrypoint. The below docker command is an (incomplete) example of how the script is executed: + +``` +docker run --entrypoint=build.sh golang +``` + +> Please note that only build steps can define commands. You cannot use commands with plugins or services. + +## Images + +Drone uses Docker images for the build environment, for plugins and for service containers. The image field is exposed in the container blocks in the Yaml: + +```diff +pipeline: + build: ++ image: golang:1.6 + commands: + - go build + - go test + + publish: ++ image: plugins/docker + repo: foo/bar + +services: + database: ++ image: mysql +``` + +Drone supports any valid Docker image from any Docker registry: + +```text +image: golang +image: golang:1.7 +image: library/golang:1.7 +image: index.docker.io/library/golang +image: index.docker.io/library/golang:1.7 +``` + +Drone does not automatically upgrade docker images. Example configuration to always pull the latest image when updates are available: + +```diff +pipeline: + build: + image: golang:latest ++ pull: true +``` + +#### Images from private registries + +You must provide registry credentials on the UI in order to pull private pipeline images defined in your Yaml configuration file. + +These credentials are never exposed to your pipeline, which means they cannot be used to push, and are safe to use with pull requests, for example. Pushing to a registry still require setting credentials for the appropriate plugin. + +Example configuration using a private image: + +```diff +pipeline: + build: ++ image: gcr.io/custom/golang + commands: + - go build + - go test +``` + +Drone matches the registry hostname to each image in your yaml. If the hostnames match, the registry credentials are used to authenticate to your registry and pull the image. Note that registry credentials are used by the Drone agent and are never exposed to your build containers. + +Example registry hostnames: + +- Image `gcr.io/foo/bar` has hostname `gcr.io` +- Image `foo/bar` has hostname `docker.io` +- Image `qux.com:8000/foo/bar` has hostname `qux.com:8000` + +Example registry hostname matching logic: + +- Hostname `gcr.io` matches image `gcr.io/foo/bar` +- Hostname `docker.io` matches `golang` +- Hostname `docker.io` matches `library/golang` +- Hostname `docker.io` matches `bradyrydzewski/golang` +- Hostname `docker.io` matches `bradyrydzewski/golang:latest` + +#### GCR Registry Support + +For specific details on configuring access to Google Container Registry, please view the docs [here](https://cloud.google.com/container-registry/docs/advanced-authentication#using_a_json_key_file). + +## Parallel Execution + +Drone supports parallel step execution for same-machine fan-in and fan-out. Parallel steps are configured using the `group` attribute. This instructs the pipeline runner to execute the named group in parallel. + +Example parallel configuration: + +```diff +pipeline: + backend: ++ group: build + image: golang + commands: + - go build + - go test + frontend: ++ group: build + image: node + commands: + - npm install + - npm run test + - npm run build + publish: + image: plugins/docker + repo: octocat/hello-world +``` + +In the above example, the `frontend` and `backend` steps are executed in parallel. The pipeline runner will not execute the `publish` step until the group completes. + +## Conditional Pipeline Execution + +Drone supports defining conditional pipelines to skip commits based on the target branch. If the branch matches the `branches:` block the pipeline is executed, otherwise it is skipped. + +Example skipping a commit when the target branch is not master: + +```diff +pipeline: + build: + image: golang + commands: + - go build + - go test + ++branches: master +``` + +Example matching multiple target branches: + +```diff +pipeline: + build: + image: golang + commands: + - go build + - go test + ++branches: [ master, develop ] +``` + +Example uses glob matching: + +```diff +pipeline: + build: + image: golang + commands: + - go build + - go test + ++branches: [ master, feature/* ] +``` + +Example includes branches: + +```diff +pipeline: + build: + image: golang + commands: + - go build + - go test + ++branches: ++ include: [ master, feature/* ] +``` + +Example excludes branches: + +```diff +pipeline: + build: + image: golang + commands: + - go build + - go test + ++branches: ++ exclude: [ develop, feature/* ] +``` + +## Conditional Step Execution + +Drone supports defining conditional pipeline steps in the `when` block. If all conditions in the `when` block evaluate to true the step is executed, otherwise it is skipped. + +Example conditional execution by branch: + +```diff +pipeline: + slack: + image: plugins/slack + channel: dev ++ when: ++ branch: master +``` + +> The step now triggers on master, but also if the target branch of a pull request is `master`. Add an event condition to limit it further to pushes on master only. + +Execute a step if the branch is `master` or `develop`: + +```diff +when: + branch: [master, develop] +``` + +Execute a step if the branch starts with `prefix/*`: + +```diff +when: + branch: prefix/* +``` + +Execute a step using custom include and exclude logic: + +```diff +when: + branch: + include: [ master, release/* ] + exclude: [ release/1.0.0, release/1.1.* ] +``` + +Execute a step if the build event is a `tag`: + +```diff +when: + event: tag +``` + +Execute a step if the build event is a `tag` created from the specified branch: + +```diff +when: + event: tag ++ branch: master +``` + +Execute a step for all non-pull request events: + +```diff +when: + event: [push, tag, deployment] +``` + +Execute a step for all build events: + +```diff +when: + event: [push, pull_request, tag, deployment] +``` + +Execute a step if the tag name starts with `release`: + +```diff +when: + tag: release* +``` + +Execute a step when the build status changes: + +```diff +when: + status: changed +``` + +Execute a step when the build is passing or failing: + +```diff +when: + status: [ failure, success ] +``` + +Execute a step for a specific platform: + +```diff +when: + platform: linux/amd64 +``` + +Execute a step for a specific platform using wildcards: + +```diff +when: + platform: [ linux/*, windows/amd64 ] +``` + +Execute a step for deployment events matching the target deployment environment: + +```diff +when: + environment: production + event: deployment +``` + +Execute a step for a single matrix permutation: + +```diff +when: + matrix: + GO_VERSION: 1.5 + REDIS_VERSION: 2.8 +``` + +Execute a step only on a certain Drone instance: + +```diff +when: + instance: stage.drone.company.com +``` + +#### Failure Execution + +Drone uses the container exit code to determine the success or failure status of a build. Non-zero exit codes fail the build and cause the pipeline to immediately exit. + +There are use cases for executing pipeline steps on failure, such as sending notifications for failed builds. Use the status constraint to override the default behavior and execute steps even when the build status is failure: + +```diff +pipeline: + slack: + image: plugins/slack + channel: dev ++ when: ++ status: [ success, failure ] +``` + +# Services + +Drone provides a services section in the Yaml file used for defining service containers. The below configuration composes database and cache containers. + +```diff +pipeline: + build: + image: golang + commands: + - go build + - go test + +services: + database: + image: mysql + + cache: + image: redis +``` + +Services are accessed using custom hostnames. In the above example the mysql service is assigned the hostname `database` and is available at `database:3306`. + +## Configuration + +Service containers generally expose environment variables to customize service startup such as default usernames, passwords and ports. Please see the official image documentation to learn more. + +```diff +services: + database: + image: mysql ++ environment: ++ - MYSQL_DATABASE=test ++ - MYSQL_ALLOW_EMPTY_PASSWORD=yes + + cache: + image: redis +``` + +## Detachment + +Service and long running containers can also be included in the pipeline section of the configuration using the detach parameter without blocking other steps. This should be used when explicit control over startup order is required. + +```diff +pipeline: + build: + image: golang + commands: + - go build + - go test + + database: + image: redis ++ detach: true + + test: + image: golang + commands: + - go test +``` + +Containers from detached steps will terminate when the pipeline ends. + +## Initialization + +Service containers require time to initialize and begin to accept connections. If you are unable to connect to a service you may need to wait a few seconds or implement a backoff. + +```diff +pipeline: + test: + image: golang + commands: ++ - sleep 15 + - go get + - go test + +services: + database: + image: mysql +``` + +# Plugins + +Plugins are Docker containers that perform pre-defined tasks and are configured as steps in your pipeline. Plugins can be used to deploy code, publish artifacts, send notification, and more. + +Example pipeline using the Docker and Slack plugins: + +```yaml +pipeline: + build: + image: golang + commands: + - go build + - go test + + publish: + image: plugins/docker + repo: foo/bar + tags: latest + + notify: + image: plugins/slack + channel: dev +``` + +## Plugin Isolation + +Plugins are executed in Docker containers and are isolated from the other steps in your build pipeline. Plugins do share the build workspace, mounted as a volume, and therefore have access to your source tree. + +## Plugin Marketplace + +Plugins are packaged and distributed as Docker containers. They are conceptually similar to software libraries (think npm) and can be published and shared with the community. You can find a list of available plugins at [http://plugins.drone.io](http://plugins.drone.io). + +# Environment variables + +Drone provides the ability to define environment variables scoped to individual build steps. Example pipeline step with custom environment variables: + +```diff +pipeline: + build: + image: golang ++ environment: ++ - CGO=0 ++ - GOOS=linux ++ - GOARCH=amd64 + commands: + - go build + - go test +``` + +Please note that the environment section is not able to expand environment variables. If you need to expand variables they should be exported in the commands section. + +```diff +pipeline: + build: + image: golang +- environment: +- - PATH=$PATH:/go + commands: ++ - export PATH=$PATH:/go + - go build + - go test +``` + +Please be warned that `${variable}` expressions are subject to pre-processing. If you do not want the pre-processor to evaluate your expression it must be escaped: + +```diff +pipeline: + build: + image: golang + commands: +- - export PATH=${PATH}:/go ++ - export PATH=$${PATH}:/go + - go build + - go test +``` + +## Built-in environment variables + +This is the reference list of all environment variables available to your build environment. These are injected into your build and plugins containers, at runtime. + +| NAME | DESC | +| ---------------------------- | -------------------------------------- | +| `CI=drone` | environment is drone | +| `DRONE=true` | environment is drone | +| `DRONE_ARCH` | environment architecture (linux/amd64) | +| `DRONE_REPO` | repository full name | +| `DRONE_REPO_OWNER` | repository owner | +| `DRONE_REPO_NAME` | repository name | +| `DRONE_REPO_SCM` | repository scm (git) | +| `DRONE_REPO_LINK` | repository link | +| `DRONE_REPO_AVATAR` | repository avatar | +| `DRONE_REPO_BRANCH` | repository default branch (master) | +| `DRONE_REPO_PRIVATE` | repository is private | +| `DRONE_REPO_TRUSTED` | repository is trusted | +| `DRONE_REMOTE_URL` | repository clone url | +| `DRONE_COMMIT_SHA` | commit sha | +| `DRONE_COMMIT_REF` | commit ref | +| `DRONE_COMMIT_BRANCH` | commit branch | +| `DRONE_COMMIT_LINK` | commit link in remote | +| `DRONE_COMMIT_MESSAGE` | commit message | +| `DRONE_COMMIT_AUTHOR` | commit author username | +| `DRONE_COMMIT_AUTHOR_EMAIL` | commit author email address | +| `DRONE_COMMIT_AUTHOR_AVATAR` | commit author avatar | +| `DRONE_BUILD_NUMBER` | build number | +| `DRONE_BUILD_EVENT` | build event (push, pull_request, tag) | +| `DRONE_BUILD_STATUS` | build status (success, failure) | +| `DRONE_BUILD_LINK` | build result link | +| `DRONE_BUILD_CREATED` | build created unix timestamp | +| `DRONE_BUILD_STARTED` | build started unix timestamp | +| `DRONE_BUILD_FINISHED` | build finished unix timestamp | +| `DRONE_PREV_BUILD_STATUS` | prior build status | +| `DRONE_PREV_BUILD_NUMBER` | prior build number | +| `DRONE_PREV_COMMIT_SHA` | prior build commit sha | +| `DRONE_JOB_NUMBER` | job number | +| `DRONE_JOB_STATUS` | job status | +| `DRONE_JOB_STARTED` | job started | +| `DRONE_JOB_FINISHED` | job finished | +| `DRONE_BRANCH` | commit branch | +| `DRONE_COMMIT` | commit sha | +| `DRONE_TAG` | commit tag | +| `DRONE_PULL_REQUEST` | pull request number | +| `DRONE_DEPLOY_TO` | deployment target (ie production) | + +## String Substitution + +Drone provides the ability to substitute environment variables at runtime. This gives us the ability to use dynamic build or commit details in our pipeline configuration. + +Example commit substitution: + +```diff +pipeline: + docker: + image: plugins/docker ++ tags: ${DRONE_COMMIT_SHA} +``` + +Example tag substitution: + +```diff +pipeline: + docker: + image: plugins/docker ++ tags: ${DRONE_TAG} +``` + +## String Operations + +Drone also emulates bash string operations. This gives us the ability to manipulate the strings prior to substitution. Example use cases might include substring and stripping prefix or suffix values. + +| OPERATION | DESC | +| ------------------ | ------------------------------------------------ | +| `${param}` | parameter substitution | +| `${param,}` | parameter substitution with lowercase first char | +| `${param,,}` | parameter substitution with lowercase | +| `${param^}` | parameter substitution with uppercase first char | +| `${param^^}` | parameter substitution with uppercase | +| `${param:pos}` | parameter substitution with substring | +| `${param:pos:len}` | parameter substitution with substring and length | +| `${param=default}` | parameter substitution with default | +| `${param##prefix}` | parameter substitution with prefix removal | +| `${param%%suffix}` | parameter substitution with suffix removal | +| `${param/old/new}` | parameter substitution with find and replace | + +Example variable substitution with substring: + +```diff +pipeline: + docker: + image: plugins/docker ++ tags: ${DRONE_COMMIT_SHA:0:8} +``` + +Example variable substitution strips `v` prefix from `v.1.0.0`: + +```diff +pipeline: + docker: + image: plugins/docker ++ tags: ${DRONE_TAG##v} +``` + +# Secrets + +Drone provides the ability to store named parameters external to the Yaml configuration file, in a central secret store. Individual steps in the yaml can request access to these named parameters at runtime. + +Secrets are exposed to your pipeline steps and plugins as uppercase environment variables and can therefore be referenced in the commands section of your pipeline. + +```diff +pipeline: + docker: + image: docker + commands: ++ - echo $DOCKER_USERNAME ++ - echo $DOCKER_PASSWORD + secrets: [ docker_username, docker_password ] +``` + +Please note parameter expressions are subject to pre-processing. When using secrets in parameter expressions they should be escaped. + +```diff +pipeline: + docker: + image: docker + commands: +- - echo ${DOCKER_USERNAME} +- - echo ${DOCKER_PASSWORD} ++ - echo $${DOCKER_USERNAME} ++ - echo $${DOCKER_PASSWORD} + secrets: [ docker_username, docker_password ] +``` + +## Adding Secrets + +Secrets are added to the Drone secret store on the UI or with the CLI. + +## Alternate Names + +There may be scenarios where you are required to store secrets using alternate names. You can map the alternate secret name to the expected name using the below syntax: + +```diff +pipeline: + docker: + image: plugins/docker + repo: octocat/hello-world + tags: latest ++ secrets: ++ - source: docker_prod_password ++ target: docker_password +``` + +## Pull Requests + +Secrets are not exposed to pull requests by default. You can override this behavior by creating the secret and enabling the `pull_request` event type. + +```diff +drone secret add \ + -repository octocat/hello-world \ + -image plugins/docker \ ++ -event pull_request \ ++ -event push \ ++ -event tag \ + -name docker_username \ + -value +``` + +Please be careful when exposing secrets to pull requests. If your repository is open source and accepts pull requests your secrets are not safe. A bad actor can submit a malicious pull request that exposes your secrets. + +## Examples + +Create the secret using default settings. The secret will be available to all images in your pipeline, and will be available to all push, tag, and deployment events (not pull request events). + +```diff +drone secret add \ + -repository octocat/hello-world \ + -name aws_access_key_id \ + -value +``` + +Create the secret and limit to a single image: + +```diff +drone secret add \ + -repository octocat/hello-world \ ++ -image plugins/s3 \ + -name aws_access_key_id \ + -value +``` + +Create the secrets and limit to a set of images: + +```diff +drone secret add \ + -repository octocat/hello-world \ ++ -image plugins/s3 \ ++ -image peloton/drone-ecs \ + -name aws_access_key_id \ + -value +``` + +Create the secret and enable for multiple hook events: + +```diff +drone secret add \ + -repository octocat/hello-world \ + -image plugins/s3 \ ++ -event pull_request \ ++ -event push \ ++ -event tag \ + -name aws_access_key_id \ + -value +``` + +Loading secrets from file using curl `@` syntax. This is the recommended approach for loading secrets from file to preserve newlines: + +```diff +drone secret add \ + -repository octocat/hello-world \ + -name ssh_key \ ++ -value @/root/ssh/id_rsa +``` + +# Volumes + +Drone gives the ability to define Docker volumes in the Yaml. You can use this parameter to mount files or folders on the host machine into your containers. + +> Volumes are only available to trusted repositories and for security reasons should only be used in private environments. + +```diff +pipeline: + build: + image: docker + commands: + - docker build --rm -t octocat/hello-world . + - docker run --rm octocat/hello-world --test + - docker push octocat/hello-world + - docker rmi octocat/hello-world + volumes: ++ - /var/run/docker.sock:/var/run/docker.sock +``` + +Please note that Drone mounts volumes on the host machine. This means you must use absolute paths when you configure volumes. Attempting to use relative paths will result in an error. + +```diff +- volumes: [ ./certs:/etc/ssl/certs ] ++ volumes: [ /etc/ssl/certs:/etc/ssl/certs ] +``` + +# Webhooks + +When you activate your repository Drone automatically add webhooks to your version control system (e.g. GitHub). There is no manual configuration required. + +Webhooks are used to trigger pipeline executions. When you push code to your repository, open a pull request, or create a tag, your version control system will automatically send a webhook to Drone which will in turn trigger pipeline execution. + +## Required Permissions + +The user who enables a repo in Drone must have `Admin` rights on that repo, so that Drone can add the webhook. + +Note that manually creating webhooks yourself is not possible. This is because webhooks are signed using a per-repository secret key which is not exposed to end users. + +## Skip Commits + +Drone gives the ability to skip individual commits by adding `[CI SKIP]` to the commit message. Note this is case-insensitive. + +```diff +git commit -m "updated README [CI SKIP]" +``` + +## Skip Branches + +Drone gives the ability to skip commits based on the target branch. The below example will skip a commit when the target branch is not master. + +```diff +pipeline: + build: + image: golang + commands: + - go build + - go test + ++branches: master +``` + +Please see the pipeline conditions [documentation]({{< ref "usage/config/pipeline-conditions.md" >}}) for more options and details. + +# Workspace + +The workspace defines the shared volume and working directory shared by all pipeline steps. The default workspace matches the below pattern, based on your repository url. + +``` +/drone/src/github.com/octocat/hello-world +``` + +The workspace can be customized using the workspace block in the Yaml file: + +```diff ++workspace: ++ base: /go ++ path: src/github.com/octocat/hello-world + +pipeline: + build: + image: golang:latest + commands: + - go get + - go test +``` + +The base attribute defines a shared base volume available to all pipeline steps. This ensures your source code, dependencies and compiled binaries are persisted and shared between steps. + +```diff +workspace: ++ base: /go + path: src/github.com/octocat/hello-world + +pipeline: + deps: + image: golang:latest + commands: + - go get + - go test + build: + image: node:latest + commands: + - go build +``` + +This would be equivalent to the following docker commands: + +``` +docker volume create my-named-volume + +docker run --volume=my-named-volume:/go golang:latest +docker run --volume=my-named-volume:/go node:latest +``` + +The path attribute defines the working directory of your build. This is where your code is cloned and will be the default working directory of every step in your build process. The path must be relative and is combined with your base path. + +```diff +workspace: + base: /go ++ path: src/github.com/octocat/hello-world +``` + +```text +git clone https://github.com/octocat/hello-world \ + /go/src/github.com/octocat/hello-world +``` + +# Cloning + +Drone automatically configures a default clone step if not explicitly defined. You can manually configure the clone step in your pipeline for customization: + +```diff ++clone: ++ git: ++ image: plugins/git + +pipeline: + build: + image: golang + commands: + - go build + - go test +``` + +Example configuration to override depth: + +```diff +clone: + git: + image: plugins/git ++ depth: 50 +``` + +Example configuration to use a custom clone plugin: + +```diff +clone: + git: ++ image: octocat/custom-git-plugin +``` + +Example configuration to clone Mercurial repository: + +```diff +clone: + hg: ++ image: plugins/hg ++ path: bitbucket.org/foo/bar +``` + +## Git Submodules + +To use the credentials that cloned the repository to clone it's submodules, update `.gitmodules` to use `https` instead of `git`: + +```diff +[submodule "my-module"] + path = my-module +- url = git@github.com:octocat/my-module.git ++ url = https://github.com/octocat/my-module.git +``` + +To use the ssh git url in `.gitmodules` for users cloning with ssh, and also use the https url in drone, add `submodule_override`: + +```diff +clone: + git: + image: plugins/git + recursive: true ++ submodule_override: ++ my-module: https://github.com/octocat/my-module.git + +pipeline: + ... +``` + +# Privileged mode + +Drone gives the ability to configure privileged mode in the Yaml. You can use this parameter to launch containers with escalated capabilities. + +> Privileged mode is only available to trusted repositories and for security reasons should only be used in private environments. + +```diff +pipeline: + build: + image: docker + environment: + - DOCKER_HOST=tcp://docker:2375 + commands: + - docker --tls=false ps + +services: + docker: + image: docker:dind + command: [ "--storage-driver=vfs", "--tls=false" ] ++ privileged: true +``` + +# Promoting + +Drone provides the ability to promote individual commits or tags (e.g. promote to production). When you promote a commit or tag it triggers a new pipeline execution with event type `deployment`. You can use the event type and target environment to limit step execution. + +```diff +pipeline: + build: + image: golang + commands: + - go build + - go test + + publish: + image: plugins/docker + registry: registry.heroku.com + repo: registry.heroku.com/my-staging-app/web + when: ++ event: deployment ++ environment: staging + + publish_to_prod: + image: plugins/docker + registry: registry.heroku.com + repo: registry.heroku.com/my-production-app/web + when: ++ event: deployment ++ environment: production +``` + +The above example demonstrates how we can configure pipeline steps to only execute when the deployment matches a specific target environment. + +## Triggering Deployments + +Deployments are triggered from the command line utility. They are triggered from an existing build. This is conceptually similar to promoting builds. + +```text +drone deploy +``` + +Promote the specified build number to your staging environment: + +```text +drone deploy octocat/hello-world 24 staging +``` + +Promote the specified build number to your production environment: + +```text +drone deploy octocat/hello-world 24 production +``` + +# Matrix builds + +Drone has integrated support for matrix builds. Drone executes a separate build task for each combination in the matrix, allowing you to build and test a single commit against multiple configurations. + +Example matrix definition: + +```yaml +matrix: + GO_VERSION: + - 1.4 + - 1.3 + REDIS_VERSION: + - 2.6 + - 2.8 + - 3.0 +``` + +Example matrix definition containing only specific combinations: + +```yaml +matrix: + include: + - GO_VERSION: 1.4 + REDIS_VERSION: 2.8 + - GO_VERSION: 1.5 + REDIS_VERSION: 2.8 + - GO_VERSION: 1.6 + REDIS_VERSION: 3.0 +``` + +## Interpolation + +Matrix variables are interpolated in the yaml using the `${VARIABLE}` syntax, before the yaml is parsed. This is an example yaml file before interpolating matrix parameters: + +```yaml +pipeline: + build: + image: golang:${GO_VERSION} + commands: + - go get + - go build + - go test + +services: + database: + image: ${DATABASE} + +matrix: + GO_VERSION: + - 1.4 + - 1.3 + DATABASE: + - mysql:5.5 + - mysql:6.5 + - mariadb:10.1 +``` + +Example Yaml file after injecting the matrix parameters: + +```diff +pipeline: + build: +- image: golang:${GO_VERSION} ++ image: golang:1.4 + commands: + - go get + - go build + - go test ++ environment: ++ - GO_VERSION=1.4 ++ - DATABASE=mysql:5.5 + +services: + database: +- image: ${DATABASE} ++ image: mysql:5.5 +``` + +## Examples + +Example matrix build based on Docker image tag: + +```yaml +pipeline: + build: + image: golang:${TAG} + commands: + - go build + - go test + +matrix: + TAG: + - 1.7 + - 1.8 + - latest +``` + +Example matrix build based on Docker image: + +```yaml +pipeline: + build: + image: ${IMAGE} + commands: + - go build + - go test + +matrix: + IMAGE: + - golang:1.7 + - golang:1.8 + - golang:latest +``` + +# Multi-pipeline builds + +By default, Drone looks for the pipeline definition in `.drone.yml` in the project root. + +The Multi-Pipeline feature allows the pipeline to be splitted to several files and placed in the `.drone/` folder + +## Example multi-pipeline definition + +```bash +.drone +├── .build.yml +├── .deploy.yml +├── .lint.yml +└── .test.yml +``` + +.drone/.build.yml + +```yaml +pipeline: + build: + image: debian:stable-slim + commands: + - echo building + - sleep 5 +``` + +.drone/.deploy.yml + +```yaml +pipeline: + deploy: + image: debian:stable-slim + commands: + - echo deploying + +depends_on: + - lint + - build + - test +``` + +.drone/.test.yml + +```yaml +pipeline: + test: + image: debian:stable-slim + commands: + - echo testing + - sleep 5 + +depends_on: + - build +``` + +.drone/.lint.yml + +```yaml +pipeline: + lint: + image: debian:stable-slim + commands: + - echo linting + - sleep 5 +``` + +## Flow control + +The pipelines run in parallel on a separate agents and share nothing. + +Dependencies between pipelines can be set with the `depends_on` element. A pipeline doesn't execute until its dependencies did not complete succesfully. + +```diff +pipeline: + deploy: + image: debian:stable-slim + commands: + - echo deploying + ++depends_on: ++ - lint ++ - build ++ - test +``` + +Pipelines that need to run even on failures should set the `run_on` tag. + +```diff +pipeline: + notify: + image: debian:stable-slim + commands: + - echo notifying + +depends_on: + - deploy + ++run_on: [ success, failure ] +``` + +Some pipelines don't need the source code, set the `skip_clone` tag to skip cloning: + +```diff + +pipeline: + notify: + image: debian:stable-slim + commands: + - echo notifying + +depends_on: + - deploy + +run_on: [ success, failure ] ++skip_clone: true +``` + +## Status lines + +Each pipeline has its own status line on Github. + +## Rational + +- faster lint/test feedback, the pipeline doesn't have to run fully to have a lint status pushed to the the remote +- better organization of the pipeline along various concerns: testing, linting, feature apps +- utilizaing more agents to speed up build + +# Badges + +Drone has integrated support for repository status badges. These badges can be added to your website or project readme file to display the status of your code. + +Badge endpoint: + +```text +:///api/badges///status.svg +``` + +The status badge displays the status for the latest build to your default branch (e.g. master). You can customize the branch by adding the `branch` query parameter. + +```diff +-:///api/badges///status.svg ++:///api/badges///status.svg?branch= +``` + +Please note status badges do not include pull request results, since the status of a pull request does not provide an accurate representation of your repository state. diff --git a/docs/usage/repo_list.png b/docs/usage/repo_list.png new file mode 100644 index 000000000..368c961ea Binary files /dev/null and b/docs/usage/repo_list.png differ diff --git a/vendor/github.com/drone/drone-go/LICENSE b/drone-go/LICENSE similarity index 100% rename from vendor/github.com/drone/drone-go/LICENSE rename to drone-go/LICENSE diff --git a/vendor/github.com/drone/drone-go/README.md b/drone-go/README.md similarity index 92% rename from vendor/github.com/drone/drone-go/README.md rename to drone-go/README.md index 263ac2f7c..5975e458f 100644 --- a/vendor/github.com/drone/drone-go/README.md +++ b/drone-go/README.md @@ -2,7 +2,7 @@ ```Go import ( - "github.com/drone/drone-go/drone" + "github.com/laszlocph/drone-oss-08/drone-go/drone" "golang.org/x/oauth2" ) diff --git a/vendor/github.com/drone/drone-go/drone/client.go b/drone-go/drone/client.go similarity index 100% rename from vendor/github.com/drone/drone-go/drone/client.go rename to drone-go/drone/client.go diff --git a/vendor/github.com/drone/drone-go/drone/const.go b/drone-go/drone/const.go similarity index 100% rename from vendor/github.com/drone/drone-go/drone/const.go rename to drone-go/drone/const.go diff --git a/vendor/github.com/drone/drone-go/drone/interface.go b/drone-go/drone/interface.go similarity index 100% rename from vendor/github.com/drone/drone-go/drone/interface.go rename to drone-go/drone/interface.go diff --git a/vendor/github.com/drone/drone-go/drone/types.go b/drone-go/drone/types.go similarity index 100% rename from vendor/github.com/drone/drone-go/drone/types.go rename to drone-go/drone/types.go diff --git a/model/config.go b/model/config.go index 8eae6e34c..05aa587ac 100644 --- a/model/config.go +++ b/model/config.go @@ -16,10 +16,11 @@ package model // ConfigStore persists pipeline configuration to storage. type ConfigStore interface { - ConfigLoad(int64) (*Config, error) - ConfigFind(*Repo, string) (*Config, error) + ConfigsForBuild(buildID int64) ([]*Config, error) + ConfigFindIdentical(repoID int64, sha string) (*Config, error) ConfigFindApproved(*Config) (bool, error) ConfigCreate(*Config) error + BuildConfigCreate(*BuildConfig) error } // Config represents a pipeline configuration. @@ -28,4 +29,11 @@ type Config struct { RepoID int64 `json:"-" meddler:"config_repo_id"` Data string `json:"data" meddler:"config_data"` Hash string `json:"hash" meddler:"config_hash"` + Name string `json:"name" meddler:"config_name"` +} + +// BuildConfig is the n:n relation between Build and Config +type BuildConfig struct { + ConfigID int64 `json:"-" meddler:"config_id"` + BuildID int64 `json:"-" meddler:"build_id"` } diff --git a/model/proc.go b/model/proc.go index c547725e6..8740c0737 100644 --- a/model/proc.go +++ b/model/proc.go @@ -14,6 +14,8 @@ package model +import "fmt" + // ProcStore persists process information to storage. type ProcStore interface { ProcLoad(int64) (*Proc, error) @@ -57,18 +59,24 @@ func (p *Proc) Failing() bool { // Tree creates a process tree from a flat process list. func Tree(procs []*Proc) []*Proc { - var ( - nodes []*Proc - parent *Proc - ) + var nodes []*Proc for _, proc := range procs { if proc.PPID == 0 { nodes = append(nodes, proc) - parent = proc - continue } else { + parent, _ := findNode(nodes, proc.PPID) parent.Children = append(parent.Children, proc) } } return nodes } + +func findNode(nodes []*Proc, pid int) (*Proc, error) { + for _, node := range nodes { + if node.PID == pid { + return node, nil + } + } + + return nil, fmt.Errorf("Corrupt proc structure") +} diff --git a/model/queue.go b/model/queue.go index 040728ae3..3b5952687 100644 --- a/model/queue.go +++ b/model/queue.go @@ -23,9 +23,11 @@ import ( // Task defines scheduled pipeline Task. type Task struct { - ID string `meddler:"task_id"` - Data []byte `meddler:"task_data"` - Labels map[string]string `meddler:"task_labels,json"` + ID string `meddler:"task_id"` + Data []byte `meddler:"task_data"` + Labels map[string]string `meddler:"task_labels,json"` + Dependencies []string `meddler:"task_dependencies,json"` + RunOn []string `meddler:"task_run_on,json"` } // TaskStore defines storage for scheduled Tasks. @@ -39,13 +41,18 @@ type TaskStore interface { // ensures the task Queue can be restored when the system starts. func WithTaskStore(q queue.Queue, s TaskStore) queue.Queue { tasks, _ := s.TaskList() + toEnqueue := []*queue.Task{} for _, task := range tasks { - q.Push(context.Background(), &queue.Task{ - ID: task.ID, - Data: task.Data, - Labels: task.Labels, + toEnqueue = append(toEnqueue, &queue.Task{ + ID: task.ID, + Data: task.Data, + Labels: task.Labels, + Dependencies: task.Dependencies, + RunOn: task.RunOn, + DepStatus: make(map[string]bool), }) } + q.PushAtOnce(context.Background(), toEnqueue) return &persistentQueue{q, s} } @@ -54,12 +61,14 @@ type persistentQueue struct { store TaskStore } -// Push pushes an task to the tail of this queue. +// Push pushes a task to the tail of this queue. func (q *persistentQueue) Push(c context.Context, task *queue.Task) error { q.store.TaskInsert(&Task{ - ID: task.ID, - Data: task.Data, - Labels: task.Labels, + ID: task.ID, + Data: task.Data, + Labels: task.Labels, + Dependencies: task.Dependencies, + RunOn: task.RunOn, }) err := q.Queue.Push(c, task) if err != nil { @@ -68,6 +77,26 @@ func (q *persistentQueue) Push(c context.Context, task *queue.Task) error { return err } +// Push pushes multiple tasks to the tail of this queue. +func (q *persistentQueue) PushAtOnce(c context.Context, tasks []*queue.Task) error { + for _, task := range tasks { + q.store.TaskInsert(&Task{ + ID: task.ID, + Data: task.Data, + Labels: task.Labels, + Dependencies: task.Dependencies, + RunOn: task.RunOn, + }) + } + err := q.Queue.PushAtOnce(c, tasks) + if err != nil { + for _, task := range tasks { + q.store.TaskDelete(task.ID) + } + } + return err +} + // Poll retrieves and removes a task head of this queue. func (q *persistentQueue) Poll(c context.Context, f queue.Filter) (*queue.Task, error) { task, err := q.Queue.Poll(c, f) diff --git a/model/repo.go b/model/repo.go index 40a523931..c02e4a4ff 100644 --- a/model/repo.go +++ b/model/repo.go @@ -55,6 +55,7 @@ type Repo struct { Config string `json:"config_file" meddler:"repo_config_path"` Hash string `json:"-" meddler:"repo_hash"` Perm *Perm `json:"-" meddler:"-"` + Fallback bool `json:"fallback" meddler:"repo_fallback"` } func (r *Repo) ResetVisibility() { @@ -105,4 +106,5 @@ type RepoPatch struct { AllowDeploy *bool `json:"allow_deploy,omitempty"` AllowTag *bool `json:"allow_tag,omitempty"` BuildCounter *int `json:"build_counter,omitempty"` + Fallback *bool `json:"fallback,omitempty"` } diff --git a/remote/bitbucket/bitbucket.go b/remote/bitbucket/bitbucket.go index 9d4f3f793..19c337ea3 100644 --- a/remote/bitbucket/bitbucket.go +++ b/remote/bitbucket/bitbucket.go @@ -202,20 +202,19 @@ func (c *config) Perm(u *model.User, owner, name string) (*model.Perm, error) { // File fetches the file from the Bitbucket repository and returns its contents. func (c *config) File(u *model.User, r *model.Repo, b *model.Build, f string) ([]byte, error) { - return c.FileRef(u, r, b.Commit, f) -} - -// FileRef fetches the file from the Bitbucket repository and returns its contents. -func (c *config) FileRef(u *model.User, r *model.Repo, ref, f string) ([]byte, error) { - config, err := c.newClient(u).FindSource(r.Owner, r.Name, ref, f) + config, err := c.newClient(u).FindSource(r.Owner, r.Name, b.Commit, f) if err != nil { return nil, err } return []byte(*config), err } +func (c *config) Dir(u *model.User, r *model.Repo, b *model.Build, f string) ([]*remote.FileMeta, error) { + return nil, fmt.Errorf("Not implemented") +} + // Status creates a build status for the Bitbucket commit. -func (c *config) Status(u *model.User, r *model.Repo, b *model.Build, link string) error { +func (c *config) Status(u *model.User, r *model.Repo, b *model.Build, link string, proc *model.Proc) error { status := internal.BuildStatus{ State: convertStatus(b.Status), Desc: convertDesc(b.Status), diff --git a/remote/bitbucket/bitbucket_test.go b/remote/bitbucket/bitbucket_test.go index 3c9af60b2..854007c9e 100644 --- a/remote/bitbucket/bitbucket_test.go +++ b/remote/bitbucket/bitbucket_test.go @@ -283,7 +283,7 @@ func Test_bitbucket(t *testing.T) { }) g.It("Should update the status", func() { - err := c.Status(fakeUser, fakeRepo, fakeBuild, "http://127.0.0.1") + err := c.Status(fakeUser, fakeRepo, fakeBuild, "http://127.0.0.1", nil) g.Assert(err == nil).IsTrue() }) diff --git a/remote/bitbucketserver/bitbucketserver.go b/remote/bitbucketserver/bitbucketserver.go index 3fe2cd3d2..e0b747327 100644 --- a/remote/bitbucketserver/bitbucketserver.go +++ b/remote/bitbucketserver/bitbucketserver.go @@ -179,14 +179,12 @@ func (c *Config) File(u *model.User, r *model.Repo, b *model.Build, f string) ([ return client.FindFileForRepo(r.Owner, r.Name, f, b.Ref) } -func (c *Config) FileRef(u *model.User, r *model.Repo, ref, f string) ([]byte, error) { - client := internal.NewClientWithToken(c.URL, c.Consumer, u.Token) - - return client.FindFileForRepo(r.Owner, r.Name, f, ref) +func (c *Config) Dir(u *model.User, r *model.Repo, b *model.Build, f string) ([]*remote.FileMeta, error) { + return nil, fmt.Errorf("Not implemented") } // Status is not supported by the bitbucketserver driver. -func (c *Config) Status(u *model.User, r *model.Repo, b *model.Build, link string) error { +func (c *Config) Status(u *model.User, r *model.Repo, b *model.Build, link string, proc *model.Proc) error { status := internal.BuildStatus{ State: convertStatus(b.Status), Desc: convertDesc(b.Status), diff --git a/remote/coding/coding.go b/remote/coding/coding.go index 115f94498..7255b2448 100644 --- a/remote/coding/coding.go +++ b/remote/coding/coding.go @@ -238,18 +238,12 @@ func (c *Coding) File(u *model.User, r *model.Repo, b *model.Build, f string) ([ return data, nil } -// FileRef fetches a file from the remote repository for the given ref -// and returns in string format. -func (c *Coding) FileRef(u *model.User, r *model.Repo, ref, f string) ([]byte, error) { - data, err := c.newClient(u).GetFile(r.Owner, r.Name, ref, f) - if err != nil { - return nil, err - } - return data, nil +func (c *Coding) Dir(u *model.User, r *model.Repo, b *model.Build, f string) ([]*remote.FileMeta, error) { + return nil, fmt.Errorf("Not implemented") } // Status sends the commit status to the remote system. -func (c *Coding) Status(u *model.User, r *model.Repo, b *model.Build, link string) error { +func (c *Coding) Status(u *model.User, r *model.Repo, b *model.Build, link string, proc *model.Proc) error { // EMPTY: not implemented in Coding OAuth API return nil } diff --git a/remote/coding/coding_test.go b/remote/coding/coding_test.go index bf17180cf..f24f34d33 100644 --- a/remote/coding/coding_test.go +++ b/remote/coding/coding_test.go @@ -184,11 +184,6 @@ func Test_coding(t *testing.T) { g.Assert(err == nil).IsTrue() g.Assert(string(data)).Equal("pipeline:\n test:\n image: golang:1.6\n commands:\n - go test\n") }) - g.It("Should return file for specified ref", func() { - data, err := c.FileRef(fakeUser, fakeRepo, "master", ".drone.yml") - g.Assert(err == nil).IsTrue() - g.Assert(string(data)).Equal("pipeline:\n test:\n image: golang:1.6\n commands:\n - go test\n") - }) }) g.Describe("When requesting a netrc config", func() { diff --git a/remote/gerrit/gerrit.go b/remote/gerrit/gerrit.go index f0fc91da5..f5fd526c3 100644 --- a/remote/gerrit/gerrit.go +++ b/remote/gerrit/gerrit.go @@ -103,13 +103,12 @@ func (c *client) File(u *model.User, r *model.Repo, b *model.Build, f string) ([ return nil, nil } -// File is not supported by the Gerrit driver. -func (c *client) FileRef(u *model.User, r *model.Repo, ref, f string) ([]byte, error) { - return nil, nil +func (c *client) Dir(u *model.User, r *model.Repo, b *model.Build, f string) ([]*remote.FileMeta, error) { + return nil, fmt.Errorf("Not implemented") } // Status is not supported by the Gogs driver. -func (c *client) Status(u *model.User, r *model.Repo, b *model.Build, link string) error { +func (c *client) Status(u *model.User, r *model.Repo, b *model.Build, link string, proc *model.Proc) error { return nil } diff --git a/remote/gitea/gitea.go b/remote/gitea/gitea.go index f1cfe4c60..14b4bdb54 100644 --- a/remote/gitea/gitea.go +++ b/remote/gitea/gitea.go @@ -249,13 +249,12 @@ func (c *client) File(u *model.User, r *model.Repo, b *model.Build, f string) ([ return cfg, err } -// FileRef fetches the file from the Gitea repository and returns its contents. -func (c *client) FileRef(u *model.User, r *model.Repo, ref, f string) ([]byte, error) { - return c.newClientToken(u.Token).GetFile(r.Owner, r.Name, ref, f) +func (c *client) Dir(u *model.User, r *model.Repo, b *model.Build, f string) ([]*remote.FileMeta, error) { + return nil, fmt.Errorf("Not implemented") } // Status is supported by the Gitea driver. -func (c *client) Status(u *model.User, r *model.Repo, b *model.Build, link string) error { +func (c *client) Status(u *model.User, r *model.Repo, b *model.Build, link string, proc *model.Proc) error { client := c.newClientToken(u.Token) status := getStatus(b.Status) diff --git a/remote/gitea/gitea_test.go b/remote/gitea/gitea_test.go index a6607bdf2..ab0933c2c 100644 --- a/remote/gitea/gitea_test.go +++ b/remote/gitea/gitea_test.go @@ -18,10 +18,10 @@ import ( "net/http/httptest" "testing" - "github.com/laszlocph/drone-oss-08/model" - "github.com/laszlocph/drone-oss-08/remote/gitea/fixtures" "github.com/franela/goblin" "github.com/gin-gonic/gin" + "github.com/laszlocph/drone-oss-08/model" + "github.com/laszlocph/drone-oss-08/remote/gitea/fixtures" ) func Test_gitea(t *testing.T) { @@ -149,7 +149,7 @@ func Test_gitea(t *testing.T) { }) g.It("Should return nil from send build status", func() { - err := c.Status(fakeUser, fakeRepo, fakeBuild, "http://gitea.io") + err := c.Status(fakeUser, fakeRepo, fakeBuild, "http://gitea.io", nil) g.Assert(err == nil).IsTrue() }) diff --git a/remote/github/convert.go b/remote/github/convert.go index b8ce64f65..4fde67e4c 100644 --- a/remote/github/convert.go +++ b/remote/github/convert.go @@ -51,7 +51,7 @@ const ( // GitHub commit status. func convertStatus(status string) string { switch status { - case model.StatusPending, model.StatusRunning, model.StatusBlocked: + case model.StatusPending, model.StatusRunning, model.StatusBlocked, model.StatusSkipped: return statusPending case model.StatusFailure, model.StatusDeclined: return statusFailure diff --git a/remote/github/github.go b/remote/github/github.go index 5c142e71b..55b2cc12e 100644 --- a/remote/github/github.go +++ b/remote/github/github.go @@ -23,6 +23,7 @@ import ( "regexp" "strconv" "strings" + "sync" "github.com/laszlocph/drone-oss-08/model" "github.com/laszlocph/drone-oss-08/remote" @@ -104,7 +105,7 @@ type client struct { // Login authenticates the session and returns the remote user details. func (c *client) Login(res http.ResponseWriter, req *http.Request) (*model.User, error) { - config := c.newConfig(httputil.GetURL(req)) + config := c.newConfig(req) // get the OAuth errors if err := req.FormValue("error"); err != "" { @@ -225,22 +226,77 @@ func (c *client) Perm(u *model.User, owner, name string) (*model.Perm, error) { // File fetches the file from the GitHub repository and returns its contents. func (c *client) File(u *model.User, r *model.Repo, b *model.Build, f string) ([]byte, error) { - return c.FileRef(u, r, b.Commit, f) -} - -// FileRef fetches the file from the GitHub repository and returns its contents. -func (c *client) FileRef(u *model.User, r *model.Repo, ref, f string) ([]byte, error) { client := c.newClientToken(u.Token) opts := new(github.RepositoryContentGetOptions) - opts.Ref = ref + opts.Ref = b.Commit data, _, _, err := client.Repositories.GetContents(r.Owner, r.Name, f, opts) if err != nil { return nil, err } + if data == nil { + return nil, fmt.Errorf("%s is a folder not a file use Dir(..)", f) + } return data.Decode() } +func (c *client) Dir(u *model.User, r *model.Repo, b *model.Build, f string) ([]*remote.FileMeta, error) { + client := c.newClientToken(u.Token) + + opts := new(github.RepositoryContentGetOptions) + opts.Ref = b.Commit + _, data, _, err := client.Repositories.GetContents(r.Owner, r.Name, f, opts) + if err != nil { + return nil, err + } + + fc := make(chan *remote.FileMeta) + errc := make(chan error) + + wg := &sync.WaitGroup{} + wg.Add(len(data)) + + for _, file := range data { + go func(path string) { + content, err := c.File(u, r, b, path) + if err != nil { + errc <- err + } else { + fc <- &remote.FileMeta{ + Name: path, + Data: content, + } + } + }(f + "/" + *file.Name) + } + + var files []*remote.FileMeta + var errors []error + + go func() { + for { + select { + case err, open := <-errc: + if open { + errors = append(errors, err) + wg.Done() + } + case fileMeta, open := <-fc: + if open { + files = append(files, fileMeta) + wg.Done() + } + } + } + }() + + wg.Wait() + close(fc) + close(errc) + + return files, nil +} + // Netrc returns a netrc file capable of authenticating GitHub requests and // cloning GitHub repositories. The netrc will use the global machine account // when configured. @@ -292,7 +348,16 @@ func (c *client) newContext() context.Context { } // helper function to return the GitHub oauth2 config -func (c *client) newConfig(redirect string) *oauth2.Config { +func (c *client) newConfig(req *http.Request) *oauth2.Config { + var redirect string + + intendedURL := req.URL.Query()["url"] + if len(intendedURL) > 0 { + redirect = fmt.Sprintf("%s/authorize?url=%s", httputil.GetURL(req), intendedURL[0]) + } else { + redirect = fmt.Sprintf("%s/authorize", httputil.GetURL(req)) + } + return &oauth2.Config{ ClientID: c.Client, ClientSecret: c.Secret, @@ -301,7 +366,7 @@ func (c *client) newConfig(redirect string) *oauth2.Config { AuthURL: fmt.Sprintf("%s/login/oauth/authorize", c.URL), TokenURL: fmt.Sprintf("%s/login/oauth/access_token", c.URL), }, - RedirectURL: fmt.Sprintf("%s/authorize", redirect), + RedirectURL: redirect, } } @@ -374,17 +439,17 @@ func matchingHooks(hooks []github.Hook, rawurl string) *github.Hook { // Status sends the commit status to the remote system. // An example would be the GitHub pull request status. -func (c *client) Status(u *model.User, r *model.Repo, b *model.Build, link string) error { +func (c *client) Status(u *model.User, r *model.Repo, b *model.Build, link string, proc *model.Proc) error { client := c.newClientToken(u.Token) switch b.Event { case "deployment": return deploymentStatus(client, r, b, link) default: - return repoStatus(client, r, b, link, c.Context) + return repoStatus(client, r, b, link, c.Context, proc) } } -func repoStatus(client *github.Client, r *model.Repo, b *model.Build, link, ctx string) error { +func repoStatus(client *github.Client, r *model.Repo, b *model.Build, link, ctx string, proc *model.Proc) error { context := ctx switch b.Event { case model.EventPull: @@ -395,10 +460,19 @@ func repoStatus(client *github.Client, r *model.Repo, b *model.Build, link, ctx } } + status := github.String(convertStatus(b.Status)) + desc := github.String(convertDesc(b.Status)) + + if proc != nil { + context += "/" + proc.Name + status = github.String(convertStatus(proc.State)) + desc = github.String(convertDesc(proc.State)) + } + data := github.RepoStatus{ Context: github.String(context), - State: github.String(convertStatus(b.Status)), - Description: github.String(convertDesc(b.Status)), + State: status, + Description: desc, TargetURL: github.String(link), } _, _, err := client.Repositories.CreateStatus(r.Owner, r.Name, b.Commit, &data) diff --git a/remote/gitlab/gitlab.go b/remote/gitlab/gitlab.go index 318c83428..910433ea8 100644 --- a/remote/gitlab/gitlab.go +++ b/remote/gitlab/gitlab.go @@ -325,28 +325,27 @@ func (g *Gitlab) Perm(u *model.User, owner, name string) (*model.Perm, error) { // File fetches a file from the remote repository and returns in string format. func (g *Gitlab) File(user *model.User, repo *model.Repo, build *model.Build, f string) ([]byte, error) { - return g.FileRef(user, repo, build.Commit, f) -} - -// FileRef fetches the file from the GitHub repository and returns its contents. -func (g *Gitlab) FileRef(u *model.User, r *model.Repo, ref, f string) ([]byte, error) { - var client = NewClient(g.URL, u.Token, g.SkipVerify) - id, err := GetProjectId(g, client, r.Owner, r.Name) + var client = NewClient(g.URL, user.Token, g.SkipVerify) + id, err := GetProjectId(g, client, repo.Owner, repo.Name) if err != nil { return nil, err } - out, err := client.RepoRawFileRef(id, ref, f) + out, err := client.RepoRawFileRef(id, build.Commit, f) if err != nil { return nil, err } return out, err } +func (c *Gitlab) Dir(u *model.User, r *model.Repo, b *model.Build, f string) ([]*remote.FileMeta, error) { + return nil, fmt.Errorf("Not implemented") +} + // NOTE Currently gitlab doesn't support status for commits and events, // also if we want get MR status in gitlab we need implement a special plugin for gitlab, // gitlab uses API to fetch build status on client side. But for now we skip this. -func (g *Gitlab) Status(u *model.User, repo *model.Repo, b *model.Build, link string) error { +func (g *Gitlab) Status(u *model.User, repo *model.Repo, b *model.Build, link string, proc *model.Proc) error { client := NewClient(g.URL, u.Token, g.SkipVerify) status := getStatus(b.Status) diff --git a/remote/gitlab3/gitlab.go b/remote/gitlab3/gitlab.go index cd7eb1f25..73aafdbdf 100644 --- a/remote/gitlab3/gitlab.go +++ b/remote/gitlab3/gitlab.go @@ -338,25 +338,14 @@ func (g *Gitlab) File(user *model.User, repo *model.Repo, build *model.Build, f return out, err } -// FileRef fetches the file from the GitHub repository and returns its contents. -func (g *Gitlab) FileRef(u *model.User, r *model.Repo, ref, f string) ([]byte, error) { - var client = NewClient(g.URL, u.Token, g.SkipVerify) - id, err := GetProjectId(g, client, r.Owner, r.Name) - if err != nil { - return nil, err - } - - out, err := client.RepoRawFileRef(id, ref, f) - if err != nil { - return nil, err - } - return out, err +func (c *Gitlab) Dir(u *model.User, r *model.Repo, b *model.Build, f string) ([]*remote.FileMeta, error) { + return nil, fmt.Errorf("Not implemented") } // NOTE Currently gitlab doesn't support status for commits and events, // also if we want get MR status in gitlab we need implement a special plugin for gitlab, // gitlab uses API to fetch build status on client side. But for now we skip this. -func (g *Gitlab) Status(u *model.User, repo *model.Repo, b *model.Build, link string) error { +func (g *Gitlab) Status(u *model.User, repo *model.Repo, b *model.Build, link string, proc *model.Proc) error { client := NewClient(g.URL, u.Token, g.SkipVerify) status := getStatus(b.Status) diff --git a/remote/gogs/gogs.go b/remote/gogs/gogs.go index 285c80795..7b641384e 100644 --- a/remote/gogs/gogs.go +++ b/remote/gogs/gogs.go @@ -22,9 +22,9 @@ import ( "net/url" "strings" + "github.com/gogits/go-gogs-client" "github.com/laszlocph/drone-oss-08/model" "github.com/laszlocph/drone-oss-08/remote" - "github.com/gogits/go-gogs-client" ) // Opts defines configuration options. @@ -202,13 +202,12 @@ func (c *client) File(u *model.User, r *model.Repo, b *model.Build, f string) ([ return cfg, err } -// FileRef fetches the file from the Gogs repository and returns its contents. -func (c *client) FileRef(u *model.User, r *model.Repo, ref, f string) ([]byte, error) { - return c.newClientToken(u.Token).GetFile(r.Owner, r.Name, ref, f) +func (c *client) Dir(u *model.User, r *model.Repo, b *model.Build, f string) ([]*remote.FileMeta, error) { + return nil, fmt.Errorf("Not implemented") } // Status is not supported by the Gogs driver. -func (c *client) Status(u *model.User, r *model.Repo, b *model.Build, link string) error { +func (c *client) Status(u *model.User, r *model.Repo, b *model.Build, link string, proc *model.Proc) error { return nil } diff --git a/remote/gogs/gogs_test.go b/remote/gogs/gogs_test.go index 961eae460..20f847646 100644 --- a/remote/gogs/gogs_test.go +++ b/remote/gogs/gogs_test.go @@ -163,7 +163,7 @@ func Test_gogs(t *testing.T) { g.It("Should return no-op for usupporeted features", func() { _, err1 := c.Auth("octocat", "4vyW6b49Z") - err2 := c.Status(nil, nil, nil, "") + err2 := c.Status(nil, nil, nil, "", nil) err3 := c.Deactivate(nil, nil, "") g.Assert(err1 != nil).IsTrue() g.Assert(err2 == nil).IsTrue() diff --git a/remote/mock/remote.go b/remote/mock/remote.go index 96f5fba16..096a25fad 100644 --- a/remote/mock/remote.go +++ b/remote/mock/remote.go @@ -1,25 +1,11 @@ -// Copyright 2018 Drone.IO Inc. -// -// 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. +// Code generated by mockery v1.0.0. DO NOT EDIT. -package mock +package mocks -import ( - "net/http" - - "github.com/laszlocph/drone-oss-08/model" - "github.com/stretchr/testify/mock" -) +import http "net/http" +import mock "github.com/stretchr/testify/mock" +import model "github.com/laszlocph/drone-oss-08/model" +import remote "github.com/laszlocph/drone-oss-08/remote" // Remote is an autogenerated mock type for the Remote type type Remote struct { @@ -75,6 +61,29 @@ func (_m *Remote) Deactivate(u *model.User, r *model.Repo, link string) error { return r0 } +// Dir provides a mock function with given fields: u, r, b, f +func (_m *Remote) Dir(u *model.User, r *model.Repo, b *model.Build, f string) ([]*remote.FileMeta, error) { + ret := _m.Called(u, r, b, f) + + var r0 []*remote.FileMeta + if rf, ok := ret.Get(0).(func(*model.User, *model.Repo, *model.Build, string) []*remote.FileMeta); ok { + r0 = rf(u, r, b, f) + } else { + if ret.Get(0) != nil { + r0 = ret.Get(0).([]*remote.FileMeta) + } + } + + var r1 error + if rf, ok := ret.Get(1).(func(*model.User, *model.Repo, *model.Build, string) error); ok { + r1 = rf(u, r, b, f) + } else { + r1 = ret.Error(1) + } + + return r0, r1 +} + // File provides a mock function with given fields: u, r, b, f func (_m *Remote) File(u *model.User, r *model.Repo, b *model.Build, f string) ([]byte, error) { ret := _m.Called(u, r, b, f) @@ -98,29 +107,6 @@ func (_m *Remote) File(u *model.User, r *model.Repo, b *model.Build, f string) ( return r0, r1 } -// FileRef provides a mock function with given fields: u, r, ref, f -func (_m *Remote) FileRef(u *model.User, r *model.Repo, ref string, f string) ([]byte, error) { - ret := _m.Called(u, r, ref, f) - - var r0 []byte - if rf, ok := ret.Get(0).(func(*model.User, *model.Repo, string, string) []byte); ok { - r0 = rf(u, r, ref, f) - } else { - if ret.Get(0) != nil { - r0 = ret.Get(0).([]byte) - } - } - - var r1 error - if rf, ok := ret.Get(1).(func(*model.User, *model.Repo, string, string) error); ok { - r1 = rf(u, r, ref, f) - } else { - r1 = ret.Error(1) - } - - return r0, r1 -} - // Hook provides a mock function with given fields: r func (_m *Remote) Hook(r *http.Request) (*model.Repo, *model.Build, error) { ret := _m.Called(r) @@ -246,15 +232,15 @@ func (_m *Remote) Repo(u *model.User, owner string, repo string) (*model.Repo, e } // Repos provides a mock function with given fields: u -func (_m *Remote) Repos(u *model.User) ([]*model.RepoLite, error) { +func (_m *Remote) Repos(u *model.User) ([]*model.Repo, error) { ret := _m.Called(u) - var r0 []*model.RepoLite - if rf, ok := ret.Get(0).(func(*model.User) []*model.RepoLite); ok { + var r0 []*model.Repo + if rf, ok := ret.Get(0).(func(*model.User) []*model.Repo); ok { r0 = rf(u) } else { if ret.Get(0) != nil { - r0 = ret.Get(0).([]*model.RepoLite) + r0 = ret.Get(0).([]*model.Repo) } } @@ -282,29 +268,6 @@ func (_m *Remote) Status(u *model.User, r *model.Repo, b *model.Build, link stri return r0 } -// TeamPerm provides a mock function with given fields: u, org -func (_m *Remote) TeamPerm(u *model.User, org string) (*model.Perm, error) { - ret := _m.Called(u, org) - - var r0 *model.Perm - if rf, ok := ret.Get(0).(func(*model.User, string) *model.Perm); ok { - r0 = rf(u, org) - } else { - if ret.Get(0) != nil { - r0 = ret.Get(0).(*model.Perm) - } - } - - var r1 error - if rf, ok := ret.Get(1).(func(*model.User, string) error); ok { - r1 = rf(u, org) - } else { - r1 = ret.Error(1) - } - - return r0, r1 -} - // Teams provides a mock function with given fields: u func (_m *Remote) Teams(u *model.User) ([]*model.Team, error) { ret := _m.Called(u) diff --git a/remote/remote.go b/remote/remote.go index e6dd43339..558aae222 100644 --- a/remote/remote.go +++ b/remote/remote.go @@ -18,7 +18,6 @@ package remote import ( "net/http" - "time" "github.com/laszlocph/drone-oss-08/model" @@ -51,13 +50,12 @@ type Remote interface { // format. File(u *model.User, r *model.Repo, b *model.Build, f string) ([]byte, error) - // FileRef fetches a file from the remote repository for the given ref - // and returns in string format. - FileRef(u *model.User, r *model.Repo, ref, f string) ([]byte, error) + // Dir fetches a folder from the remote repository + Dir(u *model.User, r *model.Repo, b *model.Build, f string) ([]*FileMeta, error) // Status sends the commit status to the remote system. // An example would be the GitHub pull request status. - Status(u *model.User, r *model.Repo, b *model.Build, link string) error + Status(u *model.User, r *model.Repo, b *model.Build, link string, proc *model.Proc) error // Netrc returns a .netrc file that can be used to clone // private repositories from a remote system. @@ -75,6 +73,18 @@ type Remote interface { Hook(r *http.Request) (*model.Repo, *model.Build, error) } +// FileMeta represents a file in version control +type FileMeta struct { + Name string + Data []byte +} + +type ByName []*FileMeta + +func (a ByName) Len() int { return len(a) } +func (a ByName) Less(i, j int) bool { return a[i].Name < a[j].Name } +func (a ByName) Swap(i, j int) { a[i], a[j] = a[j], a[i] } + // Refresher refreshes an oauth token and expiration for the given user. It // returns true if the token was refreshed, false if the token was not refreshed, // and error if it failed to refersh. @@ -115,22 +125,10 @@ func Perm(c context.Context, u *model.User, owner, repo string) (*model.Perm, er return FromContext(c).Perm(u, owner, repo) } -// File fetches a file from the remote repository and returns in string format. -func File(c context.Context, u *model.User, r *model.Repo, b *model.Build, f string) (out []byte, err error) { - for i := 0; i < 12; i++ { - out, err = FromContext(c).File(u, r, b, f) - if err == nil { - return - } - time.Sleep(5 * time.Second) - } - return -} - // Status sends the commit status to the remote system. // An example would be the GitHub pull request status. -func Status(c context.Context, u *model.User, r *model.Repo, b *model.Build, link string) error { - return FromContext(c).Status(u, r, b, link) +func Status(c context.Context, u *model.User, r *model.Repo, b *model.Build, link string, proc *model.Proc) error { + return FromContext(c).Status(u, r, b, link, proc) } // Netrc returns a .netrc file that can be used to clone @@ -168,18 +166,3 @@ func Refresh(c context.Context, u *model.User) (bool, error) { } return refresher.Refresh(u) } - -// FileBackoff fetches the file using an exponential backoff. -// TODO replace this with a proper backoff -func FileBackoff(remote Remote, u *model.User, r *model.Repo, b *model.Build, f string) (out []byte, err error) { - for i := 0; i < 5; i++ { - select { - case <-time.After(time.Second * time.Duration(i)): - out, err = remote.File(u, r, b, f) - if err == nil { - return - } - } - } - return -} diff --git a/router/router.go b/router/router.go index e9956c709..2fa37b3b2 100644 --- a/router/router.go +++ b/router/router.go @@ -148,6 +148,22 @@ func Load(mux *httptreemux.ContextMux, middleware ...gin.HandlerFunc) http.Handl ) } + queue := e.Group("/api/queue") + { + queue.GET("/pause", + session.MustAdmin(), + server.PauseQueue, + ) + queue.GET("/resume", + session.MustAdmin(), + server.ResumeQueue, + ) + queue.GET("/norunningbuilds", + session.MustAdmin(), + server.BlockTilQueueHasRunningItem, + ) + } + auth := e.Group("/authorize") { auth.GET("", server.HandleAuth) diff --git a/server/build.go b/server/build.go index 33161c3c6..b94a7e119 100644 --- a/server/build.go +++ b/server/build.go @@ -156,13 +156,12 @@ func GetProcLogs(c *gin.Context) { io.Copy(c.Writer, rc) } +// DeleteBuild cancels a build func DeleteBuild(c *gin.Context) { repo := session.Repo(c) - // parse the build number and job sequence number from - // the repquest parameter. + // parse the build number from the request parameter. num, _ := strconv.Atoi(c.Params.ByName("number")) - seq, _ := strconv.Atoi(c.Params.ByName("job")) build, err := store.GetBuildNumber(c, repo, num) if err != nil { @@ -170,27 +169,40 @@ func DeleteBuild(c *gin.Context) { return } - proc, err := store.FromContext(c).ProcFind(build, seq) + procs, err := store.FromContext(c).ProcList(build) if err != nil { c.AbortWithError(404, err) return } - if proc.State != model.StatusRunning { + cancelled := false + for _, proc := range procs { + if proc.PPID != 0 { + continue + } + + if proc.State != model.StatusRunning && proc.State != model.StatusPending { + continue + } + + proc.State = model.StatusKilled + proc.Stopped = time.Now().Unix() + if proc.Started == 0 { + proc.Started = proc.Stopped + } + proc.ExitCode = 137 + // TODO cancel child procs + store.FromContext(c).ProcUpdate(proc) + + Config.Services.Queue.Error(context.Background(), fmt.Sprint(proc.ID), queue.ErrCancel) + cancelled = true + } + + if !cancelled { c.String(400, "Cannot cancel a non-running build") return } - proc.State = model.StatusKilled - proc.Stopped = time.Now().Unix() - if proc.Started == 0 { - proc.Started = proc.Stopped - } - proc.ExitCode = 137 - // TODO cancel child procs - store.FromContext(c).ProcUpdate(proc) - - Config.Services.Queue.Error(context.Background(), fmt.Sprint(proc.ID), queue.ErrCancel) c.String(204, "") } @@ -268,7 +280,7 @@ func PostApproval(c *gin.Context) { build.Reviewer = user.Login // fetch the build file from the database - conf, err := Config.Storage.Config.ConfigLoad(build.ConfigID) + configs, err := Config.Storage.Config.ConfigsForBuild(build.ID) if err != nil { logrus.Errorf("failure to get build config for %s. %s", repo.FullName, err) c.AbortWithError(404, err) @@ -307,13 +319,10 @@ func PostApproval(c *gin.Context) { } } - defer func() { - uri := fmt.Sprintf("%s/%s/%d", httputil.GetURL(c.Request), repo.FullName, build.Number) - err = remote_.Status(user, repo, build, uri) - if err != nil { - logrus.Errorf("error setting commit status for %s/%d: %v", repo.FullName, build.Number, err) - } - }() + var yamls []*remote.FileMeta + for _, y := range configs { + yamls = append(yamls, &remote.FileMeta{Data: []byte(y.Data), Name: y.Name}) + } b := procBuilder{ Repo: repo, @@ -323,7 +332,7 @@ func PostApproval(c *gin.Context) { Secs: secs, Regs: regs, Link: httputil.GetURL(c.Request), - Yaml: conf.Data, + Yamls: yamls, Envs: envs, } buildItems, err := b.Build() @@ -336,12 +345,25 @@ func PostApproval(c *gin.Context) { return } - setBuildProcs(build, buildItems) err = store.FromContext(c).ProcCreate(build.Procs) if err != nil { logrus.Errorf("error persisting procs %s/%d: %s", repo.FullName, build.Number, err) } + defer func() { + for _, item := range buildItems { + uri := fmt.Sprintf("%s/%s/%d", httputil.GetURL(c.Request), repo.FullName, build.Number) + if len(buildItems) > 1 { + err = remote_.Status(user, repo, build, uri, item.Proc) + } else { + err = remote_.Status(user, repo, build, uri, nil) + } + if err != nil { + logrus.Errorf("error setting commit status for %s/%d: %v", repo.FullName, build.Number, err) + } + } + }() + publishToTopic(c, build, repo) queueBuild(build, repo, buildItems) } @@ -376,7 +398,7 @@ func PostDecline(c *gin.Context) { } uri := fmt.Sprintf("%s/%s/%d", httputil.GetURL(c.Request), repo.FullName, build.Number) - err = remote_.Status(user, repo, build, uri) + err = remote_.Status(user, repo, build, uri, nil) if err != nil { logrus.Errorf("error setting commit status for %s/%d: %v", repo.FullName, build.Number, err) } @@ -436,7 +458,7 @@ func PostBuild(c *gin.Context) { } // fetch the .drone.yml file from the database - conf, err := Config.Storage.Config.ConfigLoad(build.ConfigID) + configs, err := Config.Storage.Config.ConfigsForBuild(build.ID) if err != nil { logrus.Errorf("failure to get build config for %s. %s", repo.FullName, err) c.AbortWithError(404, err) @@ -474,6 +496,13 @@ func PostBuild(c *gin.Context) { return } + err = persistBuildConfigs(configs, build.ID) + if err != nil { + logrus.Errorf("failure to persist build config for %s. %s", repo.FullName, err) + c.AbortWithError(500, err) + return + } + // Read query string parameters into buildParams, exclude reserved params var buildParams = map[string]string{} for key, val := range c.Request.URL.Query() { @@ -504,6 +533,11 @@ func PostBuild(c *gin.Context) { } } + var yamls []*remote.FileMeta + for _, y := range configs { + yamls = append(yamls, &remote.FileMeta{Data: []byte(y.Data), Name: y.Name}) + } + b := procBuilder{ Repo: repo, Curr: build, @@ -512,7 +546,7 @@ func PostBuild(c *gin.Context) { Secs: secs, Regs: regs, Link: httputil.GetURL(c.Request), - Yaml: conf.Data, + Yamls: yamls, Envs: buildParams, } buildItems, err := b.Build() @@ -525,8 +559,6 @@ func PostBuild(c *gin.Context) { return } - setBuildProcs(build, buildItems) - err = store.FromContext(c).ProcCreate(build.Procs) if err != nil { logrus.Errorf("cannot restart %s#%d: %s", repo.FullName, build.Number, err) @@ -582,6 +614,20 @@ func DeleteBuildLogs(c *gin.Context) { c.String(204, "") } +func persistBuildConfigs(configs []*model.Config, buildID int64) error { + for _, conf := range configs { + buildConfig := &model.BuildConfig{ + ConfigID: conf.ID, + BuildID: buildID, + } + err := Config.Storage.Config.BuildConfigCreate(buildConfig) + if err != nil { + return err + } + } + return nil +} + var deleteStr = `[ { "proc": %q, diff --git a/server/configFetcher.go b/server/configFetcher.go new file mode 100644 index 000000000..b4977ff06 --- /dev/null +++ b/server/configFetcher.go @@ -0,0 +1,53 @@ +package server + +import ( + "strings" + "time" + + "github.com/laszlocph/drone-oss-08/model" + "github.com/laszlocph/drone-oss-08/remote" +) + +type configFetcher struct { + remote_ remote.Remote + user *model.User + repo *model.Repo + build *model.Build +} + +func (cf *configFetcher) Fetch() ([]*remote.FileMeta, error) { + for i := 0; i < 5; i++ { + select { + case <-time.After(time.Second * time.Duration(i)): + // either a file + file, fileerr := cf.remote_.File(cf.user, cf.repo, cf.build, cf.repo.Config) + if fileerr == nil { + return []*remote.FileMeta{&remote.FileMeta{ + Name: cf.repo.Config, + Data: file, + }}, nil + } + + // or a folder + dir, direrr := cf.remote_.Dir(cf.user, cf.repo, cf.build, strings.TrimSuffix(cf.repo.Config, "/")) + + if direrr == nil { + return dir, nil + } else if !cf.repo.Fallback { + return nil, direrr + } + + // or fallback + file, fileerr = cf.remote_.File(cf.user, cf.repo, cf.build, ".drone.yml") + if fileerr != nil { + return nil, fileerr + } + + return []*remote.FileMeta{&remote.FileMeta{ + Name: cf.repo.Config, + Data: file, + }}, nil + } + } + return []*remote.FileMeta{}, nil +} diff --git a/server/configFetcher_test.go b/server/configFetcher_test.go new file mode 100644 index 000000000..1ac3f6825 --- /dev/null +++ b/server/configFetcher_test.go @@ -0,0 +1,22 @@ +package server + +import ( + "testing" + + "github.com/laszlocph/drone-oss-08/model" + "github.com/laszlocph/drone-oss-08/remote/github" +) + +func TestFetchGithub(t *testing.T) { + github, err := github.New(github.Opts{URL: "https://github.com"}) + if err != nil { + t.Fatal(err) + } + configFetcher := &configFetcher{ + remote_: github, + user: &model.User{Token: "xxx"}, + repo: &model.Repo{Owner: "laszlocph", Name: "drone-multipipeline", Config: ".drone"}, + build: &model.Build{Commit: "89ab7b2d6bfb347144ac7c557e638ab402848fee"}, + } + configFetcher.Fetch() +} diff --git a/server/hook.go b/server/hook.go index 7604f22d0..0becc690c 100644 --- a/server/hook.go +++ b/server/hook.go @@ -20,6 +20,7 @@ import ( "encoding/json" "fmt" "math/rand" + "net/http" "regexp" "strconv" "time" @@ -51,6 +52,26 @@ func GetQueueInfo(c *gin.Context) { ) } +func PauseQueue(c *gin.Context) { + Config.Services.Queue.Pause() + c.Status(http.StatusOK) +} + +func ResumeQueue(c *gin.Context) { + Config.Services.Queue.Resume() + c.Status(http.StatusOK) +} + +func BlockTilQueueHasRunningItem(c *gin.Context) { + for { + info := Config.Services.Queue.Info(c) + if info.Stats.Running == 0 { + break + } + } + c.Status(http.StatusOK) +} + func PostHook(c *gin.Context) { remote_ := remote.FromContext(c) @@ -143,34 +164,21 @@ func PostHook(c *gin.Context) { } // fetch the build file from the remote - remoteYamlConfig, err := remote.FileBackoff(remote_, user, repo, build, repo.Config) + configFetcher := &configFetcher{remote_: remote_, user: user, repo: repo, build: build} + remoteYamlConfigs, err := configFetcher.Fetch() if err != nil { logrus.Errorf("error: %s: cannot find %s in %s: %s", repo.FullName, repo.Config, build.Ref, err) c.AbortWithError(404, err) return } - conf, err := findOrPersistPipelineConfig(repo, remoteYamlConfig) - if err != nil { - logrus.Errorf("failure to find or persist build config for %s. %s", repo.FullName, err) - c.AbortWithError(500, err) + + if branchFiltered(build, remoteYamlConfigs) { + c.String(200, "Branch does not match restrictions defined in yaml") return } - build.ConfigID = conf.ID - // verify that pipeline can be built at all - parsedPipelineConfig, err := yaml.ParseString(conf.Data) - if err == nil { - if !parsedPipelineConfig.Branches.Match(build.Branch) && build.Event != model.EventTag && build.Event != model.EventDeploy { - c.String(200, "Branch does not match restrictions defined in yaml") - return - } - } - - if repo.IsGated { - allowed, _ := Config.Services.Senders.SenderAllowed(user, repo, build, conf) - if !allowed { - build.Status = model.StatusBlocked - } + if repo.IsGated { // This feature is not clear to me. Reenabling once better understood + build.Status = model.StatusBlocked } // update some build fields @@ -185,6 +193,16 @@ func PostHook(c *gin.Context) { return } + // persist the build config for historical correctness, restarts, etc + for _, remoteYamlConfig := range remoteYamlConfigs { + _, err := findOrPersistPipelineConfig(build, remoteYamlConfig) + if err != nil { + logrus.Errorf("failure to find or persist build config for %s. %s", repo.FullName, err) + c.AbortWithError(500, err) + return + } + } + c.JSON(200, build) if build.Status == model.StatusBlocked { @@ -218,14 +236,6 @@ func PostHook(c *gin.Context) { // get the previous build so that we can send status change notifications last, _ := store.GetBuildLastBefore(c, repo, build.Branch, build.ID) - defer func() { - uri := fmt.Sprintf("%s/%s/%d", httputil.GetURL(c.Request), repo.FullName, build.Number) - err = remote_.Status(user, repo, build, uri) - if err != nil { - logrus.Errorf("error setting commit status for %s/%d: %v", repo.FullName, build.Number, err) - } - }() - b := procBuilder{ Repo: repo, Curr: build, @@ -235,7 +245,7 @@ func PostHook(c *gin.Context) { Regs: regs, Envs: envs, Link: httputil.GetURL(c.Request), - Yaml: conf.Data, + Yamls: remoteYamlConfigs, } buildItems, err := b.Build() if err != nil { @@ -247,66 +257,75 @@ func PostHook(c *gin.Context) { return } - setBuildProcs(build, buildItems) - err = store.FromContext(c).ProcCreate(build.Procs) if err != nil { logrus.Errorf("error persisting procs %s/%d: %s", repo.FullName, build.Number, err) } + defer func() { + for _, item := range buildItems { + uri := fmt.Sprintf("%s/%s/%d", httputil.GetURL(c.Request), repo.FullName, build.Number) + if len(buildItems) > 1 { + err = remote_.Status(user, repo, build, uri, item.Proc) + } else { + err = remote_.Status(user, repo, build, uri, nil) + } + if err != nil { + logrus.Errorf("error setting commit status for %s/%d: %v", repo.FullName, build.Number, err) + } + } + }() + publishToTopic(c, build, repo) queueBuild(build, repo, buildItems) } -func findOrPersistPipelineConfig(repo *model.Repo, remoteYamlConfig []byte) (*model.Config, error) { - sha := shasum(remoteYamlConfig) - conf, err := Config.Storage.Config.ConfigFind(repo, sha) +func branchFiltered(build *model.Build, remoteYamlConfigs []*remote.FileMeta) bool { + for _, remoteYamlConfig := range remoteYamlConfigs { + parsedPipelineConfig, err := yaml.ParseString(string(remoteYamlConfig.Data)) + if err == nil { + if !parsedPipelineConfig.Branches.Match(build.Branch) && build.Event != model.EventTag && build.Event != model.EventDeploy { + } else { + return false + } + } + } + return true +} + +func findOrPersistPipelineConfig(build *model.Build, remoteYamlConfig *remote.FileMeta) (*model.Config, error) { + sha := shasum(remoteYamlConfig.Data) + conf, err := Config.Storage.Config.ConfigFindIdentical(build.RepoID, sha) if err != nil { conf = &model.Config{ - RepoID: repo.ID, - Data: string(remoteYamlConfig), + RepoID: build.RepoID, + Data: string(remoteYamlConfig.Data), Hash: sha, + Name: sanitizePath(remoteYamlConfig.Name), } err = Config.Storage.Config.ConfigCreate(conf) if err != nil { // retry in case we receive two hooks at the same time - conf, err = Config.Storage.Config.ConfigFind(repo, sha) + conf, err = Config.Storage.Config.ConfigFindIdentical(build.RepoID, sha) if err != nil { return nil, err } } } + buildConfig := &model.BuildConfig{ + ConfigID: conf.ID, + BuildID: build.ID, + } + err = Config.Storage.Config.BuildConfigCreate(buildConfig) + if err != nil { + return nil, err + } + return conf, nil } -func setBuildProcs(build *model.Build, buildItems []*buildItem) { - pcounter := len(buildItems) - for _, item := range buildItems { - build.Procs = append(build.Procs, item.Proc) - item.Proc.BuildID = build.ID - - for _, stage := range item.Config.Stages { - var gid int - for _, step := range stage.Steps { - pcounter++ - if gid == 0 { - gid = pcounter - } - proc := &model.Proc{ - BuildID: build.ID, - Name: step.Alias, - PID: pcounter, - PPID: item.Proc.PID, - PGID: gid, - State: model.StatusPending, - } - build.Procs = append(build.Procs, proc) - } - } - } -} - +// publishes message to UI clients func publishToTopic(c *gin.Context, build *model.Build, repo *model.Repo) { message := pubsub.Message{ Labels: map[string]string{ @@ -325,7 +344,11 @@ func publishToTopic(c *gin.Context, build *model.Build, repo *model.Repo) { } func queueBuild(build *model.Build, repo *model.Repo, buildItems []*buildItem) { + var tasks []*queue.Task for _, item := range buildItems { + if item.Proc.State == model.StatusSkipped { + continue + } task := new(queue.Task) task.ID = fmt.Sprint(item.Proc.ID) task.Labels = map[string]string{} @@ -334,6 +357,9 @@ func queueBuild(build *model.Build, repo *model.Repo, buildItems []*buildItem) { } task.Labels["platform"] = item.Platform task.Labels["repo"] = repo.FullName + task.Dependencies = taskIds(item.DependsOn, buildItems) + task.RunOn = item.RunsOn + task.DepStatus = make(map[string]bool) task.Data, _ = json.Marshal(rpc.Pipeline{ ID: fmt.Sprint(item.Proc.ID), @@ -342,8 +368,21 @@ func queueBuild(build *model.Build, repo *model.Repo, buildItems []*buildItem) { }) Config.Services.Logs.Open(context.Background(), task.ID) - Config.Services.Queue.Push(context.Background(), task) + tasks = append(tasks, task) } + Config.Services.Queue.PushAtOnce(context.Background(), tasks) +} + +func taskIds(dependsOn []string, buildItems []*buildItem) []string { + taskIds := []string{} + for _, dep := range dependsOn { + for _, buildItem := range buildItems { + if buildItem.Proc.Name == dep { + taskIds = append(taskIds, fmt.Sprint(buildItem.Proc.ID)) + } + } + } + return taskIds } func shasum(raw []byte) string { diff --git a/server/hook_test.go b/server/hook_test.go deleted file mode 100644 index c34b0070d..000000000 --- a/server/hook_test.go +++ /dev/null @@ -1,45 +0,0 @@ -// Copyright 2018 Drone.IO Inc. -// -// 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 server - -import ( - "testing" - - "github.com/laszlocph/drone-oss-08/model" -) - -func TestMultilineEnvsubst(t *testing.T) { - b := procBuilder{ - Repo: &model.Repo{}, - Curr: &model.Build{ - Message: `aaa -bbb`, - }, - Last: &model.Build{}, - Netrc: &model.Netrc{}, - Secs: []*model.Secret{}, - Regs: []*model.Registry{}, - Link: "", - Yaml: `pipeline: - xxx: - image: scratch - yyy: ${DRONE_COMMIT_MESSAGE} -`, - } - - if _, err := b.Build(); err != nil { - t.Fatal(err) - } -} diff --git a/server/login.go b/server/login.go index c6354036f..a401a2981 100644 --- a/server/login.go +++ b/server/login.go @@ -38,7 +38,12 @@ func HandleLogin(c *gin.Context) { if err := r.FormValue("error"); err != "" { http.Redirect(w, r, "/login/error?code="+err, 303) } else { - http.Redirect(w, r, "/authorize", 303) + intendedURL := r.URL.Query()["url"] + if len(intendedURL) > 0 { + http.Redirect(w, r, "/authorize?url="+intendedURL[0], 303) + } else { + http.Redirect(w, r, "/authorize", 303) + } } } @@ -136,8 +141,13 @@ func HandleAuth(c *gin.Context) { } httputil.SetCookie(c.Writer, c.Request, "user_sess", tokenstr) - c.Redirect(303, "/") + intendedURL := c.Request.URL.Query()["url"] + if len(intendedURL) > 0 { + c.Redirect(303, intendedURL[0]) + } else { + c.Redirect(303, "/") + } } func GetLogout(c *gin.Context) { diff --git a/server/procBuilder.go b/server/procBuilder.go index f5ddee813..5eaadeefa 100644 --- a/server/procBuilder.go +++ b/server/procBuilder.go @@ -18,6 +18,7 @@ import ( "fmt" "math/rand" "net/url" + "sort" "strings" "github.com/drone/envsubst" @@ -28,6 +29,7 @@ import ( "github.com/laszlocph/drone-oss-08/cncd/pipeline/pipeline/frontend/yaml/linter" "github.com/laszlocph/drone-oss-08/cncd/pipeline/pipeline/frontend/yaml/matrix" "github.com/laszlocph/drone-oss-08/model" + "github.com/laszlocph/drone-oss-08/remote" ) // Takes the hook data and the yaml and returns in internal data model @@ -39,156 +41,198 @@ type procBuilder struct { Secs []*model.Secret Regs []*model.Registry Link string - Yaml string + Yamls []*remote.FileMeta Envs map[string]string } type buildItem struct { - Proc *model.Proc - Platform string - Labels map[string]string - Config *backend.Config + Proc *model.Proc + Platform string + Labels map[string]string + DependsOn []string + RunsOn []string + Config *backend.Config } func (b *procBuilder) Build() ([]*buildItem, error) { - - axes, err := matrix.ParseString(b.Yaml) - if err != nil { - return nil, err - } - if len(axes) == 0 { - axes = append(axes, matrix.Axis{}) - } - var items []*buildItem - for i, axis := range axes { - proc := &model.Proc{ - BuildID: b.Curr.ID, - PID: i + 1, - PGID: i + 1, - State: model.StatusPending, - Environ: axis, - } - metadata := metadataFromStruct(b.Repo, b.Curr, b.Last, proc, b.Link) - environ := metadata.Environ() - for k, v := range metadata.EnvironDrone() { - environ[k] = v - } - for k, v := range axis { - environ[k] = v - } + sort.Sort(remote.ByName(b.Yamls)) - var secrets []compiler.Secret - for _, sec := range b.Secs { - if !sec.Match(b.Curr.Event) { - continue - } - secrets = append(secrets, compiler.Secret{ - Name: sec.Name, - Value: sec.Value, - Match: sec.Images, - }) - } - - y := b.Yaml - s, err := envsubst.Eval(y, func(name string) string { - env := environ[name] - if strings.Contains(env, "\n") { - env = fmt.Sprintf("%q", env) - } - return env - }) + for j, y := range b.Yamls { + // matrix axes + axes, err := matrix.ParseString(string(y.Data)) if err != nil { return nil, err } - y = s - - parsed, err := yaml.ParseString(y) - if err != nil { - return nil, err - } - metadata.Sys.Arch = parsed.Platform - if metadata.Sys.Arch == "" { - metadata.Sys.Arch = "linux/amd64" + if len(axes) == 0 { + axes = append(axes, matrix.Axis{}) } - lerr := linter.New( - linter.WithTrusted(b.Repo.IsTrusted), - ).Lint(parsed) - if lerr != nil { - return nil, lerr - } + for i, axis := range axes { + proc := &model.Proc{ + BuildID: b.Curr.ID, + PID: j + i + 1, + PGID: j + i + 1, + State: model.StatusPending, + Environ: axis, + Name: sanitizePath(y.Name), + } + b.Curr.Procs = append(b.Curr.Procs, proc) - var registries []compiler.Registry - for _, reg := range b.Regs { - registries = append(registries, compiler.Registry{ - Hostname: reg.Address, - Username: reg.Username, - Password: reg.Password, - Email: reg.Email, - }) - } + metadata := metadataFromStruct(b.Repo, b.Curr, b.Last, proc, b.Link) + environ := b.environmentVariables(metadata, axis) - ir := compiler.New( - compiler.WithEnviron(environ), - compiler.WithEnviron(b.Envs), - compiler.WithEscalated(Config.Pipeline.Privileged...), - compiler.WithResourceLimit(Config.Pipeline.Limits.MemSwapLimit, Config.Pipeline.Limits.MemLimit, Config.Pipeline.Limits.ShmSize, Config.Pipeline.Limits.CPUQuota, Config.Pipeline.Limits.CPUShares, Config.Pipeline.Limits.CPUSet), - compiler.WithVolumes(Config.Pipeline.Volumes...), - compiler.WithNetworks(Config.Pipeline.Networks...), - compiler.WithLocal(false), - compiler.WithOption( - compiler.WithNetrc( - b.Netrc.Login, - b.Netrc.Password, - b.Netrc.Machine, - ), - b.Repo.IsPrivate, - ), - compiler.WithRegistry(registries...), - compiler.WithSecret(secrets...), - compiler.WithPrefix( - fmt.Sprintf( - "%d_%d", - proc.ID, - rand.Int(), - ), - ), - compiler.WithEnviron(proc.Environ), - compiler.WithProxy(), - compiler.WithWorkspaceFromURL("/drone", b.Repo.Link), - compiler.WithMetadata(metadata), - ).Compile(parsed) + // substitute vars + substituted, err := b.envsubst_(string(y.Data), environ) + if err != nil { + return nil, err + } - // for _, sec := range b.Secs { - // if !sec.MatchEvent(b.Curr.Event) { - // continue - // } - // if b.Curr.Verified || sec.SkipVerify { - // ir.Secrets = append(ir.Secrets, &backend.Secret{ - // Mask: sec.Conceal, - // Name: sec.Name, - // Value: sec.Value, - // }) - // } - // } + // parse yaml pipeline + parsed, err := yaml.ParseString(substituted) + if err != nil { + return nil, err + } - item := &buildItem{ - Proc: proc, - Config: ir, - Labels: parsed.Labels, - Platform: metadata.Sys.Arch, + // lint pipeline + lerr := linter.New( + linter.WithTrusted(b.Repo.IsTrusted), + ).Lint(parsed) + if lerr != nil { + return nil, lerr + } + + if !parsed.Branches.Match(b.Curr.Branch) { + proc.State = model.StatusSkipped + } + + metadata.SetPlatform(parsed.Platform) + + ir := b.toInternalRepresentation(parsed, environ, metadata, proc.ID) + + item := &buildItem{ + Proc: proc, + Config: ir, + Labels: parsed.Labels, + DependsOn: parsed.DependsOn, + RunsOn: parsed.RunsOn, + Platform: metadata.Sys.Arch, + } + if item.Labels == nil { + item.Labels = map[string]string{} + } + items = append(items, item) } - if item.Labels == nil { - item.Labels = map[string]string{} - } - items = append(items, item) } + setBuildSteps(b.Curr, items) + return items, nil } +func (b *procBuilder) envsubst_(y string, environ map[string]string) (string, error) { + return envsubst.Eval(y, func(name string) string { + env := environ[name] + if strings.Contains(env, "\n") { + env = fmt.Sprintf("%q", env) + } + return env + }) +} + +func (b *procBuilder) environmentVariables(metadata frontend.Metadata, axis matrix.Axis) map[string]string { + environ := metadata.Environ() + for k, v := range metadata.EnvironDrone() { + environ[k] = v + } + for k, v := range axis { + environ[k] = v + } + return environ +} + +func (b *procBuilder) toInternalRepresentation(parsed *yaml.Config, environ map[string]string, metadata frontend.Metadata, procID int64) *backend.Config { + var secrets []compiler.Secret + for _, sec := range b.Secs { + if !sec.Match(b.Curr.Event) { + continue + } + secrets = append(secrets, compiler.Secret{ + Name: sec.Name, + Value: sec.Value, + Match: sec.Images, + }) + } + + var registries []compiler.Registry + for _, reg := range b.Regs { + registries = append(registries, compiler.Registry{ + Hostname: reg.Address, + Username: reg.Username, + Password: reg.Password, + Email: reg.Email, + }) + } + + return compiler.New( + compiler.WithEnviron(environ), + compiler.WithEnviron(b.Envs), + compiler.WithEscalated(Config.Pipeline.Privileged...), + compiler.WithResourceLimit(Config.Pipeline.Limits.MemSwapLimit, Config.Pipeline.Limits.MemLimit, Config.Pipeline.Limits.ShmSize, Config.Pipeline.Limits.CPUQuota, Config.Pipeline.Limits.CPUShares, Config.Pipeline.Limits.CPUSet), + compiler.WithVolumes(Config.Pipeline.Volumes...), + compiler.WithNetworks(Config.Pipeline.Networks...), + compiler.WithLocal(false), + compiler.WithOption( + compiler.WithNetrc( + b.Netrc.Login, + b.Netrc.Password, + b.Netrc.Machine, + ), + b.Repo.IsPrivate, + ), + compiler.WithRegistry(registries...), + compiler.WithSecret(secrets...), + compiler.WithPrefix( + fmt.Sprintf( + "%d_%d", + procID, + rand.Int(), + ), + ), + compiler.WithProxy(), + compiler.WithWorkspaceFromURL("/drone", b.Repo.Link), + compiler.WithMetadata(metadata), + ).Compile(parsed) +} + +func setBuildSteps(build *model.Build, buildItems []*buildItem) { + pcounter := len(buildItems) + for _, item := range buildItems { + for _, stage := range item.Config.Stages { + var gid int + for _, step := range stage.Steps { + pcounter++ + if gid == 0 { + gid = pcounter + } + proc := &model.Proc{ + BuildID: build.ID, + Name: step.Alias, + PID: pcounter, + PPID: item.Proc.PID, + PGID: gid, + State: model.StatusPending, + } + if item.Proc.State == model.StatusSkipped { + proc.State = model.StatusSkipped + } + build.Procs = append(build.Procs, proc) + } + } + } +} + // return the metadata from the cli context. func metadataFromStruct(repo *model.Repo, build, last *model.Build, proc *model.Proc, link string) frontend.Metadata { host := link @@ -261,3 +305,10 @@ func metadataFromStruct(repo *model.Repo, build, last *model.Build, proc *model. }, } } + +func sanitizePath(path string) string { + path = strings.TrimSuffix(path, ".yml") + path = strings.TrimPrefix(path, ".drone/") + path = strings.TrimPrefix(path, ".") + return path +} diff --git a/server/procBuilder_test.go b/server/procBuilder_test.go new file mode 100644 index 000000000..4452dfaf4 --- /dev/null +++ b/server/procBuilder_test.go @@ -0,0 +1,206 @@ +// Copyright 2018 Drone.IO Inc. +// +// 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 server + +import ( + "fmt" + "testing" + + "github.com/laszlocph/drone-oss-08/model" + "github.com/laszlocph/drone-oss-08/remote" +) + +func TestMultilineEnvsubst(t *testing.T) { + b := procBuilder{ + Repo: &model.Repo{}, + Curr: &model.Build{ + Message: `aaa +bbb`, + }, + Last: &model.Build{}, + Netrc: &model.Netrc{}, + Secs: []*model.Secret{}, + Regs: []*model.Registry{}, + Link: "", + Yamls: []*remote.FileMeta{ + &remote.FileMeta{Data: []byte(` +pipeline: + xxx: + image: scratch + yyy: ${DRONE_COMMIT_MESSAGE} +`)}, + &remote.FileMeta{Data: []byte(` +pipeline: + build: + image: scratch + yyy: ${DRONE_COMMIT_MESSAGE} +`)}, + }} + + if buildItems, err := b.Build(); err != nil { + t.Fatal(err) + } else { + fmt.Println(buildItems) + } +} + +func TestMultiPipeline(t *testing.T) { + b := procBuilder{ + Repo: &model.Repo{}, + Curr: &model.Build{}, + Last: &model.Build{}, + Netrc: &model.Netrc{}, + Secs: []*model.Secret{}, + Regs: []*model.Registry{}, + Link: "", + Yamls: []*remote.FileMeta{ + &remote.FileMeta{Data: []byte(` +pipeline: + xxx: + image: scratch + yyy: ${DRONE_COMMIT_MESSAGE} +`)}, + &remote.FileMeta{Data: []byte(` +pipeline: + build: + image: scratch + yyy: ${DRONE_COMMIT_MESSAGE} +`)}, + }, + } + + buildItems, err := b.Build() + if err != nil { + t.Fatal(err) + } + if len(buildItems) != 2 { + t.Fatal("Should have generated 2 buildItems") + } +} + +func TestDependsOn(t *testing.T) { + b := procBuilder{ + Repo: &model.Repo{}, + Curr: &model.Build{}, + Last: &model.Build{}, + Netrc: &model.Netrc{}, + Secs: []*model.Secret{}, + Regs: []*model.Registry{}, + Link: "", + Yamls: []*remote.FileMeta{ + &remote.FileMeta{Data: []byte(` +pipeline: + deploy: + image: scratch + +depends_on: + - lint + - test + - build +`)}, + }, + } + + buildItems, err := b.Build() + if err != nil { + t.Fatal(err) + } + if len(buildItems[0].DependsOn) != 3 { + t.Fatal("Should have 3 dependencies") + } + if buildItems[0].DependsOn[1] != "test" { + t.Fatal("Should depend on test") + } +} + +func TestRunsOn(t *testing.T) { + b := procBuilder{ + Repo: &model.Repo{}, + Curr: &model.Build{}, + Last: &model.Build{}, + Netrc: &model.Netrc{}, + Secs: []*model.Secret{}, + Regs: []*model.Registry{}, + Link: "", + Yamls: []*remote.FileMeta{ + &remote.FileMeta{Data: []byte(` +pipeline: + deploy: + image: scratch + +runs_on: + - success + - failure +`)}, + }, + } + + buildItems, err := b.Build() + if err != nil { + t.Fatal(err) + } + if len(buildItems[0].RunsOn) != 2 { + t.Fatal("Should run on success and failure") + } + if buildItems[0].RunsOn[1] != "failure" { + t.Fatal("Should run on failure") + } +} + +func TestBranchFilter(t *testing.T) { + b := procBuilder{ + Repo: &model.Repo{}, + Curr: &model.Build{Branch: "dev"}, + Last: &model.Build{}, + Netrc: &model.Netrc{}, + Secs: []*model.Secret{}, + Regs: []*model.Registry{}, + Link: "", + Yamls: []*remote.FileMeta{ + &remote.FileMeta{Data: []byte(` +pipeline: + xxx: + image: scratch + yyy: ${DRONE_COMMIT_MESSAGE} +branches: master +`)}, + &remote.FileMeta{Data: []byte(` +pipeline: + build: + image: scratch + yyy: ${DRONE_COMMIT_MESSAGE} +`)}, + }, + } + + buildItems, err := b.Build() + if err != nil { + t.Fatal(err) + } + if len(buildItems) != 2 { + t.Fatal("Should have generated 2 buildItems") + } + if buildItems[0].Proc.State != model.StatusSkipped { + t.Fatal("Should not run on dev branch") + } + for _, child := range buildItems[0].Proc.Children { + if child.State != model.StatusSkipped { + t.Fatal("Children should skipped status too") + } + } + if buildItems[1].Proc.State != model.StatusPending { + t.Fatal("Should not run on dev branch") + } +} diff --git a/server/repo.go b/server/repo.go index 70c68269e..46c8ff07e 100644 --- a/server/repo.go +++ b/server/repo.go @@ -150,6 +150,9 @@ func PatchRepo(c *gin.Context) { if in.BuildCounter != nil { repo.Counter = *in.BuildCounter } + if in.Fallback != nil { + repo.Fallback = *in.Fallback + } err := store.UpdateRepo(c, repo) if err != nil { diff --git a/server/rpc.go b/server/rpc.go index 61ff8648a..2fa6640b0 100644 --- a/server/rpc.go +++ b/server/rpc.go @@ -33,6 +33,8 @@ import ( "github.com/laszlocph/drone-oss-08/cncd/pipeline/pipeline/rpc/proto" "github.com/laszlocph/drone-oss-08/cncd/pubsub" "github.com/laszlocph/drone-oss-08/cncd/queue" + "github.com/prometheus/client_golang/prometheus" + "github.com/prometheus/client_golang/prometheus/promauto" "github.com/laszlocph/drone-oss-08/model" "github.com/laszlocph/drone-oss-08/remote" @@ -41,11 +43,6 @@ import ( "github.com/drone/expr" ) -// This file is a complete disaster because I'm trying to wedge in some -// experimental code. Please pardon our appearance during renovations. - -// Config is an evil global configuration that will be used as we transition / -// refactor the codebase to move away from storing these values in the Context. var Config = struct { Services struct { Pubsub pubsub.Publisher @@ -91,12 +88,14 @@ var Config = struct { }{} type RPC struct { - remote remote.Remote - queue queue.Queue - pubsub pubsub.Publisher - logger logging.Log - store store.Store - host string + remote remote.Remote + queue queue.Queue + pubsub pubsub.Publisher + logger logging.Log + store store.Store + host string + buildTime *prometheus.GaugeVec + buildCount *prometheus.CounterVec } // Next implements the rpc.Next function @@ -113,26 +112,22 @@ func (s *RPC) Next(c context.Context, filter rpc.Filter) (*rpc.Pipeline, error) if err != nil { return nil, err } - task, err := s.queue.Poll(c, fn) - if err != nil { - return nil, err - } else if task == nil { - return nil, nil + for { + task, err := s.queue.Poll(c, fn) + if err != nil { + return nil, err + } else if task == nil { + return nil, nil + } + + if task.ShouldRun() { + pipeline := new(rpc.Pipeline) + err = json.Unmarshal(task.Data, pipeline) + return pipeline, err + } else { + s.Done(c, task.ID, rpc.State{}) + } } - pipeline := new(rpc.Pipeline) - - // check if the process was previously cancelled - // cancelled, _ := s.checkCancelled(pipeline) - // if cancelled { - // logrus.Debugf("ignore pid %v: cancelled by user", pipeline.ID) - // if derr := s.queue.Done(c, pipeline.ID); derr != nil { - // logrus.Errorf("error: done: cannot ack proc_id %v: %s", pipeline.ID, err) - // } - // return nil, nil - // } - - err = json.Unmarshal(task.Data, pipeline) - return pipeline, err } // Wait implements the rpc.Wait function @@ -383,76 +378,147 @@ func (s *RPC) Done(c context.Context, id string, state rpc.State) error { return err } - proc.Stopped = state.Finished - proc.Error = state.Error - proc.ExitCode = state.ExitCode - proc.State = model.StatusSuccess - if proc.ExitCode != 0 || proc.Error != "" { - proc.State = model.StatusFailure - } - if err := s.store.ProcUpdate(proc); err != nil { - log.Printf("error: done: cannot update proc_id %d state: %s", procID, err) - } + s.updateProcState(proc, state) - if err := s.queue.Done(c, id); err != nil { + var queueErr error + if proc.Failing() { + queueErr = s.queue.Error(c, id, fmt.Errorf("Proc finished with exitcode %d, %s", state.ExitCode, state.Error)) + } else { + queueErr = s.queue.Done(c, id) + } + if queueErr != nil { log.Printf("error: done: cannot ack proc_id %d: %s", procID, err) } - // TODO handle this error procs, _ := s.store.ProcList(build) - for _, p := range procs { - if p.Running() && p.PPID == proc.PID { - p.State = model.StatusSkipped - if p.Started != 0 { - p.State = model.StatusSuccess // for deamons that are killed - p.Stopped = proc.Stopped - } - if err := s.store.ProcUpdate(p); err != nil { - log.Printf("error: done: cannot update proc_id %d child state: %s", p.ID, err) - } - } - } + s.completeChildrenIfParentCompleted(procs, proc) - running := false - status := model.StatusSuccess - for _, p := range procs { - if p.PPID == 0 { - if p.Running() { - running = true - } - if p.Failing() { - status = p.State - } - } - } - if !running { - build.Status = status + if !isThereRunningStage(procs) { + build.Status = buildStatus(procs) build.Finished = proc.Stopped if err := s.store.UpdateBuild(build); err != nil { log.Printf("error: done: cannot update build_id %d final state: %s", build.ID, err) } - // update the status - user, err := s.store.GetUser(repo.UserID) - if err == nil { - if refresher, ok := s.remote.(remote.Refresher); ok { - ok, _ := refresher.Refresh(user) - if ok { - s.store.UpdateUser(user) - } - } - uri := fmt.Sprintf("%s/%s/%d", s.host, repo.FullName, build.Number) - err = s.remote.Status(user, repo, build, uri) - if err != nil { - logrus.Errorf("error setting commit status for %s/%d: %v", repo.FullName, build.Number, err) - } + if !isMultiPipeline(procs) { + s.updateRemoteStatus(repo, build, nil) } } + if isMultiPipeline(procs) { + s.updateRemoteStatus(repo, build, proc) + } + if err := s.logger.Close(c, id); err != nil { log.Printf("error: done: cannot close build_id %d logger: %s", proc.ID, err) } + s.notify(c, repo, build, procs) + + if build.Status == model.StatusSuccess || build.Status == model.StatusFailure { + s.buildCount.WithLabelValues(repo.FullName, build.Branch, build.Status, "total").Inc() + s.buildTime.WithLabelValues(repo.FullName, build.Branch, build.Status, "total").Set(float64(build.Finished - build.Started)) + } + if isMultiPipeline(procs) { + s.buildTime.WithLabelValues(repo.FullName, build.Branch, proc.State, proc.Name).Set(float64(proc.Stopped - proc.Started)) + } + + return nil +} + +func isMultiPipeline(procs []*model.Proc) bool { + countPPIDZero := 0 + for _, proc := range procs { + if proc.PPID == 0 { + countPPIDZero++ + } + } + return countPPIDZero > 1 +} + +// Log implements the rpc.Log function +func (s *RPC) Log(c context.Context, id string, line *rpc.Line) error { + entry := new(logging.Entry) + entry.Data, _ = json.Marshal(line) + s.logger.Write(c, id, entry) + return nil +} + +func (s *RPC) updateProcState(proc *model.Proc, state rpc.State) { + proc.Stopped = state.Finished + proc.Error = state.Error + proc.ExitCode = state.ExitCode + if state.Started == 0 { + proc.State = model.StatusSkipped + } else { + proc.State = model.StatusSuccess + } + if proc.ExitCode != 0 || proc.Error != "" { + proc.State = model.StatusFailure + } + if err := s.store.ProcUpdate(proc); err != nil { + log.Printf("error: done: cannot update proc_id %d state: %s", proc.ID, err) + } +} + +func (s *RPC) completeChildrenIfParentCompleted(procs []*model.Proc, completedProc *model.Proc) { + for _, p := range procs { + if p.Running() && p.PPID == completedProc.PID { + p.State = model.StatusSkipped + if p.Started != 0 { + p.State = model.StatusSuccess // for deamons that are killed + p.Stopped = completedProc.Stopped + } + if err := s.store.ProcUpdate(p); err != nil { + log.Printf("error: done: cannot update proc_id %d child state: %s", p.ID, err) + } + } + } +} + +func isThereRunningStage(procs []*model.Proc) bool { + for _, p := range procs { + if p.PPID == 0 { + if p.Running() { + return true + } + } + } + return false +} + +func buildStatus(procs []*model.Proc) string { + status := model.StatusSuccess + + for _, p := range procs { + if p.PPID == 0 { + if p.Failing() { + status = p.State + } + } + } + + return status +} + +func (s *RPC) updateRemoteStatus(repo *model.Repo, build *model.Build, proc *model.Proc) { + user, err := s.store.GetUser(repo.UserID) + if err == nil { + if refresher, ok := s.remote.(remote.Refresher); ok { + ok, _ := refresher.Refresh(user) + if ok { + s.store.UpdateUser(user) + } + } + uri := fmt.Sprintf("%s/%s/%d", s.host, repo.FullName, build.Number) + err = s.remote.Status(user, repo, build, uri, proc) + if err != nil { + logrus.Errorf("error setting commit status for %s/%d: %v", repo.FullName, build.Number, err) + } + } +} + +func (s *RPC) notify(c context.Context, repo *model.Repo, build *model.Build, procs []*model.Proc) { build.Procs = model.Tree(procs) message := pubsub.Message{ Labels: map[string]string{ @@ -465,31 +531,6 @@ func (s *RPC) Done(c context.Context, id string, state rpc.State) error { Build: *build, }) s.pubsub.Publish(c, "topic/events", message) - - return nil -} - -// Log implements the rpc.Log function -func (s *RPC) Log(c context.Context, id string, line *rpc.Line) error { - entry := new(logging.Entry) - entry.Data, _ = json.Marshal(line) - s.logger.Write(c, id, entry) - return nil -} - -func (s *RPC) checkCancelled(pipeline *rpc.Pipeline) (bool, error) { - pid, err := strconv.ParseInt(pipeline.ID, 10, 64) - if err != nil { - return false, err - } - proc, err := s.store.ProcLoad(pid) - if err != nil { - return false, err - } - if proc.State == model.StatusKilled { - return true, nil - } - return false, err } func createFilterFunc(filter rpc.Filter) (queue.Filter, error) { @@ -524,30 +565,41 @@ func createFilterFunc(filter rpc.Filter) (queue.Filter, error) { // DroneServer is a grpc server implementation. type DroneServer struct { - Remote remote.Remote - Queue queue.Queue - Pubsub pubsub.Publisher - Logger logging.Log - Store store.Store - Host string + peer RPC +} + +func NewDroneServer(remote remote.Remote, queue queue.Queue, logger logging.Log, pubsub pubsub.Publisher, store store.Store, host string) *DroneServer { + buildTime := promauto.NewGaugeVec(prometheus.GaugeOpts{ + Namespace: "drone", + Name: "build_time", + Help: "Build time.", + }, []string{"repo", "branch", "status", "pipeline"}) + buildCount := promauto.NewCounterVec(prometheus.CounterOpts{ + Namespace: "drone", + Name: "build_count", + Help: "Build count.", + }, []string{"repo", "branch", "status", "pipeline"}) + peer := RPC{ + remote: remote, + store: store, + queue: queue, + pubsub: pubsub, + logger: logger, + host: host, + buildTime: buildTime, + buildCount: buildCount, + } + return &DroneServer{peer: peer} } func (s *DroneServer) Next(c oldcontext.Context, req *proto.NextRequest) (*proto.NextReply, error) { - peer := RPC{ - remote: s.Remote, - store: s.Store, - queue: s.Queue, - pubsub: s.Pubsub, - logger: s.Logger, - host: s.Host, - } filter := rpc.Filter{ Labels: req.GetFilter().GetLabels(), Expr: req.GetFilter().GetExpr(), } res := new(proto.NextReply) - pipeline, err := peer.Next(c, filter) + pipeline, err := s.peer.Next(c, filter) if err != nil { return res, err } @@ -561,53 +613,9 @@ func (s *DroneServer) Next(c oldcontext.Context, req *proto.NextRequest) (*proto res.Pipeline.Payload, _ = json.Marshal(pipeline.Config) return res, err - - // fn := func(task *queue.Task) bool { - // for k, v := range req.GetFilter().Labels { - // if task.Labels[k] != v { - // return false - // } - // } - // return true - // } - // task, err := s.Queue.Poll(c, fn) - // if err != nil { - // return nil, err - // } else if task == nil { - // return nil, nil - // } - // - // pipeline := new(rpc.Pipeline) - // json.Unmarshal(task.Data, pipeline) - // - // res := new(proto.NextReply) - // res.Pipeline = new(proto.Pipeline) - // res.Pipeline.Id = pipeline.ID - // res.Pipeline.Timeout = pipeline.Timeout - // res.Pipeline.Payload, _ = json.Marshal(pipeline.Config) - // - // // check if the process was previously cancelled - // // cancelled, _ := s.checkCancelled(pipeline) - // // if cancelled { - // // logrus.Debugf("ignore pid %v: cancelled by user", pipeline.ID) - // // if derr := s.queue.Done(c, pipeline.ID); derr != nil { - // // logrus.Errorf("error: done: cannot ack proc_id %v: %s", pipeline.ID, err) - // // } - // // return nil, nil - // // } - // - // return res, nil } func (s *DroneServer) Init(c oldcontext.Context, req *proto.InitRequest) (*proto.Empty, error) { - peer := RPC{ - remote: s.Remote, - store: s.Store, - queue: s.Queue, - pubsub: s.Pubsub, - logger: s.Logger, - host: s.Host, - } state := rpc.State{ Error: req.GetState().GetError(), ExitCode: int(req.GetState().GetExitCode()), @@ -617,19 +625,11 @@ func (s *DroneServer) Init(c oldcontext.Context, req *proto.InitRequest) (*proto Exited: req.GetState().GetExited(), } res := new(proto.Empty) - err := peer.Init(c, req.GetId(), state) + err := s.peer.Init(c, req.GetId(), state) return res, err } func (s *DroneServer) Update(c oldcontext.Context, req *proto.UpdateRequest) (*proto.Empty, error) { - peer := RPC{ - remote: s.Remote, - store: s.Store, - queue: s.Queue, - pubsub: s.Pubsub, - logger: s.Logger, - host: s.Host, - } state := rpc.State{ Error: req.GetState().GetError(), ExitCode: int(req.GetState().GetExitCode()), @@ -639,19 +639,11 @@ func (s *DroneServer) Update(c oldcontext.Context, req *proto.UpdateRequest) (*p Exited: req.GetState().GetExited(), } res := new(proto.Empty) - err := peer.Update(c, req.GetId(), state) + err := s.peer.Update(c, req.GetId(), state) return res, err } func (s *DroneServer) Upload(c oldcontext.Context, req *proto.UploadRequest) (*proto.Empty, error) { - peer := RPC{ - remote: s.Remote, - store: s.Store, - queue: s.Queue, - pubsub: s.Pubsub, - logger: s.Logger, - host: s.Host, - } file := &rpc.File{ Data: req.GetFile().GetData(), Mime: req.GetFile().GetMime(), @@ -663,19 +655,11 @@ func (s *DroneServer) Upload(c oldcontext.Context, req *proto.UploadRequest) (*p } res := new(proto.Empty) - err := peer.Upload(c, req.GetId(), file) + err := s.peer.Upload(c, req.GetId(), file) return res, err } func (s *DroneServer) Done(c oldcontext.Context, req *proto.DoneRequest) (*proto.Empty, error) { - peer := RPC{ - remote: s.Remote, - store: s.Store, - queue: s.Queue, - pubsub: s.Pubsub, - logger: s.Logger, - host: s.Host, - } state := rpc.State{ Error: req.GetState().GetError(), ExitCode: int(req.GetState().GetExitCode()), @@ -685,47 +669,23 @@ func (s *DroneServer) Done(c oldcontext.Context, req *proto.DoneRequest) (*proto Exited: req.GetState().GetExited(), } res := new(proto.Empty) - err := peer.Done(c, req.GetId(), state) + err := s.peer.Done(c, req.GetId(), state) return res, err } func (s *DroneServer) Wait(c oldcontext.Context, req *proto.WaitRequest) (*proto.Empty, error) { - peer := RPC{ - remote: s.Remote, - store: s.Store, - queue: s.Queue, - pubsub: s.Pubsub, - logger: s.Logger, - host: s.Host, - } res := new(proto.Empty) - err := peer.Wait(c, req.GetId()) + err := s.peer.Wait(c, req.GetId()) return res, err } func (s *DroneServer) Extend(c oldcontext.Context, req *proto.ExtendRequest) (*proto.Empty, error) { - peer := RPC{ - remote: s.Remote, - store: s.Store, - queue: s.Queue, - pubsub: s.Pubsub, - logger: s.Logger, - host: s.Host, - } res := new(proto.Empty) - err := peer.Extend(c, req.GetId()) + err := s.peer.Extend(c, req.GetId()) return res, err } func (s *DroneServer) Log(c oldcontext.Context, req *proto.LogRequest) (*proto.Empty, error) { - peer := RPC{ - remote: s.Remote, - store: s.Store, - queue: s.Queue, - pubsub: s.Pubsub, - logger: s.Logger, - host: s.Host, - } line := &rpc.Line{ Out: req.GetLine().GetOut(), Pos: int(req.GetLine().GetPos()), @@ -733,6 +693,6 @@ func (s *DroneServer) Log(c oldcontext.Context, req *proto.LogRequest) (*proto.E Proc: req.GetLine().GetProc(), } res := new(proto.Empty) - err := peer.Log(c, req.GetId(), line) + err := s.peer.Log(c, req.GetId(), line) return res, err } diff --git a/server/stream.go b/server/stream.go index 56915e332..c4d42341c 100644 --- a/server/stream.go +++ b/server/stream.go @@ -77,7 +77,6 @@ func EventStreamSSE(c *gin.Context) { }() go func() { - // TODO remove this from global config Config.Services.Pubsub.Subscribe(ctx, "topic/events", func(m pubsub.Message) { defer func() { recover() // fix #2480 diff --git a/store/datastore/builds_test.go b/store/datastore/builds_test.go index cc8f8f12e..452936e1c 100644 --- a/store/datastore/builds_test.go +++ b/store/datastore/builds_test.go @@ -18,8 +18,8 @@ import ( "fmt" "testing" - "github.com/laszlocph/drone-oss-08/model" "github.com/franela/goblin" + "github.com/laszlocph/drone-oss-08/model" ) func TestBuilds(t *testing.T) { diff --git a/store/datastore/config.go b/store/datastore/config.go index 7ac585347..1be819bff 100644 --- a/store/datastore/config.go +++ b/store/datastore/config.go @@ -22,17 +22,17 @@ import ( "github.com/russross/meddler" ) -func (db *datastore) ConfigLoad(id int64) (*model.Config, error) { +func (db *datastore) ConfigsForBuild(buildID int64) ([]*model.Config, error) { stmt := sql.Lookup(db.driver, "config-find-id") - conf := new(model.Config) - err := meddler.QueryRow(db, conf, stmt, id) - return conf, err + var configs = []*model.Config{} + err := meddler.QueryAll(db, &configs, stmt, buildID) + return configs, err } -func (db *datastore) ConfigFind(repo *model.Repo, hash string) (*model.Config, error) { +func (db *datastore) ConfigFindIdentical(repoID int64, hash string) (*model.Config, error) { stmt := sql.Lookup(db.driver, "config-find-repo-hash") conf := new(model.Config) - err := meddler.QueryRow(db, conf, stmt, repo.ID, hash) + err := meddler.QueryRow(db, conf, stmt, repoID, hash) return conf, err } @@ -51,3 +51,7 @@ func (db *datastore) ConfigFindApproved(config *model.Config) (bool, error) { func (db *datastore) ConfigCreate(config *model.Config) error { return meddler.Insert(db, "config", config) } + +func (db *datastore) BuildConfigCreate(buildConfig *model.BuildConfig) error { + return meddler.Insert(db, "build_config", buildConfig) +} diff --git a/store/datastore/config_test.go b/store/datastore/config_test.go index dedef4466..35f2d2398 100644 --- a/store/datastore/config_test.go +++ b/store/datastore/config_test.go @@ -23,6 +23,9 @@ import ( func TestConfig(t *testing.T) { s := newTest() defer func() { + s.Exec("delete from repos") + s.Exec("delete from builds") + s.Exec("delete from procs") s.Exec("delete from config") s.Close() }() @@ -32,18 +35,49 @@ func TestConfig(t *testing.T) { hash = "8d8647c9aa90d893bfb79dddbe901f03e258588121e5202632f8ae5738590b26" ) - if err := s.ConfigCreate( - &model.Config{ - RepoID: 2, - Data: data, - Hash: hash, - }, - ); err != nil { + repo := &model.Repo{ + UserID: 1, + FullName: "bradrydzewski/drone", + Owner: "bradrydzewski", + Name: "drone", + } + if err := s.CreateRepo(repo); err != nil { + t.Errorf("Unexpected error: insert repo: %s", err) + return + } + + config := &model.Config{ + RepoID: repo.ID, + Data: data, + Hash: hash, + Name: "default", + } + if err := s.ConfigCreate(config); err != nil { t.Errorf("Unexpected error: insert config: %s", err) return } - config, err := s.ConfigFind(&model.Repo{ID: 2}, hash) + build := &model.Build{ + RepoID: repo.ID, + Status: model.StatusRunning, + Commit: "85f8c029b902ed9400bc600bac301a0aadb144ac", + } + if err := s.CreateBuild(build); err != nil { + t.Errorf("Unexpected error: insert build: %s", err) + return + } + + if err := s.BuildConfigCreate( + &model.BuildConfig{ + ConfigID: config.ID, + BuildID: build.ID, + }, + ); err != nil { + t.Errorf("Unexpected error: insert build config: %s", err) + return + } + + config, err := s.ConfigFindIdentical(repo.ID, hash) if err != nil { t.Error(err) return @@ -51,7 +85,7 @@ func TestConfig(t *testing.T) { if got, want := config.ID, int64(1); got != want { t.Errorf("Want config id %d, got %d", want, got) } - if got, want := config.RepoID, int64(2); got != want { + if got, want := config.RepoID, repo.ID; got != want { t.Errorf("Want config repo id %d, got %d", want, got) } if got, want := config.Data, data; got != want { @@ -60,13 +94,16 @@ func TestConfig(t *testing.T) { if got, want := config.Hash, hash; got != want { t.Errorf("Want config hash %s, got %s", want, got) } + if got, want := config.Name, "default"; got != want { + t.Errorf("Want config name %s, got %s", want, got) + } - loaded, err := s.ConfigLoad(config.ID) + loaded, err := s.ConfigsForBuild(build.ID) if err != nil { t.Errorf("Want config by id, got error %q", err) return } - if got, want := loaded.ID, config.ID; got != want { + if got, want := loaded[0].ID, config.ID; got != want { t.Errorf("Want config by id %d, got %d", want, got) } } @@ -74,9 +111,10 @@ func TestConfig(t *testing.T) { func TestConfigApproved(t *testing.T) { s := newTest() defer func() { - s.Exec("delete from config") - s.Exec("delete from builds") s.Exec("delete from repos") + s.Exec("delete from builds") + s.Exec("delete from procs") + s.Exec("delete from config") s.Close() }() @@ -86,49 +124,83 @@ func TestConfigApproved(t *testing.T) { Owner: "bradrydzewski", Name: "drone", } - s.CreateRepo(repo) + if err := s.CreateRepo(repo); err != nil { + t.Errorf("Unexpected error: insert repo: %s", err) + return + } var ( - data = "pipeline: [ { image: golang, commands: [ go build, go test ] } ]" - hash = "8d8647c9aa90d893bfb79dddbe901f03e258588121e5202632f8ae5738590b26" - conf = &model.Config{ + data = "pipeline: [ { image: golang, commands: [ go build, go test ] } ]" + hash = "8d8647c9aa90d893bfb79dddbe901f03e258588121e5202632f8ae5738590b26" + buildBlocked = &model.Build{ RepoID: repo.ID, - Data: data, - Hash: hash, + Status: model.StatusBlocked, + Commit: "85f8c029b902ed9400bc600bac301a0aadb144ac", + } + buildPending = &model.Build{ + RepoID: repo.ID, + Status: model.StatusPending, + Commit: "85f8c029b902ed9400bc600bac301a0aadb144ac", + } + buildRunning = &model.Build{ + RepoID: repo.ID, + Status: model.StatusRunning, + Commit: "85f8c029b902ed9400bc600bac301a0aadb144ac", } ) + if err := s.CreateBuild(buildBlocked); err != nil { + t.Errorf("Unexpected error: insert build: %s", err) + return + } + if err := s.CreateBuild(buildPending); err != nil { + t.Errorf("Unexpected error: insert build: %s", err) + return + } + conf := &model.Config{ + RepoID: repo.ID, + Data: data, + Hash: hash, + } if err := s.ConfigCreate(conf); err != nil { t.Errorf("Unexpected error: insert config: %s", err) return } - s.CreateBuild(&model.Build{ - RepoID: repo.ID, + buildConfig := &model.BuildConfig{ ConfigID: conf.ID, - Status: model.StatusBlocked, - Commit: "85f8c029b902ed9400bc600bac301a0aadb144ac", - }) - s.CreateBuild(&model.Build{ - RepoID: repo.ID, - ConfigID: conf.ID, - Status: model.StatusPending, - Commit: "85f8c029b902ed9400bc600bac301a0aadb144ac", - }) - - if ok, _ := s.ConfigFindApproved(conf); ok == true { - t.Errorf("Want config not approved, when blocked or pending") + BuildID: buildBlocked.ID, + } + if err := s.BuildConfigCreate(buildConfig); err != nil { + t.Errorf("Unexpected error: insert build_config: %s", err) return } - s.CreateBuild(&model.Build{ - RepoID: repo.ID, - ConfigID: conf.ID, - Status: model.StatusRunning, - Commit: "85f8c029b902ed9400bc600bac301a0aadb144ac", - }) + if approved, err := s.ConfigFindApproved(conf); approved != false || err != nil { + t.Errorf("Want config not approved, when blocked or pending. %v", err) + return + } - if ok, _ := s.ConfigFindApproved(conf); ok == false { - t.Errorf("Want config approved, when running.") + s.CreateBuild(buildRunning) + conf2 := &model.Config{ + RepoID: repo.ID, + Data: data, + Hash: "xxx", + } + if err := s.ConfigCreate(conf2); err != nil { + t.Errorf("Unexpected error: insert config: %s", err) + return + } + buildConfig2 := &model.BuildConfig{ + ConfigID: conf2.ID, + BuildID: buildRunning.ID, + } + if err := s.BuildConfigCreate(buildConfig2); err != nil { + t.Errorf("Unexpected error: insert config: %s", err) + return + } + + if approved, err := s.ConfigFindApproved(conf2); approved != true || err != nil { + t.Errorf("Want config approved, when running. %v", err) return } } @@ -136,6 +208,9 @@ func TestConfigApproved(t *testing.T) { func TestConfigIndexes(t *testing.T) { s := newTest() defer func() { + s.Exec("delete from repos") + s.Exec("delete from builds") + s.Exec("delete from procs") s.Exec("delete from config") s.Close() }() diff --git a/store/datastore/ddl/mysql/ddl_gen.go b/store/datastore/ddl/mysql/ddl_gen.go index b12efe126..4d4048f6f 100644 --- a/store/datastore/ddl/mysql/ddl_gen.go +++ b/store/datastore/ddl/mysql/ddl_gen.go @@ -1,17 +1,3 @@ -// Copyright 2018 Drone.IO Inc. -// -// 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 mysql import ( @@ -170,6 +156,38 @@ var migrations = []struct { name: "alter-table-update-file-meta", stmt: alterTableUpdateFileMeta, }, + { + name: "create-table-build-config", + stmt: createTableBuildConfig, + }, + { + name: "alter-table-add-config-name", + stmt: alterTableAddConfigName, + }, + { + name: "update-table-set-config-name", + stmt: updateTableSetConfigName, + }, + { + name: "populate-build-config", + stmt: populateBuildConfig, + }, + { + name: "alter-table-add-task-dependencies", + stmt: alterTableAddTaskDependencies, + }, + { + name: "alter-table-add-task-run-on", + stmt: alterTableAddTaskRunOn, + }, + { + name: "alter-table-add-repo-fallback", + stmt: alterTableAddRepoFallback, + }, + { + name: "update-table-set-repo-fallback", + stmt: updateTableSetRepoFallback, + }, } // Migrate performs the database migration. If the migration fails @@ -636,3 +654,62 @@ UPDATE files SET ,file_meta_failed=0 ,file_meta_skipped=0 ` + +// +// 019_create_table_build_config.sql +// + +var createTableBuildConfig = ` +CREATE TABLE IF NOT EXISTS build_config ( + config_id INTEGER NOT NULL +,build_id INTEGER NOT NULL +,PRIMARY KEY (config_id, build_id) +,FOREIGN KEY (config_id) REFERENCES config (config_id) +,FOREIGN KEY (build_id) REFERENCES builds (build_id) +); +` + +// +// 020_add_column_config_name.sql +// + +var alterTableAddConfigName = ` +ALTER TABLE config ADD COLUMN config_name TEXT +` + +var updateTableSetConfigName = ` +UPDATE config SET config_name = "drone" +` + +// +// 021_populate_build_config.sql +// + +var populateBuildConfig = ` +INSERT INTO build_config (config_id, build_id) +SELECT build_config_id, build_id FROM builds +` + +// +// 022_add_task_columns.sql +// + +var alterTableAddTaskDependencies = ` +ALTER TABLE tasks ADD COLUMN task_dependencies MEDIUMBLOB +` + +var alterTableAddTaskRunOn = ` +ALTER TABLE tasks ADD COLUMN task_run_on MEDIUMBLOB +` + +// +// 023_add_repo_fallback_column.sql +// + +var alterTableAddRepoFallback = ` +ALTER TABLE repos ADD COLUMN repo_fallback BOOLEAN +` + +var updateTableSetRepoFallback = ` +UPDATE repos SET repo_fallback='false' +` diff --git a/store/datastore/ddl/mysql/files/019_create_table_build_config.sql b/store/datastore/ddl/mysql/files/019_create_table_build_config.sql new file mode 100644 index 000000000..be5d58718 --- /dev/null +++ b/store/datastore/ddl/mysql/files/019_create_table_build_config.sql @@ -0,0 +1,9 @@ +-- name: create-table-build-config + +CREATE TABLE IF NOT EXISTS build_config ( + config_id INTEGER NOT NULL +,build_id INTEGER NOT NULL +,PRIMARY KEY (config_id, build_id) +,FOREIGN KEY (config_id) REFERENCES config (config_id) +,FOREIGN KEY (build_id) REFERENCES builds (build_id) +); diff --git a/store/datastore/ddl/mysql/files/020_add_column_config_name.sql b/store/datastore/ddl/mysql/files/020_add_column_config_name.sql new file mode 100644 index 000000000..0140b4035 --- /dev/null +++ b/store/datastore/ddl/mysql/files/020_add_column_config_name.sql @@ -0,0 +1,7 @@ +-- name: alter-table-add-config-name + +ALTER TABLE config ADD COLUMN config_name TEXT + +-- name: update-table-set-config-name + +UPDATE config SET config_name = "drone" \ No newline at end of file diff --git a/store/datastore/ddl/mysql/files/021_populate_build_config.sql b/store/datastore/ddl/mysql/files/021_populate_build_config.sql new file mode 100644 index 000000000..e004b81c9 --- /dev/null +++ b/store/datastore/ddl/mysql/files/021_populate_build_config.sql @@ -0,0 +1,4 @@ +-- name: populate-build-config + +INSERT INTO build_config (config_id, build_id) +SELECT build_config_id, build_id FROM builds diff --git a/store/datastore/ddl/mysql/files/022_add_task_columns.sql b/store/datastore/ddl/mysql/files/022_add_task_columns.sql new file mode 100644 index 000000000..b98a2180a --- /dev/null +++ b/store/datastore/ddl/mysql/files/022_add_task_columns.sql @@ -0,0 +1,6 @@ +-- name: alter-table-add-task-dependencies +ALTER TABLE tasks ADD COLUMN task_dependencies MEDIUMBLOB + +-- name: alter-table-add-task-run-on + +ALTER TABLE tasks ADD COLUMN task_run_on MEDIUMBLOB \ No newline at end of file diff --git a/store/datastore/ddl/mysql/files/023_add_repo_fallback_column.sql b/store/datastore/ddl/mysql/files/023_add_repo_fallback_column.sql new file mode 100644 index 000000000..3e3384854 --- /dev/null +++ b/store/datastore/ddl/mysql/files/023_add_repo_fallback_column.sql @@ -0,0 +1,5 @@ +-- name: alter-table-add-repo-fallback +ALTER TABLE repos ADD COLUMN repo_fallback BOOLEAN + +-- name: update-table-set-repo-fallback +UPDATE repos SET repo_fallback='false' diff --git a/store/datastore/ddl/postgres/ddl_gen.go b/store/datastore/ddl/postgres/ddl_gen.go index 3915069fa..c55674bb9 100644 --- a/store/datastore/ddl/postgres/ddl_gen.go +++ b/store/datastore/ddl/postgres/ddl_gen.go @@ -1,17 +1,3 @@ -// Copyright 2018 Drone.IO Inc. -// -// 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 postgres import ( @@ -170,6 +156,38 @@ var migrations = []struct { name: "alter-table-update-file-meta", stmt: alterTableUpdateFileMeta, }, + { + name: "create-table-build-config", + stmt: createTableBuildConfig, + }, + { + name: "alter-table-add-config-name", + stmt: alterTableAddConfigName, + }, + { + name: "update-table-set-config-name", + stmt: updateTableSetConfigName, + }, + { + name: "populate-build-config", + stmt: populateBuildConfig, + }, + { + name: "alter-table-add-task-dependencies", + stmt: alterTableAddTaskDependencies, + }, + { + name: "alter-table-add-task-run-on", + stmt: alterTableAddTaskRunOn, + }, + { + name: "alter-table-add-repo-fallback", + stmt: alterTableAddRepoFallback, + }, + { + name: "update-table-set-repo-fallback", + stmt: updateTableSetRepoFallback, + }, } // Migrate performs the database migration. If the migration fails @@ -530,7 +548,7 @@ CREATE INDEX IF NOT EXISTS sender_repo_ix ON senders (sender_repo_id); // var alterTableAddRepoVisibility = ` -ALTER TABLE repos ADD COLUMN repo_visibility VARCHAR(50) +ALTER TABLE repos ADD COLUMN repo_visibility VARCHAR(50); ` var updateTableSetRepoVisibility = ` @@ -538,7 +556,7 @@ UPDATE repos SET repo_visibility = (CASE WHEN repo_private = false THEN 'public' ELSE 'private' - END) + END); ` // @@ -554,12 +572,13 @@ UPDATE repos SET repo_counter = ( SELECT max(build_number) FROM builds WHERE builds.build_repo_id = repos.repo_id -) +); ` var updateTableSetRepoSeqDefault = ` UPDATE repos SET repo_counter = 0 WHERE repo_counter IS NULL +; ` // @@ -567,11 +586,11 @@ WHERE repo_counter IS NULL // var alterTableAddRepoActive = ` -ALTER TABLE repos ADD COLUMN repo_active BOOLEAN +ALTER TABLE repos ADD COLUMN repo_active BOOLEAN; ` var updateTableSetRepoActive = ` -UPDATE repos SET repo_active = true +UPDATE repos SET repo_active = true; ` // @@ -583,7 +602,7 @@ ALTER TABLE users ADD COLUMN user_synced INTEGER; ` var updateTableSetUserSynced = ` -UPDATE users SET user_synced = 0 +UPDATE users SET user_synced = 0; ` // @@ -615,19 +634,19 @@ CREATE INDEX IF NOT EXISTS ix_perms_user ON perms (perm_user_id); // var alterTableAddFilePid = ` -ALTER TABLE files ADD COLUMN file_pid INTEGER +ALTER TABLE files ADD COLUMN file_pid INTEGER; ` var alterTableAddFileMetaPassed = ` -ALTER TABLE files ADD COLUMN file_meta_passed INTEGER +ALTER TABLE files ADD COLUMN file_meta_passed INTEGER; ` var alterTableAddFileMetaFailed = ` -ALTER TABLE files ADD COLUMN file_meta_failed INTEGER +ALTER TABLE files ADD COLUMN file_meta_failed INTEGER; ` var alterTableAddFileMetaSkipped = ` -ALTER TABLE files ADD COLUMN file_meta_skipped INTEGER +ALTER TABLE files ADD COLUMN file_meta_skipped INTEGER; ` var alterTableUpdateFileMeta = ` @@ -635,4 +654,64 @@ UPDATE files SET file_meta_passed=0 ,file_meta_failed=0 ,file_meta_skipped=0 +; +` + +// +// 019_create_table_build_config.sql +// + +var createTableBuildConfig = ` +CREATE TABLE IF NOT EXISTS build_config ( + config_id INTEGER NOT NULL +,build_id INTEGER NOT NULL +,PRIMARY KEY (config_id, build_id) +,FOREIGN KEY (config_id) REFERENCES config (config_id) +,FOREIGN KEY (build_id) REFERENCES builds (build_id) +); +` + +// +// 020_add_column_config_name.sql +// + +var alterTableAddConfigName = ` +ALTER TABLE config ADD COLUMN config_name TEXT +` + +var updateTableSetConfigName = ` +UPDATE config SET config_name = 'drone' +` + +// +// 021_populate_build_config.sql +// + +var populateBuildConfig = ` +INSERT INTO build_config (config_id, build_id) +SELECT build_config_id, build_id FROM builds +` + +// +// 022_add_task_columns.sql +// + +var alterTableAddTaskDependencies = ` +ALTER TABLE tasks ADD COLUMN task_dependencies BYTEA +` + +var alterTableAddTaskRunOn = ` +ALTER TABLE tasks ADD COLUMN task_run_on BYTEA +` + +// +// 023_add_repo_fallback_column.sql +// + +var alterTableAddRepoFallback = ` +ALTER TABLE repos ADD COLUMN repo_fallback BOOLEAN +` + +var updateTableSetRepoFallback = ` +UPDATE repos SET repo_fallback='false' ` diff --git a/store/datastore/ddl/postgres/files/019_create_table_build_config.sql b/store/datastore/ddl/postgres/files/019_create_table_build_config.sql new file mode 100644 index 000000000..be5d58718 --- /dev/null +++ b/store/datastore/ddl/postgres/files/019_create_table_build_config.sql @@ -0,0 +1,9 @@ +-- name: create-table-build-config + +CREATE TABLE IF NOT EXISTS build_config ( + config_id INTEGER NOT NULL +,build_id INTEGER NOT NULL +,PRIMARY KEY (config_id, build_id) +,FOREIGN KEY (config_id) REFERENCES config (config_id) +,FOREIGN KEY (build_id) REFERENCES builds (build_id) +); diff --git a/store/datastore/ddl/postgres/files/020_add_column_config_name.sql b/store/datastore/ddl/postgres/files/020_add_column_config_name.sql new file mode 100644 index 000000000..b1581d11d --- /dev/null +++ b/store/datastore/ddl/postgres/files/020_add_column_config_name.sql @@ -0,0 +1,7 @@ +-- name: alter-table-add-config-name + +ALTER TABLE config ADD COLUMN config_name TEXT + +-- name: update-table-set-config-name + +UPDATE config SET config_name = 'drone' \ No newline at end of file diff --git a/store/datastore/ddl/postgres/files/021_populate_build_config.sql b/store/datastore/ddl/postgres/files/021_populate_build_config.sql new file mode 100644 index 000000000..e004b81c9 --- /dev/null +++ b/store/datastore/ddl/postgres/files/021_populate_build_config.sql @@ -0,0 +1,4 @@ +-- name: populate-build-config + +INSERT INTO build_config (config_id, build_id) +SELECT build_config_id, build_id FROM builds diff --git a/store/datastore/ddl/postgres/files/022_add_task_columns.sql b/store/datastore/ddl/postgres/files/022_add_task_columns.sql new file mode 100644 index 000000000..555e1d882 --- /dev/null +++ b/store/datastore/ddl/postgres/files/022_add_task_columns.sql @@ -0,0 +1,6 @@ +-- name: alter-table-add-task-dependencies +ALTER TABLE tasks ADD COLUMN task_dependencies BYTEA + +-- name: alter-table-add-task-run-on + +ALTER TABLE tasks ADD COLUMN task_run_on BYTEA diff --git a/store/datastore/ddl/postgres/files/023_add_repo_fallback_column.sql b/store/datastore/ddl/postgres/files/023_add_repo_fallback_column.sql new file mode 100644 index 000000000..3e3384854 --- /dev/null +++ b/store/datastore/ddl/postgres/files/023_add_repo_fallback_column.sql @@ -0,0 +1,5 @@ +-- name: alter-table-add-repo-fallback +ALTER TABLE repos ADD COLUMN repo_fallback BOOLEAN + +-- name: update-table-set-repo-fallback +UPDATE repos SET repo_fallback='false' diff --git a/store/datastore/ddl/sqlite/ddl_gen.go b/store/datastore/ddl/sqlite/ddl_gen.go index 3318a9358..d5fe8eb5c 100644 --- a/store/datastore/ddl/sqlite/ddl_gen.go +++ b/store/datastore/ddl/sqlite/ddl_gen.go @@ -1,17 +1,3 @@ -// Copyright 2018 Drone.IO Inc. -// -// 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 sqlite import ( @@ -174,6 +160,38 @@ var migrations = []struct { name: "alter-table-update-file-meta", stmt: alterTableUpdateFileMeta, }, + { + name: "create-table-build-config", + stmt: createTableBuildConfig, + }, + { + name: "alter-table-add-config-name", + stmt: alterTableAddConfigName, + }, + { + name: "update-table-set-config-name", + stmt: updateTableSetConfigName, + }, + { + name: "populate-build-config", + stmt: populateBuildConfig, + }, + { + name: "alter-table-add-task-dependencies", + stmt: alterTableAddTaskDependencies, + }, + { + name: "alter-table-add-task-run-on", + stmt: alterTableAddTaskRunOn, + }, + { + name: "alter-table-add-repo-fallback", + stmt: alterTableAddRepoFallback, + }, + { + name: "update-table-set-repo-fallback", + stmt: updateTableSetRepoFallback, + }, } // Migrate performs the database migration. If the migration fails @@ -637,3 +655,62 @@ UPDATE files SET ,file_meta_failed=0 ,file_meta_skipped=0 ` + +// +// 019_create_table_build_config.sql +// + +var createTableBuildConfig = ` +CREATE TABLE IF NOT EXISTS build_config ( + config_id INTEGER NOT NULL +,build_id INTEGER NOT NULL +,PRIMARY KEY (config_id, build_id) +,FOREIGN KEY (config_id) REFERENCES config (config_id) +,FOREIGN KEY (build_id) REFERENCES builds (build_id) +); +` + +// +// 020_add_column_config_name.sql +// + +var alterTableAddConfigName = ` +ALTER TABLE config ADD COLUMN config_name TEXT +` + +var updateTableSetConfigName = ` +UPDATE config SET config_name = "drone" +` + +// +// 021_populate_build_config.sql +// + +var populateBuildConfig = ` +INSERT INTO build_config (config_id, build_id) +SELECT build_config_id, build_id FROM builds +` + +// +// 022_add_task_columns.sql +// + +var alterTableAddTaskDependencies = ` +ALTER TABLE tasks ADD COLUMN task_dependencies BLOB +` + +var alterTableAddTaskRunOn = ` +ALTER TABLE tasks ADD COLUMN task_run_on BLOB +` + +// +// 023_add_repo_fallback_column.sql +// + +var alterTableAddRepoFallback = ` +ALTER TABLE repos ADD COLUMN repo_fallback BOOLEAN +` + +var updateTableSetRepoFallback = ` +UPDATE repos SET repo_fallback='false' +` diff --git a/store/datastore/ddl/sqlite/files/019_create_table_build_config.sql b/store/datastore/ddl/sqlite/files/019_create_table_build_config.sql new file mode 100644 index 000000000..be5d58718 --- /dev/null +++ b/store/datastore/ddl/sqlite/files/019_create_table_build_config.sql @@ -0,0 +1,9 @@ +-- name: create-table-build-config + +CREATE TABLE IF NOT EXISTS build_config ( + config_id INTEGER NOT NULL +,build_id INTEGER NOT NULL +,PRIMARY KEY (config_id, build_id) +,FOREIGN KEY (config_id) REFERENCES config (config_id) +,FOREIGN KEY (build_id) REFERENCES builds (build_id) +); diff --git a/store/datastore/ddl/sqlite/files/020_add_column_config_name.sql b/store/datastore/ddl/sqlite/files/020_add_column_config_name.sql new file mode 100644 index 000000000..0140b4035 --- /dev/null +++ b/store/datastore/ddl/sqlite/files/020_add_column_config_name.sql @@ -0,0 +1,7 @@ +-- name: alter-table-add-config-name + +ALTER TABLE config ADD COLUMN config_name TEXT + +-- name: update-table-set-config-name + +UPDATE config SET config_name = "drone" \ No newline at end of file diff --git a/store/datastore/ddl/sqlite/files/021_populate_build_config.sql b/store/datastore/ddl/sqlite/files/021_populate_build_config.sql new file mode 100644 index 000000000..e004b81c9 --- /dev/null +++ b/store/datastore/ddl/sqlite/files/021_populate_build_config.sql @@ -0,0 +1,4 @@ +-- name: populate-build-config + +INSERT INTO build_config (config_id, build_id) +SELECT build_config_id, build_id FROM builds diff --git a/store/datastore/ddl/sqlite/files/022_add_task_columns.sql b/store/datastore/ddl/sqlite/files/022_add_task_columns.sql new file mode 100644 index 000000000..01b16d89a --- /dev/null +++ b/store/datastore/ddl/sqlite/files/022_add_task_columns.sql @@ -0,0 +1,6 @@ +-- name: alter-table-add-task-dependencies +ALTER TABLE tasks ADD COLUMN task_dependencies BLOB + +-- name: alter-table-add-task-run-on + +ALTER TABLE tasks ADD COLUMN task_run_on BLOB diff --git a/store/datastore/ddl/sqlite/files/023_add_repo_fallback_column.sql b/store/datastore/ddl/sqlite/files/023_add_repo_fallback_column.sql new file mode 100644 index 000000000..3e3384854 --- /dev/null +++ b/store/datastore/ddl/sqlite/files/023_add_repo_fallback_column.sql @@ -0,0 +1,5 @@ +-- name: alter-table-add-repo-fallback +ALTER TABLE repos ADD COLUMN repo_fallback BOOLEAN + +-- name: update-table-set-repo-fallback +UPDATE repos SET repo_fallback='false' diff --git a/store/datastore/sql/mysql/files/config.sql b/store/datastore/sql/mysql/files/config.sql index 8f29bd5d1..64f63e7a0 100644 --- a/store/datastore/sql/mysql/files/config.sql +++ b/store/datastore/sql/mysql/files/config.sql @@ -1,12 +1,14 @@ -- name: config-find-id SELECT - config_id + config.config_id ,config_repo_id ,config_hash ,config_data +,config_name FROM config -WHERE config_id = ? +LEFT JOIN build_config ON config.config_id = build_config.config_id +WHERE build_config.build_id = ? -- name: config-find-repo-hash @@ -15,6 +17,7 @@ SELECT ,config_repo_id ,config_hash ,config_data +,config_name FROM config WHERE config_repo_id = ? AND config_hash = ? @@ -23,6 +26,10 @@ WHERE config_repo_id = ? SELECT build_id FROM builds WHERE build_repo_id = ? -AND build_config_id = ? +AND build_id in ( + SELECT build_id + FROM build_config + WHERE build_config.config_id = ? + ) AND build_status NOT IN ('blocked', 'pending') LIMIT 1 diff --git a/store/datastore/sql/mysql/files/task.sql b/store/datastore/sql/mysql/files/task.sql index 5b61c7c5e..61890f721 100644 --- a/store/datastore/sql/mysql/files/task.sql +++ b/store/datastore/sql/mysql/files/task.sql @@ -4,6 +4,8 @@ SELECT task_id ,task_data ,task_labels +,task_dependencies +,task_run_on FROM tasks -- name: task-delete diff --git a/store/datastore/sql/mysql/sql_gen.go b/store/datastore/sql/mysql/sql_gen.go index 313a53eb4..73990a460 100644 --- a/store/datastore/sql/mysql/sql_gen.go +++ b/store/datastore/sql/mysql/sql_gen.go @@ -55,12 +55,14 @@ var index = map[string]string{ var configFindId = ` SELECT - config_id + config.config_id ,config_repo_id ,config_hash ,config_data +,config_name FROM config -WHERE config_id = ? +LEFT JOIN build_config ON config.config_id = build_config.config_id +WHERE build_config.build_id = ? ` var configFindRepoHash = ` @@ -69,6 +71,7 @@ SELECT ,config_repo_id ,config_hash ,config_data +,config_name FROM config WHERE config_repo_id = ? AND config_hash = ? @@ -77,7 +80,11 @@ WHERE config_repo_id = ? var configFindApproved = ` SELECT build_id FROM builds WHERE build_repo_id = ? -AND build_config_id = ? +AND build_id in ( + SELECT build_id + FROM build_config + WHERE build_config.config_id = ? + ) AND build_status NOT IN ('blocked', 'pending') LIMIT 1 ` @@ -547,6 +554,8 @@ SELECT task_id ,task_data ,task_labels +,task_dependencies +,task_run_on FROM tasks ` diff --git a/store/datastore/sql/postgres/files/config.sql b/store/datastore/sql/postgres/files/config.sql index 2dc954384..17076138b 100644 --- a/store/datastore/sql/postgres/files/config.sql +++ b/store/datastore/sql/postgres/files/config.sql @@ -1,12 +1,14 @@ -- name: config-find-id SELECT - config_id + config.config_id ,config_repo_id ,config_hash ,config_data +,config_name FROM config -WHERE config_id = $1 +LEFT JOIN build_config ON config.config_id = build_config.config_id +WHERE build_config.build_id = $1 -- name: config-find-repo-hash @@ -15,6 +17,7 @@ SELECT ,config_repo_id ,config_hash ,config_data +,config_name FROM config WHERE config_repo_id = $1 AND config_hash = $2 @@ -23,6 +26,10 @@ WHERE config_repo_id = $1 SELECT build_id FROM builds WHERE build_repo_id = $1 -AND build_config_id = $2 +AND build_id in ( + SELECT build_id + FROM build_config + WHERE build_config.config_id = $2 + ) AND build_status NOT IN ('blocked', 'pending') LIMIT 1 diff --git a/store/datastore/sql/postgres/files/tasks.sql b/store/datastore/sql/postgres/files/tasks.sql index 896c66e51..515ec5e83 100644 --- a/store/datastore/sql/postgres/files/tasks.sql +++ b/store/datastore/sql/postgres/files/tasks.sql @@ -4,6 +4,8 @@ SELECT task_id ,task_data ,task_labels +,task_dependencies +,task_run_on FROM tasks -- name: task-delete diff --git a/store/datastore/sql/postgres/sql_gen.go b/store/datastore/sql/postgres/sql_gen.go index 83bdb729b..84eb1df9a 100644 --- a/store/datastore/sql/postgres/sql_gen.go +++ b/store/datastore/sql/postgres/sql_gen.go @@ -55,12 +55,14 @@ var index = map[string]string{ var configFindId = ` SELECT - config_id + config.config_id ,config_repo_id ,config_hash ,config_data +,config_name FROM config -WHERE config_id = $1 +LEFT JOIN build_config ON config.config_id = build_config.config_id +WHERE build_config.build_id = $1 ` var configFindRepoHash = ` @@ -69,6 +71,7 @@ SELECT ,config_repo_id ,config_hash ,config_data +,config_name FROM config WHERE config_repo_id = $1 AND config_hash = $2 @@ -77,7 +80,11 @@ WHERE config_repo_id = $1 var configFindApproved = ` SELECT build_id FROM builds WHERE build_repo_id = $1 -AND build_config_id = $2 +AND build_id in ( + SELECT build_id + FROM build_config + WHERE build_config.config_id = $2 + ) AND build_status NOT IN ('blocked', 'pending') LIMIT 1 ` @@ -95,7 +102,7 @@ WHERE repo_active = true var countBuilds = ` SELECT count(1) -FROM builds; +FROM builds ` var feedLatestBuild = ` @@ -552,6 +559,8 @@ SELECT task_id ,task_data ,task_labels +,task_dependencies +,task_run_on FROM tasks ` diff --git a/store/datastore/sql/sqlite/files/config.sql b/store/datastore/sql/sqlite/files/config.sql index 8f29bd5d1..64f63e7a0 100644 --- a/store/datastore/sql/sqlite/files/config.sql +++ b/store/datastore/sql/sqlite/files/config.sql @@ -1,12 +1,14 @@ -- name: config-find-id SELECT - config_id + config.config_id ,config_repo_id ,config_hash ,config_data +,config_name FROM config -WHERE config_id = ? +LEFT JOIN build_config ON config.config_id = build_config.config_id +WHERE build_config.build_id = ? -- name: config-find-repo-hash @@ -15,6 +17,7 @@ SELECT ,config_repo_id ,config_hash ,config_data +,config_name FROM config WHERE config_repo_id = ? AND config_hash = ? @@ -23,6 +26,10 @@ WHERE config_repo_id = ? SELECT build_id FROM builds WHERE build_repo_id = ? -AND build_config_id = ? +AND build_id in ( + SELECT build_id + FROM build_config + WHERE build_config.config_id = ? + ) AND build_status NOT IN ('blocked', 'pending') LIMIT 1 diff --git a/store/datastore/sql/sqlite/files/task.sql b/store/datastore/sql/sqlite/files/task.sql index 5b61c7c5e..61890f721 100644 --- a/store/datastore/sql/sqlite/files/task.sql +++ b/store/datastore/sql/sqlite/files/task.sql @@ -4,6 +4,8 @@ SELECT task_id ,task_data ,task_labels +,task_dependencies +,task_run_on FROM tasks -- name: task-delete diff --git a/store/datastore/sql/sqlite/sql_gen.go b/store/datastore/sql/sqlite/sql_gen.go index e98d2e36e..08a2c1b43 100644 --- a/store/datastore/sql/sqlite/sql_gen.go +++ b/store/datastore/sql/sqlite/sql_gen.go @@ -55,12 +55,14 @@ var index = map[string]string{ var configFindId = ` SELECT - config_id + config.config_id ,config_repo_id ,config_hash ,config_data +,config_name FROM config -WHERE config_id = ? +LEFT JOIN build_config ON config.config_id = build_config.config_id +WHERE build_config.build_id = ? ` var configFindRepoHash = ` @@ -69,6 +71,7 @@ SELECT ,config_repo_id ,config_hash ,config_data +,config_name FROM config WHERE config_repo_id = ? AND config_hash = ? @@ -77,7 +80,11 @@ WHERE config_repo_id = ? var configFindApproved = ` SELECT build_id FROM builds WHERE build_repo_id = ? -AND build_config_id = ? +AND build_id in ( + SELECT build_id + FROM build_config + WHERE build_config.config_id = ? + ) AND build_status NOT IN ('blocked', 'pending') LIMIT 1 ` @@ -547,6 +554,8 @@ SELECT task_id ,task_data ,task_labels +,task_dependencies +,task_run_on FROM tasks ` diff --git a/store/store.go b/store/store.go index 0ffd47328..f0849fa2c 100644 --- a/store/store.go +++ b/store/store.go @@ -111,10 +111,11 @@ type Store interface { PermDelete(perm *model.Perm) error PermFlush(user *model.User, before int64) error - ConfigLoad(int64) (*model.Config, error) - ConfigFind(*model.Repo, string) (*model.Config, error) + ConfigsForBuild(buildID int64) ([]*model.Config, error) + ConfigFindIdentical(repoID int64, sha string) (*model.Config, error) ConfigFindApproved(*model.Config) (bool, error) ConfigCreate(*model.Config) error + BuildConfigCreate(*model.BuildConfig) error SenderFind(*model.Repo, string) (*model.Sender, error) SenderList(*model.Repo) ([]*model.Sender, error) diff --git a/vendor/github.com/laszlocph/drone-ui/dist/dist_gen.go b/vendor/github.com/laszlocph/drone-ui/dist/dist_gen.go index 19d52727b..70a053675 100644 --- a/vendor/github.com/laszlocph/drone-ui/dist/dist_gen.go +++ b/vendor/github.com/laszlocph/drone-ui/dist/dist_gen.go @@ -136,20 +136,20 @@ func MustLookup(path string) []byte { // Index of all files var files = map[string]file{ - "/static/bundle.1a4416c2ce69f18a1693.js": { + "/static/bundle.c0660a6c1acce7fadd1d.js": { data: file0, FileInfo: &fileInfo{ - name: "bundle.1a4416c2ce69f18a1693.js", - size: 368620, - modTime: time.Unix(1559709405, 0), + name: "bundle.c0660a6c1acce7fadd1d.js", + size: 366162, + modTime: time.Unix(1562217477, 0), }, }, - "/static/vendor.f67324c74172d8a7f0ba.js": { + "/static/vendor.c03c5c43acf82530bbeb.js": { data: file1, FileInfo: &fileInfo{ - name: "vendor.f67324c74172d8a7f0ba.js", - size: 272277, - modTime: time.Unix(1559709405, 0), + name: "vendor.c03c5c43acf82530bbeb.js", + size: 272274, + modTime: time.Unix(1562217477, 0), }, }, "/favicon.png": { @@ -157,7 +157,7 @@ var files = map[string]file{ FileInfo: &fileInfo{ name: "favicon.png", size: 1374, - modTime: time.Unix(1559709405, 0), + modTime: time.Unix(1562217477, 0), }, }, "/index.html": { @@ -165,7 +165,7 @@ var files = map[string]file{ FileInfo: &fileInfo{ name: "index.html", size: 388, - modTime: time.Unix(1559709405, 0), + modTime: time.Unix(1562217477, 0), }, }, } @@ -174,7 +174,7 @@ var files = map[string]file{ // embedded files. // -// /static/bundle.1a4416c2ce69f18a1693.js +// /static/bundle.c0660a6c1acce7fadd1d.js var file0 = []byte(`webpackJsonp([0],[ /* 0 */, /* 1 */, @@ -307,7 +307,7 @@ var singleton = null; var singletonCounter = 0; var stylesInsertedAtTop = []; -var fixUrls = __webpack_require__(415); +var fixUrls = __webpack_require__(419); module.exports = function(list, options) { if (typeof DEBUG !== "undefined" && DEBUG) { @@ -727,6 +727,8 @@ exports.repositorySlug = exports.compareRepository = exports.disableRepository = var _message = __webpack_require__(67); +var _feed = __webpack_require__(126); + /** * Get the named repository and store the results in * the state tree. @@ -845,6 +847,7 @@ var enableRepository = exports.enableRepository = function enableRepository(tree client.activateRepo(owner, name).then(function (result) { (0, _message.displayMessage)(tree, "Successfully activated your repository"); tree.set(["repos", "data", result.full_name, "active"], true); + (0, _feed.fetchFeed)(tree, client); })["catch"](function () { (0, _message.displayMessage)(tree, "Failed to activate your repository"); }); @@ -863,6 +866,7 @@ var disableRepository = exports.disableRepository = function disableRepository(t client.deleteRepo(owner, name).then(function (result) { (0, _message.displayMessage)(tree, "Successfully disabled your repository"); tree.set(["repos", "data", result.full_name, "active"], false); + (0, _feed.fetchFeed)(tree, client); })["catch"](function () { (0, _message.displayMessage)(tree, "Failed to disabled your repository"); }); @@ -917,59 +921,59 @@ var repositorySlug = exports.repositorySlug = function repositorySlug(owner, nam exports.__esModule = true; exports.TimelapseIcon = exports.TagIcon = exports.SyncIcon = exports.StarIcon = exports.ScheduleIcon = exports.RemoveIcon = exports.RefreshIcon = exports.PlayIcon = exports.PauseIcon = exports.MergeIcon = exports.MenuIcon = exports.LinkIcon = exports.LaunchIcon = exports.ExpandIcon = exports.DeployIcon = exports.CommitIcon = exports.ClockIcon = exports.CloseIcon = exports.CheckIcon = exports.BranchIcon = exports.BackIcon = undefined; -var _back = __webpack_require__(451); +var _back = __webpack_require__(448); var _back2 = _interopRequireDefault(_back); -var _branch = __webpack_require__(452); +var _branch = __webpack_require__(449); var _branch2 = _interopRequireDefault(_branch); -var _check = __webpack_require__(453); +var _check = __webpack_require__(450); var _check2 = _interopRequireDefault(_check); -var _clock = __webpack_require__(454); +var _clock = __webpack_require__(451); var _clock2 = _interopRequireDefault(_clock); -var _close = __webpack_require__(126); +var _close = __webpack_require__(127); var _close2 = _interopRequireDefault(_close); -var _commit = __webpack_require__(455); +var _commit = __webpack_require__(452); var _commit2 = _interopRequireDefault(_commit); -var _deploy = __webpack_require__(456); +var _deploy = __webpack_require__(453); var _deploy2 = _interopRequireDefault(_deploy); -var _expand = __webpack_require__(457); +var _expand = __webpack_require__(454); var _expand2 = _interopRequireDefault(_expand); -var _launch = __webpack_require__(458); +var _launch = __webpack_require__(455); var _launch2 = _interopRequireDefault(_launch); -var _link = __webpack_require__(459); +var _link = __webpack_require__(456); var _link2 = _interopRequireDefault(_link); -var _menu = __webpack_require__(187); +var _menu = __webpack_require__(188); var _menu2 = _interopRequireDefault(_menu); -var _merge = __webpack_require__(460); +var _merge = __webpack_require__(457); var _merge2 = _interopRequireDefault(_merge); -var _pause = __webpack_require__(461); +var _pause = __webpack_require__(458); var _pause2 = _interopRequireDefault(_pause); -var _play = __webpack_require__(462); +var _play = __webpack_require__(459); var _play2 = _interopRequireDefault(_play); @@ -977,27 +981,27 @@ var _refresh = __webpack_require__(189); var _refresh2 = _interopRequireDefault(_refresh); -var _remove = __webpack_require__(463); +var _remove = __webpack_require__(460); var _remove2 = _interopRequireDefault(_remove); -var _schedule = __webpack_require__(464); +var _schedule = __webpack_require__(461); var _schedule2 = _interopRequireDefault(_schedule); -var _star = __webpack_require__(465); +var _star = __webpack_require__(462); var _star2 = _interopRequireDefault(_star); -var _sync = __webpack_require__(466); +var _sync = __webpack_require__(463); var _sync2 = _interopRequireDefault(_sync); -var _tag = __webpack_require__(467); +var _tag = __webpack_require__(464); var _tag2 = _interopRequireDefault(_tag); -var _timelapse = __webpack_require__(468); +var _timelapse = __webpack_require__(465); var _timelapse2 = _interopRequireDefault(_timelapse); @@ -1108,7 +1112,7 @@ var hideMessage = exports.hideMessage = function hideMessage(tree) { exports.__esModule = true; -exports.StatusLabel = exports["default"] = undefined; +exports.StatusText = exports.StatusLabel = exports["default"] = undefined; var _react = __webpack_require__(1); @@ -1120,7 +1124,7 @@ var _classnames2 = _interopRequireDefault(_classnames); var _status = __webpack_require__(87); -var _status2 = __webpack_require__(449); +var _status2 = __webpack_require__(446); var _status3 = _interopRequireDefault(_status2); @@ -1222,6 +1226,24 @@ var StatusLabel = exports.StatusLabel = function StatusLabel(_ref) { ); }; +var StatusText = exports.StatusText = function StatusText(_ref2) { + var status = _ref2.status, + text = _ref2.text; + + return _react2["default"].createElement( + "div", + { + className: (0, _classnames2["default"])(_status3["default"].label, _status3["default"][status]), + style: "text-transform: capitalize;padding: 5px 10px;" + }, + _react2["default"].createElement( + "div", + null, + text + ) + ); +}; + /***/ }), /* 87 */ /***/ (function(module, exports, __webpack_require__) { @@ -1297,6 +1319,105 @@ exports.STATUS_STARTED = STATUS_STARTED; "use strict"; +exports.__esModule = true; +exports.fetchFeedOnce = fetchFeedOnce; +exports.subscribeToFeedOnce = subscribeToFeedOnce; +/** + * Get the event feed and store the results in the + * state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + */ +var fetchFeed = exports.fetchFeed = function fetchFeed(tree, client) { + client.getBuildFeed({ latest: true }).then(function (results) { + var list = {}; + var sorted = results.sort(compareFeedItem); + sorted.map(function (repo) { + list[repo.full_name] = repo; + }); + if (sorted && sorted.length > 0) { + tree.set(["feed", "latest"], sorted[0]); + } + tree.set(["feed", "loaded"], true); + tree.set(["feed", "data"], list); + })["catch"](function (error) { + tree.set(["feed", "loaded"], true); + tree.set(["feed", "error"], error); + }); +}; + +/** + * Ensures the fetchFeed function is invoked exactly once. + * TODO replace this with a decorator + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + */ +function fetchFeedOnce(tree, client) { + if (fetchFeedOnce.fired) { + return; + } + fetchFeedOnce.fired = true; + return fetchFeed(tree, client); +} + +/** + * Subscribes to the server-side event feed and synchonizes + * event data with the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + */ +var subscribeToFeed = exports.subscribeToFeed = function subscribeToFeed(tree, client) { + return client.on(function (data) { + var repo = data.repo, + build = data.build; + + + if (tree.exists("feed", "data", repo.full_name)) { + var cursor = tree.select(["feed", "data", repo.full_name]); + cursor.merge(build); + } + + if (tree.exists("builds", "data", repo.full_name)) { + tree.set(["builds", "data", repo.full_name, build.number], build); + } + }); +}; + +/** + * Ensures the subscribeToFeed function is invoked exactly once. + * TODO replace this with a decorator + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + */ +function subscribeToFeedOnce(tree, client) { + if (subscribeToFeedOnce.fired) { + return; + } + subscribeToFeedOnce.fired = true; + return subscribeToFeed(tree, client); +} + +/** + * Compare two feed items by name. + * @param {Object} a - A feed item. + * @param {Object} b - A feed item. + * @returns {number} + */ +var compareFeedItem = exports.compareFeedItem = function compareFeedItem(a, b) { + return (b.started_at || b.created_at || -1) - (a.started_at || a.created_at || -1); +}; + +/***/ }), +/* 127 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + exports.__esModule = true; exports["default"] = undefined; @@ -1341,7 +1462,7 @@ var CloseIcon = function (_Component) { exports["default"] = CloseIcon; /***/ }), -/* 127 */ +/* 128 */ /***/ (function(module, exports, __webpack_require__) { "use strict"; @@ -1360,11 +1481,11 @@ var _reactTimeago = __webpack_require__(190); var _reactTimeago2 = _interopRequireDefault(_reactTimeago); -var _duration = __webpack_require__(471); +var _duration = __webpack_require__(468); var _duration2 = _interopRequireDefault(_duration); -var _build_time = __webpack_require__(472); +var _build_time = __webpack_require__(469); var _build_time2 = _interopRequireDefault(_build_time); @@ -1438,7 +1559,7 @@ var Runtime = function (_Component) { exports["default"] = Runtime; /***/ }), -/* 128 */ +/* 129 */ /***/ (function(module, exports, __webpack_require__) { "use strict"; @@ -1622,7 +1743,7 @@ var assertBuildMatrix = exports.assertBuildMatrix = function assertBuildMatrix(b }; /***/ }), -/* 129 */ +/* 130 */ /***/ (function(module, exports, __webpack_require__) { "use strict"; @@ -1637,7 +1758,7 @@ var _react2 = _interopRequireDefault(_react); var _index = __webpack_require__(40); -var _breadcrumb = __webpack_require__(525); +var _breadcrumb = __webpack_require__(522); var _breadcrumb2 = _interopRequireDefault(_breadcrumb); @@ -1689,7 +1810,7 @@ var Breadcrumb = function (_Component) { exports["default"] = Breadcrumb; /***/ }), -/* 130 */ +/* 131 */ /***/ (function(module, exports, __webpack_require__) { "use strict"; @@ -1746,7 +1867,6 @@ var assertProcRunning = exports.assertProcRunning = function assertProcRunning(p }; /***/ }), -/* 131 */, /* 132 */, /* 133 */, /* 134 */, @@ -1802,7 +1922,8 @@ var assertProcRunning = exports.assertProcRunning = function assertProcRunning(p /* 184 */, /* 185 */, /* 186 */, -/* 187 */ +/* 187 */, +/* 188 */ /***/ (function(module, exports, __webpack_require__) { "use strict"; @@ -1851,105 +1972,6 @@ var MenuIcon = function (_Component) { exports["default"] = MenuIcon; -/***/ }), -/* 188 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; - - -exports.__esModule = true; -exports.fetchFeedOnce = fetchFeedOnce; -exports.subscribeToFeedOnce = subscribeToFeedOnce; -/** - * Get the event feed and store the results in the - * state tree. - * - * @param {Object} tree - The drone state tree. - * @param {Object} client - The drone client. - */ -var fetchFeed = exports.fetchFeed = function fetchFeed(tree, client) { - client.getBuildFeed({ latest: true }).then(function (results) { - var list = {}; - var sorted = results.sort(compareFeedItem); - sorted.map(function (repo) { - list[repo.full_name] = repo; - }); - if (sorted && sorted.length > 0) { - tree.set(["feed", "latest"], sorted[0]); - } - tree.set(["feed", "loaded"], true); - tree.set(["feed", "data"], list); - })["catch"](function (error) { - tree.set(["feed", "loaded"], true); - tree.set(["feed", "error"], error); - }); -}; - -/** - * Ensures the fetchFeed function is invoked exactly once. - * TODO replace this with a decorator - * - * @param {Object} tree - The drone state tree. - * @param {Object} client - The drone client. - */ -function fetchFeedOnce(tree, client) { - if (fetchFeedOnce.fired) { - return; - } - fetchFeedOnce.fired = true; - return fetchFeed(tree, client); -} - -/** - * Subscribes to the server-side event feed and synchonizes - * event data with the state tree. - * - * @param {Object} tree - The drone state tree. - * @param {Object} client - The drone client. - */ -var subscribeToFeed = exports.subscribeToFeed = function subscribeToFeed(tree, client) { - return client.on(function (data) { - var repo = data.repo, - build = data.build; - - - if (tree.exists("feed", "data", repo.full_name)) { - var cursor = tree.select(["feed", "data", repo.full_name]); - cursor.merge(build); - } - - if (tree.exists("builds", "data", repo.full_name)) { - tree.set(["builds", "data", repo.full_name, build.number], build); - } - }); -}; - -/** - * Ensures the subscribeToFeed function is invoked exactly once. - * TODO replace this with a decorator - * - * @param {Object} tree - The drone state tree. - * @param {Object} client - The drone client. - */ -function subscribeToFeedOnce(tree, client) { - if (subscribeToFeedOnce.fired) { - return; - } - subscribeToFeedOnce.fired = true; - return subscribeToFeed(tree, client); -} - -/** - * Compare two feed items by name. - * @param {Object} a - A feed item. - * @param {Object} b - A feed item. - * @returns {number} - */ -var compareFeedItem = exports.compareFeedItem = function compareFeedItem(a, b) { - return (b.started_at || b.created_at || -1) - (a.started_at || a.created_at || -1); -}; - /***/ }), /* 189 */ /***/ (function(module, exports, __webpack_require__) { @@ -2038,7 +2060,7 @@ var _index = __webpack_require__(40); var _events = __webpack_require__(192); -var _build_event = __webpack_require__(511); +var _build_event = __webpack_require__(508); var _build_event2 = _interopRequireDefault(_build_event); @@ -2148,7 +2170,7 @@ var _reactRouterDom = __webpack_require__(27); var _higherOrder = __webpack_require__(16); -var _sync = __webpack_require__(533); +var _sync = __webpack_require__(530); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; } @@ -2179,7 +2201,7 @@ var RedirectRoot = (_dec = (0, _higherOrder.branch)(binding), _dec(_class = func var user = nextProps.user; if (!user && window) { - window.location.href = "/login"; + window.location.href = "/login?url=" + window.location.href; } }; @@ -2219,7 +2241,7 @@ var _propTypes = __webpack_require__(12); var _propTypes2 = _interopRequireDefault(_propTypes); -var _menu = __webpack_require__(538); +var _menu = __webpack_require__(535); var _menu2 = _interopRequireDefault(_menu); @@ -2343,7 +2365,7 @@ exports["default"] = RepoMenu; "use strict"; -__webpack_require__(132); +__webpack_require__(133); var _react = __webpack_require__(1); @@ -2595,11 +2617,11 @@ var _inject = __webpack_require__(22); var _screens = __webpack_require__(411); -var _titles = __webpack_require__(423); +var _titles = __webpack_require__(420); var _titles2 = _interopRequireDefault(_titles); -var _layout = __webpack_require__(444); +var _layout = __webpack_require__(441); var _layout2 = _interopRequireDefault(_layout); @@ -2607,11 +2629,11 @@ var _redirect = __webpack_require__(194); var _redirect2 = _interopRequireDefault(_redirect); -var _feed = __webpack_require__(188); +var _feed = __webpack_require__(126); var _reactRouterDom = __webpack_require__(27); -var _drone = __webpack_require__(582); +var _drone = __webpack_require__(579); var _drone2 = _interopRequireDefault(_drone); @@ -2651,7 +2673,6 @@ var App = function (_Component) { _reactRouterDom.Switch, null, _react2["default"].createElement(_reactRouterDom.Route, { path: "/", exact: true, component: _redirect2["default"] }), - _react2["default"].createElement(_reactRouterDom.Route, { path: "/login/form", exact: true, component: _screens.LoginForm }), _react2["default"].createElement(_reactRouterDom.Route, { path: "/login/error", exact: true, component: _screens.LoginError }), _react2["default"].createElement(_reactRouterDom.Route, { path: "/", exact: false, component: _layout2["default"] }) ) @@ -2779,7 +2800,7 @@ exports["default"] = tree; exports.__esModule = true; -var _droneJs = __webpack_require__(171); +var _droneJs = __webpack_require__(172); var _droneJs2 = _interopRequireDefault(_droneJs); @@ -2795,19 +2816,14 @@ exports["default"] = _droneJs2["default"].fromWindow(); exports.__esModule = true; -exports.LoginError = exports.LoginForm = undefined; +exports.LoginError = undefined; -var _form = __webpack_require__(412); - -var _form2 = _interopRequireDefault(_form); - -var _error = __webpack_require__(416); +var _error = __webpack_require__(412); var _error2 = _interopRequireDefault(_error); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; } -exports.LoginForm = _form2["default"]; exports.LoginError = _error2["default"]; /***/ }), @@ -2823,46 +2839,129 @@ var _react = __webpack_require__(1); var _react2 = _interopRequireDefault(_react); -var _index = __webpack_require__(413); +var _queryString = __webpack_require__(173); + +var _queryString2 = _interopRequireDefault(_queryString); + +var _report = __webpack_require__(416); + +var _report2 = _interopRequireDefault(_report); + +var _index = __webpack_require__(417); var _index2 = _interopRequireDefault(_index); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; } -var LoginForm = function LoginForm(props) { - return _react2["default"].createElement( - "div", - { className: _index2["default"].login }, - _react2["default"].createElement( - "form", - { method: "post", action: "/authorize" }, - _react2["default"].createElement( - "p", - null, - "Login with your version control system username and password." - ), - _react2["default"].createElement("input", { - placeholder: "Username", - name: "username", - type: "text", - spellCheck: "false" - }), - _react2["default"].createElement("input", { placeholder: "Password", name: "password", type: "password" }), - _react2["default"].createElement("input", { value: "Login", type: "submit" }) - ) - ); -}; +function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } -exports["default"] = LoginForm; +function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } + +function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } + +var DEFAULT_ERROR = "The system failed to process your Login request."; + +var Error = function (_Component) { + _inherits(Error, _Component); + + function Error() { + _classCallCheck(this, Error); + + return _possibleConstructorReturn(this, _Component.apply(this, arguments)); + } + + Error.prototype.render = function render() { + var parsed = _queryString2["default"].parse(window.location.search); + var error = DEFAULT_ERROR; + + switch (parsed.code || parsed.error) { + case "oauth_error": + break; + case "access_denied": + break; + } + + return _react2["default"].createElement( + "div", + { className: _index2["default"].root }, + _react2["default"].createElement( + "div", + { className: _index2["default"].alert }, + _react2["default"].createElement( + "div", + null, + _react2["default"].createElement(_report2["default"], null) + ), + _react2["default"].createElement( + "div", + null, + error + ) + ) + ); + }; + + return Error; +}(_react.Component); + +exports["default"] = Error; /***/ }), -/* 413 */ +/* 413 */, +/* 414 */, +/* 415 */, +/* 416 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +exports.__esModule = true; +exports["default"] = undefined; + +var _react = __webpack_require__(1); + +var _react2 = _interopRequireDefault(_react); + +function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; } + +function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } + +function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } + +function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } + +var ReportIcon = function (_Component) { + _inherits(ReportIcon, _Component); + + function ReportIcon() { + _classCallCheck(this, ReportIcon); + + return _possibleConstructorReturn(this, _Component.apply(this, arguments)); + } + + ReportIcon.prototype.render = function render() { + return _react2["default"].createElement( + "svg", + { className: this.props.className, viewBox: "0 0 24 24" }, + _react2["default"].createElement("path", { d: "M15.73 3H8.27L3 8.27v7.46L8.27 21h7.46L21 15.73V8.27L15.73 3zM12 17.3c-.72 0-1.3-.58-1.3-1.3 0-.72.58-1.3 1.3-1.3.72 0 1.3.58 1.3 1.3 0 .72-.58 1.3-1.3 1.3zm1-4.3h-2V7h2v6z" }), + _react2["default"].createElement("path", { d: "M0 0h24v24H0z", fill: "none" }) + ); + }; + + return ReportIcon; +}(_react.Component); + +exports["default"] = ReportIcon; + +/***/ }), +/* 417 */ /***/ (function(module, exports, __webpack_require__) { // style-loader: Adds some css to the DOM by adding a