This repository contains the code to setup the Delivery Pipeline.
Proposed directory structure. Serves as a starting point.
Contains Docker file to create Jenkins Master image with custom startup scripts.
Docker file to create Jenkins worker nodes
Build the docker image with: ``docker build -t scos/jenkins .```
To run Docker and skip the setup wizard
docker run -d -p 8080:8080 -p 50000:50000 --name jenkins -e JAVA_OPTS=-Djenkins.install.runSetupWizard=false scos/jenkins
To get list of installed plugins
curl -sSL "http://localhost:8080/pluginManager/api/xml?depth=1&xpath=/*/*/shortName|/*/*/version&wrapper=plugins" | perl -pe 's/.*?<shortName>([\w-]+).*?<version>([^<]+)()(<\/\w+>)+/\1 \2\n/g'|sed 's/ /:/'
Goals:
-
build a Jenkins master Docker image that meets the following criteria:
- requires login (Active Directory/Github credentials/AWS Users - still have to decide on this)
- has plugins installed. These include BlueOcean plugins and ec2 plugin (needs some cleanup)
- has ec2 plugin configured automatically via groovy code (This may need to be environment specific)
- AWS credentials are passed through the environment (this is needed for spinning up ECS instances from the master)
- master URL needs to be accessible from the slaves (need to figure that one out)
-
build a Jenkins slave Docker image that meets the following criteria:
- has Docker installed
- has inspec installed
- has other dependencies like npm? Not sure at this point. Should we also consider building different types of slave images?
- the idea is as we run the inspec tests it starts things like Grafana in a sibling Docker container
-
provision Jenkins in AWS (I am looking for feedback here)
- a VPC used by Continuous Delivery if needed(all this is Terraformed ideally)
- setup public/private subnets or put all in a public subnet? (we are still protected by the VPN)
- security groups
- provision ECS instances
- Jenkins master is accessible to developers through VPN
- saves jenkins_home state in EFS (is this going to be a Docker volume?)
- setup backup (future)
- HA for Jenkins master(future)
TODO:
- Jenkins Master image
- select cluster when Jenkins starts up - done
- extract task definition template in separate file - done
- configure Nexus so Jenkins can push to it: http://codeheaven.io/using-nexus-3-as-your-repository-part-3-docker-images/
- configure SCOS Github Repo so is automatically monitored
- handle secrets with a Jenkins plugin for now
- fix security warnings
- configure LDAP
- Jenkins Slave image
- docker, terraform and inspec installed
- other docker containers start as siblings - Jenkins has a docker plugin. How can we use it?
- Terraform
- EFS for Jenkins - done
- move Jenkins to private subnets - done
- tune number of instances to start
- implements scale up and scale down policies
- configure Route 53
- migrate ELB from classic
- restrict Jenkins access by IP?
- extract LB variables - done
- use EC2 image that has up to date ECS agent - done
- Nexus
- Nexus confirmation of EFS persistence by restarting Nexus container and confirming configuration remains - done
- Nexus setup of LDAP
- Change Nexus local admin password from its default setting - DONE
- Change Nexus script to same approach used by Jenkins - DONE
- Jenkins file for sample repository - pick monitoring
- runs docker inspec tests in a sibling docker container
- publishes to Nexus if successful
- Other
- Resolve conflict with the elb security group name.
- Consider an approach where security groups are reusable. E.g. use same security groups for Delivery Pipeline load balancers
Sometimes the number of security errors and warning under the "Manage Jenkins" page get a little high. Here is how to deal with this:
- Log in to Jenkins
- Go to Jenkins > Manage Jenkins in the toolbar on the left
- Let people know you are working on Jenkins updates and click on the "Prepare to Shutdown" button towards the bottom of the screen to put Jenkins into a mode where it won't run new builds.
- If Jenkins has a warning near the top of the screen about the Jenkins version itself needing updates, then click on the "Update automatically" button it presents to you and wait for it to update and reboot.
- Click on the "Manage Plugins" button to get a list of all the plugins needing updates.
- For each plugin listed under the "Updates" tab, check its check box and then click on "Download now and install after restart" (the wording will be similar, at least). (Optionally) click the "Check now" button to make sure you have the most recent list of plugins to update. You should be taken to the update center and given a status of the installs. Wait until they are all done downloading and ready for a restart. NOTE: Take note of ANY plugins saying that they update the format of their configuration or otherwise have red notices on them. You will likely need to manually verify and correct the behavior/config of those plugins after an upgrade
- Jenkins is notoriously bad about actually restarting after the plugins have downloaded, even if you check the option. So go to "/restart" in your browser to actually force a restart.
- Wait for Jenkins to restart and verify under "Manage Jenkins" and "Manage Plugins" that no more updates are necessary. Jenkins will also warn you, under "Manage Jenkins" if any plugin configuration need updated due to the upgrade.
Now that the plugins and/or Jenkins are updated, we need to take note of them in source control (here) to make sure we have them in case we need to restore them.
- Grab a list of all the plugins we have. At present, you can't use the commands listed earlier on in this README as you need to logged in.
- Log in and click on "Manage Jenkins"
- Click on the "Script Console" button
- Enter and run the following script to get a sorted and formatted list of all plugins:
Jenkins.instance.pluginManager.plugins.sort(false).each{
plugin ->
println ("${plugin.getShortName()}:${plugin.getVersion()}")
}
- Copy the list of plugins returned onto your clipboard
- Paste the copied list into the
src/docker/jenkins/master/plugins.txt
file and verify that its git diff isn't too huge (if it is, it's possible you messed up) - Open a PR for the change and confirm that the CI build for it makes a new master docker image successfully.
- Once you merge into master, check the CI merge build to confirm what the new master docker image tag is.
- In the scos-alm repository, update the docker image tag in the
jenkins.tf
file. - Put that change through a PR, merge, etc. and then
terraform
plan and apply those changes from your local workstation (they are not done by CI as they affect CI). - Once the
terraform
steps are done, you'll need to manually force AWS ECS to roll the master, as it won't on its own. Make sure to let people know that Jenkins is going down (you can even do a "Prepare for Shutdown" as described earlier.
- Go to ECS in AWS and on click on the main Jenkins cluster in the "Clusters" tab
- Click on the "tasks" tab and the check the main master task. It will have "continuous delivery" in its name.
- Click on the "stop" button, and ECS will stop the container and run a new one with the latest image.
- After a few minutes verify that Jenkins is back up and running with no warning or errors under "Manage Jenkins"