2019-03-08 15:33:03 -05:00
---
version : 2
2019-01-28 11:18:30 -05:00
2019-03-08 15:33:03 -05:00
references :
images :
2019-03-26 17:04:58 -04:00
go : &GOLANG_IMAGE circleci/golang:1.12.1
2019-03-08 15:33:03 -05:00
middleman : &MIDDLEMAN_IMAGE hashicorp/middleman-hashicorp:0.3.35
ember : &EMBER_IMAGE circleci/node:8-browsers
paths :
test-results : &TEST_RESULTS_DIR /tmp/test-results
2019-01-28 11:18:30 -05:00
2019-03-08 15:33:03 -05:00
cache :
yarn : &YARN_CACHE_KEY consul-ui-v1-{{ checksum "ui-v2/yarn.lock" }}
rubygem : &RUBYGEM_CACHE_KEY static-site-gems-v1-{{ checksum "Gemfile.lock" }}
environment : &ENVIRONMENT
TEST_RESULTS_DIR : *TEST_RESULTS_DIR
GOTESTSUM_RELEASE : 0.3 .3
EMAIL : noreply@hashicorp.com
GIT_AUTHOR_NAME : circleci-consul
GIT_COMMITTER_NAME : circleci-consul
2019-04-02 12:24:48 -04:00
S3_ARTIFACT_BUCKET : consul-dev-artifacts
2019-01-28 11:18:30 -05:00
2018-11-16 14:01:02 -05:00
jobs :
2019-07-16 18:52:24 -04:00
# lint consul tests
lint-consul-retry :
docker :
- image : *GOLANG_IMAGE
steps :
- checkout
- run : go get -u github.com/hashicorp/lint-consul-retry && lint-consul-retry
2019-04-25 12:26:33 -04:00
# Runs go fmt and go vet
go-fmt-and-vet :
docker :
2019-05-17 11:42:56 -04:00
- image : *GOLANG_IMAGE
2019-04-25 12:26:33 -04:00
steps :
2019-05-17 11:42:56 -04:00
- checkout
- restore_cache :
keys :
- consul-modcache-v1-{{ checksum "go.mod" }}
- run :
command : go mod download
- save_cache :
key : consul-modcache-v1-{{ checksum "go.mod" }}
paths :
- /go/pkg/mod
- run :
name : check go fmt
command : |
files=$(go fmt ./...)
if [ -n "$files" ]; then
echo "The following file(s) do not conform to go fmt:"
echo "$files"
exit 1
fi
- run :
command : go vet ./...
environment :
<< : *ENVIRONMENT
go-test :
docker :
- image : *GOLANG_IMAGE
2019-06-20 10:49:40 -04:00
parallelism : 8
2019-05-17 11:42:56 -04:00
environment :
<< : *ENVIRONMENT
GOTAGS : '' # No tags for OSS but there are for enterprise
steps :
- checkout
- restore_cache : # restore cache from dev-build job
keys :
- consul-modcache-v1-{{ checksum "go.mod" }}
- attach_workspace :
at : /go/bin
- run : mkdir -p $TEST_RESULTS_DIR
- run : sudo apt-get update && sudo apt-get install -y rsyslog
- run : sudo service rsyslog start
- run : |
2019-07-22 14:01:22 -04:00
PACKAGE_NAMES=$(go list ./... | grep -v github.com/hashicorp/consul/agent/proxyprocess | circleci tests split --split-by=timings --timings-type=classname)
gotestsum --format=short-verbose --junitfile $TEST_RESULTS_DIR/gotestsum-report.xml -- -tags=$GOTAGS -p 3 $PACKAGE_NAMES
2019-05-17 11:42:56 -04:00
- store_test_results :
path : /tmp/test-results
- store_artifacts :
path : /tmp/test-results
# split off a job for the API package since it is separate
go-test-api :
docker :
- image : *GOLANG_IMAGE
2019-04-25 12:26:33 -04:00
environment :
<< : *ENVIRONMENT
2019-05-17 11:42:56 -04:00
GOTAGS : '' # No tags for OSS but there are for enterprise
steps :
- checkout
- restore_cache : # restore cache from dev-build job
keys :
- consul-modcache-v1-{{ checksum "go.mod" }}
- attach_workspace :
at : /go/bin
- run : mkdir -p $TEST_RESULTS_DIR
- run :
working_directory : api
command : |
2019-07-22 14:01:22 -04:00
PACKAGE_NAMES=$(go list ./... | circleci tests split --split-by=timings --timings-type=classname)
gotestsum --format=short-verbose --junitfile $TEST_RESULTS_DIR/gotestsum-report.xml -- -tags=$GOTAGS $PACKAGE_NAMES
2019-05-17 11:42:56 -04:00
- store_test_results :
path : /tmp/test-results
- store_artifacts :
path : /tmp/test-results
2019-04-25 12:26:33 -04:00
2019-03-20 15:24:17 -04:00
# build all distros
build-distros : &build-distros
2019-03-08 15:33:03 -05:00
docker :
- image : *GOLANG_IMAGE
2019-03-20 15:24:17 -04:00
environment : &build-env
GOXPARALLEL : 2 # CircleCI containers are 2 CPU x 4GB RAM
2019-06-20 10:49:40 -04:00
resource_class : large
2019-01-28 11:18:30 -05:00
steps :
- checkout
2019-03-20 15:24:17 -04:00
- run : make tools
- run : ./build-support/scripts/build-local.sh
2019-01-28 11:18:30 -05:00
# save dev build to CircleCI
- store_artifacts :
2019-03-20 15:24:17 -04:00
path : ./pkg/bin
2019-04-02 12:24:48 -04:00
# build all 386 architecture supported OS binaries
2019-03-20 15:24:17 -04:00
build-386 :
<< : *build-distros
environment :
<< : *build-env
XC_OS : "darwin freebsd linux windows"
XC_ARCH : "386"
2019-04-02 12:24:48 -04:00
# build all amd64 architecture supported OS binaries
2019-03-20 15:24:17 -04:00
build-amd64 :
<< : *build-distros
environment :
<< : *build-env
XC_OS : "darwin freebsd linux solaris windows"
XC_ARCH : "amd64"
2019-04-02 12:24:48 -04:00
# build all arm/arm64 architecture supported OS binaries
2019-03-20 15:24:17 -04:00
build-arm-arm64 :
<< : *build-distros
environment :
<< : *build-env
XC_OS : linux
XC_ARCH : "arm arm64"
2019-04-02 12:24:48 -04:00
# create a development build
2019-03-20 15:24:17 -04:00
dev-build :
docker :
- image : *GOLANG_IMAGE
steps :
- checkout
2019-05-17 11:42:56 -04:00
- restore_cache :
keys :
- consul-modcache-v1-{{ checksum "go.mod" }}
- run :
command : make dev
- save_cache :
key : consul-modcache-v1-{{ checksum "go.mod" }}
paths :
- /go/pkg/mod
2019-01-28 11:18:30 -05:00
# save dev build to pass to downstream jobs
- persist_to_workspace :
root : /go/bin
paths :
- consul
2019-04-02 12:24:48 -04:00
# upload development build to s3
dev-upload-s3 :
docker :
- image : circleci/python:stretch
environment :
<< : *ENVIRONMENT
steps :
- run :
name : Install awscli
command : sudo pip install awscli
# get consul binary
- attach_workspace :
at : bin/
- run :
name : package binary
command : tar -czf consul.tar.gz -C bin/ .
- run :
name : Upload to s3
command : |
2019-04-04 16:03:07 -04:00
if [ -n "${S3_ARTIFACT_PATH}" ]; then
aws s3 cp \
--metadata "CIRCLECI=${CIRCLECI},CIRCLE_BUILD_URL=${CIRCLE_BUILD_URL},CIRCLE_BRANCH=${CIRCLE_BRANCH}" \
"consul.tar.gz" "s3://${S3_ARTIFACT_BUCKET}/${S3_ARTIFACT_PATH}/${CIRCLE_SHA1}.tar.gz"
else
echo "CircleCI - S3_ARTIFACT_PATH was not set"
exit 1
fi
2019-04-02 12:24:48 -04:00
2019-01-28 11:18:30 -05:00
# Nomad 0.8 builds on go0.10
# Run integration tests on nomad/v0.8.7
nomad-integration-0_8 :
2019-03-08 15:33:03 -05:00
docker :
- image : circleci/golang:1.10
environment :
<< : *ENVIRONMENT
NOMAD_WORKING_DIR : &NOMAD_WORKING_DIR /go/src/github.com/hashicorp/nomad
NOMAD_VERSION : v0.8.7
steps : &NOMAD_INTEGRATION_TEST_STEPS
- run : git clone https://github.com/hashicorp/nomad.git --branch ${NOMAD_VERSION} ${NOMAD_WORKING_DIR}
# get consul binary
- attach_workspace :
at : /go/bin
# make test result directory
- run : mkdir -p $TEST_RESULTS_DIR
# make dev build of nomad
- run :
command : make dev
working_directory : *NOMAD_WORKING_DIR
# update gotestsum
2019-03-26 17:04:58 -04:00
- run : curl -sSL "https://github.com/gotestyourself/gotestsum/releases/download/v${GOTESTSUM_RELEASE}/gotestsum_${GOTESTSUM_RELEASE}_linux_amd64.tar.gz" | sudo tar --overwrite -xz -C /usr/local/bin gotestsum
2019-03-08 15:33:03 -05:00
# run integration tests
- run :
command : gotestsum --format=short-verbose --junitfile $TEST_RESULTS_DIR/results.xml -- ./command/agent/consul -run TestConsul
working_directory : *NOMAD_WORKING_DIR
# store test results for CircleCI
- store_test_results :
path : *TEST_RESULTS_DIR
- store_artifacts :
path : *TEST_RESULTS_DIR
2019-01-28 11:18:30 -05:00
# run integration tests on nomad/master
nomad-integration-master :
2019-03-08 15:33:03 -05:00
docker :
- image : *GOLANG_IMAGE
environment :
<< : *ENVIRONMENT
NOMAD_WORKING_DIR : /go/src/github.com/hashicorp/nomad
NOMAD_VERSION : master
steps : *NOMAD_INTEGRATION_TEST_STEPS
2019-01-28 11:18:30 -05:00
2018-11-16 14:01:02 -05:00
build-website :
# setting the working_directory along with the checkout path allows us to not have
# to cd into the website/ directory for commands
working_directory : ~/project/website
docker :
2019-03-08 15:33:03 -05:00
- image : *MIDDLEMAN_IMAGE
2018-11-16 14:01:02 -05:00
steps :
- checkout :
path : ~/project
# restores gem cache
- restore_cache :
2019-03-08 15:33:03 -05:00
key : *RUBYGEM_CACHE_KEY
2018-11-16 14:01:02 -05:00
- run :
name : install gems
command : bundle check || bundle install --path vendor/bundle --retry=3
# saves gem cache if we have changed the Gemfile
- save_cache :
2019-03-08 15:33:03 -05:00
key : *RUBYGEM_CACHE_KEY
2018-11-16 14:01:02 -05:00
paths :
- ~/project/website/vendor/bundle
2019-06-14 16:55:18 -04:00
# exclude guides directory since they moved to learn.hashicorp.com
# keep index.html which points to learn
- run :
name : exclude guides
command : find ./source/docs/guides -type f -not -name 'index.html.md' -delete
2018-11-16 14:01:02 -05:00
- run :
name : middleman build
command : bundle exec middleman build
2019-04-04 18:54:47 -04:00
# saves website build directory
- persist_to_workspace :
root : .
paths :
- build
deploy-website :
# setting the working_directory along with the checkout path allows us to not have
# to cd into the website/ directory for commands
working_directory : ~/project/website
docker :
- image : *MIDDLEMAN_IMAGE
steps :
- checkout :
path : ~/project
2019-04-25 12:24:20 -04:00
2019-04-04 18:54:47 -04:00
# attach website build directory
- attach_workspace :
at : ~/project/website
2019-04-25 12:24:20 -04:00
# restores gem cache
- restore_cache :
key : *RUBYGEM_CACHE_KEY
# rerun build with 'ENV=production' to add analytics
- run :
name : install gems
command : bundle check || bundle install --path vendor/bundle --retry=3
2019-06-14 16:55:18 -04:00
# exclude guides directory since they moved to learn.hashicorp.com
# keep index.html which points to learn
- run :
name : exclude guides
command : find ./source/docs/guides -type f -not -name 'index.html.md' -delete
2019-04-25 12:24:20 -04:00
# rerun build with 'ENV=production' to add analytics
- run :
name : middleman build
command : bundle exec middleman build
2018-11-16 14:01:02 -05:00
- run :
name : website deploy
command : ./scripts/deploy.sh
2019-04-04 18:54:47 -04:00
# Link check on a temporary netlify deployed site
docs-link-checker :
docker :
- image : circleci/node:lts
steps :
- checkout
# attach website build directory
- attach_workspace :
at : ~/project/website
- run : ./website/scripts/link-check.sh
2019-01-03 22:57:40 -05:00
# build frontend yarn cache
2018-12-20 13:51:48 -05:00
frontend-cache :
docker :
2019-03-08 15:33:03 -05:00
- image : *EMBER_IMAGE
2018-12-20 13:51:48 -05:00
steps :
- checkout
2019-03-08 15:33:03 -05:00
2018-12-20 13:51:48 -05:00
# cache yarn deps
- restore_cache :
2019-03-08 15:33:03 -05:00
key : *YARN_CACHE_KEY
2018-12-20 13:51:48 -05:00
- run :
name : install yarn packages
command : cd ui-v2 && yarn install
2019-03-08 15:33:03 -05:00
2018-12-20 13:51:48 -05:00
- save_cache :
2019-03-08 15:33:03 -05:00
key : *YARN_CACHE_KEY
2018-12-20 13:51:48 -05:00
paths :
- ui-v2/node_modules
2019-03-08 15:33:03 -05:00
2018-12-20 13:51:48 -05:00
# build ember so frontend tests run faster
ember-build :
docker :
2019-03-08 15:33:03 -05:00
- image : *EMBER_IMAGE
2018-12-20 13:51:48 -05:00
steps :
- checkout
- restore_cache :
2019-03-08 15:33:03 -05:00
key : *YARN_CACHE_KEY
2019-01-03 12:56:54 -05:00
- run : cd ui-v2 && yarn build-ci --output-path=dist
2018-12-20 13:51:48 -05:00
# saves the build to a workspace to be passed to a downstream job
- persist_to_workspace :
root : ui-v2
paths :
- dist
2019-01-03 22:57:40 -05:00
# run ember frontend tests
2018-12-20 13:51:48 -05:00
ember-test :
docker :
2019-03-08 15:33:03 -05:00
- image : *EMBER_IMAGE
2018-12-20 13:51:48 -05:00
environment :
EMBER_TEST_PARALLEL : true #enables test parallelization with ember-exam
EMBER_TEST_REPORT : test-results/report.xml #outputs test report for CircleCI test summary
parallelism : 4
steps :
- checkout
- restore_cache :
2019-03-08 15:33:03 -05:00
key : *YARN_CACHE_KEY
2018-12-20 13:51:48 -05:00
- attach_workspace :
at : ui-v2
- run :
working_directory : ui-v2
command : node_modules/ember-cli/bin/ember exam --split=$CIRCLE_NODE_TOTAL --partition=`expr $CIRCLE_NODE_INDEX + 1` --path dist --silent -r xunit
- store_test_results :
path : ui-v2/test-results
2019-04-29 17:27:57 +01:00
# Envoy integration tests. Require docker dev binary to be built already
envoy-integration-test-1.8.0 :
docker :
# We only really need bash and docker-compose which is installed on all
# Circle images but pick Go since we have to pick one of them.
- image : *GOLANG_IMAGE
environment :
ENVOY_VERSIONS : "1.8.0"
steps :
&ENVOY_INTEGRATION_TEST_STEPS
- checkout
# Get go binary from workspace
- attach_workspace :
at : .
- setup_remote_docker :
docker_layer_caching : true
# Build the consul-dev image from the already built binary
- run : docker build -t consul-dev -f ./build-support/docker/Consul-Dev.dockerfile .
- run :
name : Envoy Integration Tests
command : make test-envoy-integ SKIP_DOCKER_BUILD=1
environment :
# tput complains if this isn't set to something.
TERM : ansi
- store_artifacts :
path : ./test/integration/connect/envoy/workdir/logs
destination : container-logs
envoy-integration-test-1.9.1 :
docker :
- image : *GOLANG_IMAGE
environment :
ENVOY_VERSIONS : "1.9.1"
steps : *ENVOY_INTEGRATION_TEST_STEPS
2019-06-07 07:10:43 -05:00
envoy-integration-test-1.10.0 :
docker :
- image : *GOLANG_IMAGE
environment :
ENVOY_VERSIONS : "1.10.0"
steps : *ENVOY_INTEGRATION_TEST_STEPS
2019-06-18 15:56:58 -04:00
# This job merges master into the 'current' branch (${CIRCLE_BRANCH}) specified
# in the branch filtering of the workflow
merge-master :
docker :
- image : *GOLANG_IMAGE
steps :
- add_ssh_keys :
fingerprints :
- c6:96:98:82:dc:04:6c:39:dd:ac:83:05:e3:15:1c:98
- checkout
- run :
name : Merge Consul OSS master branch into current branch
command : |
set -eu -o pipefail
# Configure Git
git config --global user.email "hashicorp-ci@users.noreply.github.com"
git config --global user.name "hashicorp-ci"
# Fetch latest master
git fetch origin
# Create a merge branch to run tests on
git_merge_branch="ci/master-merge-$(date +%Y%m%d%H%M%S)"
git checkout -b "${git_merge_branch}"
latest_oss_commit="$(git rev-parse origin/master)"
2019-07-23 20:18:36 -04:00
if ! errors=$(git merge -m "Merge Consul OSS branch 'master' at commit ${latest_oss_commit}" "${latest_oss_commit}"); then
printf "oss/master merge into ${CIRCLE_BRANCH} failed because git was unable to auto-merge!\n${errors}"
curl -X POST -H 'Content-type : application/json' \
--data \
"{ \
\"attachments\": [ \
{ \
\"fallback\": \"master merge into ${CIRCLE_BRANCH} failed because git was unable to auto-merge!\", \
\"text\": \"Nightly *master* merge into *${CIRCLE_BRANCH}* failed!\n\nBuild Log: ${CIRCLE_BUILD_URL}\n\nGit was unable to auto-merge due to possible merge conflict.\n\n*Errors:*\n${errors}\", \
\"footer\": \"${CIRCLE_PROJECT_USERNAME}/${CIRCLE_PROJECT_REPONAME}\", \
\"ts\": \"$(date +%s)\", \
\"color\": \"danger\" \
} \
] \
}" ${CONSUL_SLACK_WEBHOOK_URL}
exit 1
fi
2019-06-18 15:56:58 -04:00
git push origin "${git_merge_branch}"
sleep 15 # Wait for merge branch to start CircleCI pipeline
# Wait for OSS merge branch CircleCI pipeline to finish
# return shallow results for better performance
project_url="https://circleci.com/api/v1.1/project/github/hashicorp/consul/tree/${git_merge_branch}?shallow=true"
echo "Waiting for master merge branch CI pipeline to finish..."
builds=
unfinished_builds="will be populated in the until loop below"
min_waited=0
until [[ -z "${unfinished_builds}" ]]; do
builds="$(curl \
--header 'Accept : application/json' \
--show-error \
--silent \
"${project_url}" )"
unfinished_builds="$(echo "${builds}" \
| jq --raw-output '.[] | select(.lifecycle!="finished") | .workflows.job_name')"
sleep 60
let "min_waited += 1"
echo "Waited ${min_waited} min..."
done
# Fail this job if the merge branch CI pipeline failed
# unsuccessful_builds will be a multiline string variable with a line format of " $job_name: $build_url"
unsuccessful_builds="$(echo "${builds}" \
| jq --raw-output '.[] | select(.outcome!="success") | "\(.workflows.job_name): \(.build_url)"')"
if [[ -n "${unsuccessful_builds}" ]]; then
printf "master merge CI pipeline jobs failed:\n${unsuccessful_builds}\n"
2019-06-24 18:55:44 -04:00
curl -X POST -H 'Content-type : application/json' \
--data \
"{ \
\"attachments\": [ \
{ \
\"fallback\": \"Nightly Master Merge Failed!\", \
\"text\": \"Nightly *master* merge into *release/1-6* failed!\n\nBuild Log: ${CIRCLE_BUILD_URL}\n\nAttempted merge from: https://github.com/${CIRCLE_PROJECT_USERNAME}/${CIRCLE_PROJECT_REPONAME}/tree/${git_merge_branch}\n\nThere may be a merge conflict to resolve.\", \
\"footer\": \"${CIRCLE_PROJECT_USERNAME}/${CIRCLE_PROJECT_REPONAME}\", \
\"ts\": \"$(date +%s)\", \
\"color\": \"danger\" \
} \
] \
}" ${CONSUL_SLACK_WEBHOOK_URL}
2019-06-18 15:56:58 -04:00
exit 1
fi
2019-06-24 18:55:44 -04:00
current_ref=$(git rev-parse HEAD)
# Set CLA check to pass
echo "Setting CLA check"
curl -u "${HASHICORP_CI_GITHUB_TOKEN}:" -X POST \
--header "Accept: application/json" \
--show-error \
--silent \
--fail \
--output /dev/null \
-d "{ \"state\": \"success\", \
\"target_url\": \"https://cla.hashicorp.com/hashicorp/consul\", \
\"description\": \"Contributor License Agreement is signed.\", \
\"context\": \"license/cla\"}" https://api.github.com/repos/hashicorp/consul/statuses/${current_ref}
2019-06-18 15:56:58 -04:00
# CircleCI jobs passed, merging to release branch
echo "master merge CI pipeline passed successfully so merging to ${CIRCLE_BRANCH}!"
git checkout "${CIRCLE_BRANCH}"
git merge "${git_merge_branch}"
git push origin "${CIRCLE_BRANCH}"
git push --delete origin "${git_merge_branch}"
2019-06-24 18:55:44 -04:00
curl -X POST -H 'Content-type : application/json' \
--data \
"{ \
\"attachments\": [ \
{ \
\"fallback\": \"Nightly master merge success!\", \
\"text\": \"Nightly *master* merge into *release/1-6* succeeded!\", \
\"footer\": \"${CIRCLE_PROJECT_USERNAME}/${CIRCLE_PROJECT_REPONAME}\", \
\"ts\": \"$(date +%s)\", \
\"color\": \"good\" \
} \
] \
}" ${CONSUL_SLACK_WEBHOOK_URL}
2019-06-18 15:56:58 -04:00
2018-11-16 14:01:02 -05:00
workflows :
version : 2
2019-06-18 15:56:58 -04:00
merge-master-to-release :
jobs :
- merge-master
triggers :
- schedule :
cron : "0 2 * * *" # 2AM UTC <> 10PM EST <> 7PM PST should have no impact
filters :
branches :
only :
- release/1-6
2019-07-22 18:08:54 -04:00
go-tests :
2019-03-20 15:24:17 -04:00
jobs :
2019-07-16 18:52:24 -04:00
- lint-consul-retry
2019-07-22 18:08:54 -04:00
- go-fmt-and-vet
- dev-build :
2019-07-16 18:52:24 -04:00
requires :
- lint-consul-retry
2019-04-25 12:26:33 -04:00
- go-fmt-and-vet
2019-05-17 11:42:56 -04:00
- go-test : &go-test
requires :
- dev-build
- go-test-api : *go-test
2019-07-22 18:08:54 -04:00
build-distros :
jobs :
- build-386
- build-amd64
- build-arm-arm64
test-integrations :
jobs :
- dev-build
2019-04-02 12:24:48 -04:00
- dev-upload-s3 :
requires :
- dev-build
filters :
branches :
ignore :
- /^pull\/.*$/ # only push dev builds from non forks
2019-01-28 11:18:30 -05:00
- nomad-integration-master :
requires :
2019-03-08 15:33:03 -05:00
- dev-build
2019-01-28 11:18:30 -05:00
- nomad-integration-0_8 :
requires :
- dev-build
2019-04-29 17:27:57 +01:00
- envoy-integration-test-1.8.0 :
requires :
- dev-build
- envoy-integration-test-1.9.1 :
requires :
- dev-build
2019-06-07 07:10:43 -05:00
- envoy-integration-test-1.10.0 :
requires :
- dev-build
2018-11-16 14:01:02 -05:00
website :
jobs :
2019-04-04 18:54:47 -04:00
- build-website
- docs-link-checker :
requires :
- build-website
2019-04-05 10:38:27 -04:00
filters :
branches :
ignore :
- /^pull\/.*$/ # only run link checker on non forks
2019-04-04 18:54:47 -04:00
- deploy-website :
requires :
- docs-link-checker
2018-11-16 14:01:02 -05:00
context : static-sites
filters :
branches :
only : stable-website
2018-12-20 13:51:48 -05:00
frontend :
jobs :
- frontend-cache
- ember-build :
requires :
- frontend-cache
- ember-test :
requires :
- ember-build