-
Notifications
You must be signed in to change notification settings - Fork 70
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* Add basic outline for CI/CD demo More content coming soon, not ready for formal review * Add service install details and images * Add site-test files, complete preliminary deploy * Update Jenkinsfile The Jenkinsfile was using the deprecated "appid" for Marathon, fix. * Update job name, other little fixes * Fix services running image for site-test service * Add back Jenkins poll SCM image link * Add linkchecker and CI test * Update Jenkins success image * Add basic outline for CI/CD demo More content coming soon, not ready for formal review * Add service install details and images * Add CI/CD Demo * Update Jenkinsfile The Jenkinsfile was using the deprecated "appid" for Marathon, fix. * Update job name, other little fixes * Fix services running image for site-test service * Add back Jenkins poll SCM image link * Add linkchecker and CI test * Update Jenkins success image * Fix up merge conflict with README * Clarify the hostname & port to use in Jenkins
- Loading branch information
Showing
14 changed files
with
253 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,156 @@ | ||
# Continuous Delivery with GitLab and Jenkins | ||
|
||
This demo presents a continuous integration and deployment (CI/CD) workflow using GitLab and Jenkins on DC/OS. In this demo we’ll be creating a simple website, testing it to make sure all the links work, and then deploying it in production if it passes. | ||
|
||
- Estimated time for completion: 30min | ||
- Target audience: Anyone interested in simplifying the deployment of a CI/CD pipeline. | ||
|
||
**Table of Contents**: | ||
|
||
- [Architecture](#architecture) | ||
- [Prerequisites](#prerequisites) | ||
- [Install](#install) | ||
- [Use the demo](#use) | ||
|
||
## Architecture | ||
|
||
## Prerequisites | ||
|
||
- A running [DC/OS 1.11](https://dcos.io/releases/) or higher cluster with at least 3 private agents and 1 public agent. Each agent should have 2 CPUs and 5 GB of RAM available. The [DC/OS CLI](https://dcos.io/docs/1.11/usage/cli/install/) also needs to be installed. | ||
- The [dcos/demo](https://github.com/dcos/demos/) Git repo must be available locally, use: `git clone https://github.com/dcos/demos/` if you haven't done so yet. | ||
- [SSH](https://docs.mesosphere.com/1.11/administering-clusters/sshcluster/) cluster access must be set up. | ||
- Ports 22222 and 50000 opened on the public agent where Marathon-lb is running. | ||
- The ability to add a DNS A record to use for GitLab that you and the cluster have access to, if you don't have a domain or run DNS, you may use a free service like <a href="https://www.noip.com/">noip.com</a>. | ||
|
||
The DC/OS services used in the demo are as follows: | ||
|
||
- GitLab | ||
- Jenkins | ||
- Marathon-lb | ||
|
||
## Setup | ||
|
||
Get the <a href="https://docs.mesosphere.com/1.11/administering-clusters/locate-public-agent/">address of your public agent</a> and set up a DNS A record to use for GitLab. In this demo we'll be using cd.example.com, you'll want to replace this in all examples. | ||
|
||
If you have SSL certificates configured which will be used by the Public Agent, you're all set. Otherwise, you'll want to log into your private nodes and configure Docker to <a href="https://docs.docker.com/registry/insecure/">use an insecure registry</a>. Using an insecure registry is not appropriate for production, but for the purpose of a private demo it may be used. | ||
|
||
First, log into each of the private nodes, you should get the Mesos ID and then use that to ssh into the nodes: | ||
|
||
``` | ||
dcos node | ||
dcos node ssh --master-proxy --mesos-id=4cb15493-2ca7-4b8b-8d4d-83b76cdcf7bb-S1 | ||
``` | ||
|
||
Then configure Docker on each one: | ||
|
||
``` | ||
sudo mkdir -p /etc/systemd/system/docker.service.d/ | ||
sudo vim /etc/systemd/system/docker.service.d/insecure-registry.conf | ||
``` | ||
|
||
The contents of insecure-registry.conf should be the following: | ||
|
||
``` | ||
[Service] | ||
Environment='DOCKER_OPTS=--insecure-registry=cd.example.com:50000' | ||
``` | ||
|
||
Remember to replace `cd.example.com` with the A record you have pointed at your public agent. | ||
|
||
Now reload Docker and confirm the parameter has been passed: | ||
|
||
``` | ||
sudo systemctl daemon-reload | ||
sudo systemctl restart docker | ||
systemctl show --property=Environment docker | ||
``` | ||
|
||
Warning: By default, restarting Docker will cause your running containers to restart. | ||
|
||
Later, we will also need to configure Jenkins to pull from an insecure registry during the deployment step. | ||
|
||
## Install | ||
|
||
Now you'll want to install Marathon-LB. We're not making any changes, so it's easy to simply do this from the DC/OS CLI if you wish: | ||
|
||
``` | ||
dcos package install marathon-lb | ||
``` | ||
|
||
Next, install Jenkins. In production you'll want to make sure Jenkins is running on one of the nodes where shared, persistent storage is located. For this demo it's fine to simply pin it to a private node of your choosing and save the files in /tmp on that node. Pinning to a node can easily be done by installing Jenkins through the web UI and filling out the Storage tab and setting a "Pinned Hostname" that points to one of your private nodes. | ||
|
||
data:image/s3,"s3://crabby-images/d286f/d286fe9294206cfa22f8de3f7eaf05a1ff08db1d" alt="Jenkins Pinned Hostname" | ||
|
||
Once Jenkins is installed, if you don't have SSL certificates set up you'll need to configure Jenkins to accept an insecure registry so it can launch Docker images from it. This can be completed by following instructions in the <a href="https://docs.mesosphere.com/services/jenkins/advanced-configuration/">Jenkins Advanced Configuration</a> section of the DC/OS service documentation. You will use the DNS name you set up and the GitLab registry port (default 50000) when following these instructions. | ||
|
||
Now it's time to install GitLab. Just like Jenkins, we'll want to pin this to a hostname for this demo just in case the container restarts. In a production environment you'll want to mount external storage shared storage so that GitLab can move between nodes, but for a quick demo you may use the limited in-container non-persistent storage. In order to pin it to a node, go into the "Single Node" section of the configuration and put in the IP address of the private node you wish to run it on in the section for "pinned hostname". | ||
|
||
data:image/s3,"s3://crabby-images/d2269/d22699dc2fcfada6ab6848b0028739946d933d09" alt="GitLab Pinned Hostname" | ||
|
||
You'll also want to set the "virtual host" in the routing section to match the name you created in DNS. | ||
|
||
data:image/s3,"s3://crabby-images/6c08d/6c08d0c627fd638270aafab0306ac2ff0440cb26" alt="GitLab Virtual Host" | ||
|
||
Once GitLab is installed, immediately navigate to the address you set up (we're using http://cd.example.com as an example) and set a password for GitLab. Once a password is set, you can log in with the "root" user and this password. | ||
|
||
This concludes the tooling you need installed, you can now get started setting up your repository and deployment pipeline! | ||
|
||
## Set Up Repository and Job | ||
|
||
Once you've logged into GitLab, you want click on "New Project" to create a repository. You'll want to give it a Project name of "site-test" and make it Public. | ||
|
||
Tip: In production, the root user wouldn't typically own repositories like this, they would be owned by less privileged users. | ||
|
||
You can now clone this empty repository to your local system: | ||
|
||
``` | ||
git clone http://cd.example.com/root/site-test.git | ||
Cloning into 'site-test'... | ||
warning: You appear to have cloned an empty repository. | ||
``` | ||
|
||
One of the prerequisites for this demo was cloning the dcos/demos repository from GitHub as well. In the `demos/cicd/1.11/site-test/` directory you will find all the files that should now go into your newly created site-test repository, going through them one by one: | ||
|
||
* index.html - Basic site file that we will deploy | ||
* Dockerfile - Very basic Dockerfile which installs the index.html | ||
* Jenkinsfile - Basic Jenkinsfile that defines the pipeline in Jenkins for building and deploying the test site | ||
* marathon.json - Marathon definition for deploying the resulting image | ||
|
||
As you're copying these into your new site-test repository, update the Jenkinsfile and marathon.json so that our cd.example.com is replaced with the address you're using for GitLab. | ||
|
||
Now we'll want to create a pipeline job in Jenkins. On the main page for Jenkins, select "New Item". The item name will be "site-test" and it's a "Pipeline" after hitting "OK" on that screen, you'll get to the page for configuring the pipeline. Scroll down to "Build Triggers" and select "Poll SCM" which takes cron-style scheduling. For this demo, we want to run it every minute, so in the text box put: * * * * * | ||
|
||
data:image/s3,"s3://crabby-images/5c125/5c125a975f8d1a7f836e5db72d9c53682e3d0083" alt="Jenkins Poll SCM" | ||
|
||
In the Pipeline section of the same screen the definition should be "Pipeline script from SCM" and then select "Git". This will give you a place to put a Git repository, which in our example is http://cd.example.com/root/site-test.git and we will want to add credentials, since these will also be used for sending the image to the GitLab Docker registry, add the credential now and give it an ID of "gitlab". | ||
|
||
data:image/s3,"s3://crabby-images/9687a/9687a56f9446597b46e8de09b397583d512084df" alt="Jenkins GitLab Credentials" | ||
|
||
Once this is saved, you have a pipeline. | ||
|
||
Now return to your local site-test directory. We'll want to commit all the files you've added and push them to GitLab: | ||
|
||
``` | ||
git add -A | ||
git commit -m "Initial commit" | ||
git push | ||
``` | ||
|
||
You may navigate to your http://cd.example.com/root/site-test to see the files now uploaded there. | ||
|
||
Now return to Jenkins and nagivate to your site-test pipeline, it will run within a minute. | ||
|
||
data:image/s3,"s3://crabby-images/08240/08240b5ac97f2644478a8c2b04a108185b39c07f" alt="Jenkins Success" | ||
|
||
If the pipeline fails for any reason, there are logs there in the UI you can use to debug where the problem is. | ||
|
||
Once the deployment succeeds, you can go back to the DC/OS UI and you will now see an "site-test" service running. Naviate to /service/site-test to see the website running. | ||
|
||
data:image/s3,"s3://crabby-images/d2a68/d2a6873eb837ba6e1997bed0d7fe401e1c8ee3fd" alt="DC/OS Serivce" | ||
|
||
Congratulations, you now have a deployment! | ||
|
||
### | ||
|
||
Should you have any questions or suggestions concerning the demo, please raise an [issue](https://jira.mesosphere.com/) in Jira or let us know via the [[email protected]](mailto:[email protected]) mailing list. | ||
|
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
FROM nginx | ||
COPY index.html /usr/share/nginx/html/index.html | ||
RUN apt-get -qq update | ||
RUN apt-get install -y linkchecker |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,58 @@ | ||
def gitCommit() { | ||
sh "git rev-parse HEAD > GIT_COMMIT" | ||
def gitCommit = readFile('GIT_COMMIT').trim() | ||
sh "rm -f GIT_COMMIT" | ||
return gitCommit | ||
} | ||
|
||
node { | ||
// Checkout source code from Git | ||
stage 'Checkout' | ||
checkout scm | ||
|
||
// Build Docker image | ||
stage 'Build' | ||
sh "docker build -t cd.example.com:50000/root/site-test:${gitCommit()} ." | ||
|
||
// Log in and push image to GitLab | ||
stage 'Publish' | ||
withCredentials( | ||
[[ | ||
$class: 'UsernamePasswordMultiBinding', | ||
credentialsId: 'gitlab', | ||
passwordVariable: 'GITLAB_PASSWORD', | ||
usernameVariable: 'GITLAB_USERNAME' | ||
]] | ||
) { | ||
sh "docker login -u ${env.GITLAB_USERNAME} -p ${env.GITLAB_PASSWORD} cd.example.com:50000" | ||
sh "docker push cd.example.com:50000/root/site-test:${gitCommit()}" | ||
} | ||
|
||
// Test links in file | ||
stage 'Test' | ||
withCredentials( | ||
[[ | ||
$class: 'UsernamePasswordMultiBinding', | ||
credentialsId: 'gitlab', | ||
passwordVariable: 'GITLAB_PASSWORD', | ||
usernameVariable: 'GITLAB_USERNAME' | ||
]] | ||
) { | ||
sh "docker login -u ${env.GITLAB_USERNAME} -p ${env.GITLAB_PASSWORD} cd.example.com:50000" | ||
sh "docker run -p 8085:80 -d --name=test-container-${env.BUILD_NUMBER} cd.example.com:50000/root/site-test:${gitCommit()}" | ||
sh "docker exec test-container-${env.BUILD_NUMBER} linkchecker /usr/share/nginx/html/index.html" | ||
sh "docker kill test-container-${env.BUILD_NUMBER}" | ||
sh "docker rm test-container-${env.BUILD_NUMBER}" | ||
} | ||
|
||
// Deploy | ||
stage 'Deploy' | ||
|
||
marathon( | ||
url: 'http://marathon.mesos:8080', | ||
forceUpdate: false, | ||
filename: 'marathon.json', | ||
id: 'site-test', | ||
docker: "cd.example.com:50000/root/site-test:${gitCommit()}".toString() | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
<!DOCTYPE html> | ||
<html> | ||
<body> | ||
|
||
<h1>Site test</h1> | ||
|
||
<p>This is a test website. Learn more about DC/OS at <a href="https://dcos.io">dcos.io</a>!</p> | ||
|
||
</body> | ||
</html> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
{ | ||
"id": "/site-test", | ||
"cpus": 1, | ||
"mem": 128, | ||
"instances": 1, | ||
"container": { | ||
"docker": { | ||
"image": "cd.example.com:50000/root/site-test:latest", | ||
"portMappings": [ | ||
{ | ||
"containerPort": 80, | ||
"protocol": "tcp", | ||
"name": "http" | ||
} | ||
], | ||
"network": "BRIDGE" | ||
} | ||
}, | ||
"labels": { | ||
"DCOS_SERVICE_PORT_INDEX": "0", | ||
"DCOS_SERVICE_SCHEME": "http", | ||
"DCOS_SERVICE_NAME": "gitlab" | ||
} | ||
} |