DEV Community

Cover image for Cypress and Angular in a Jenkins Pipeline via Docker
Justin
Justin

Posted on • Updated on

Cypress and Angular in a Jenkins Pipeline via Docker

What's the objective?

In this post I'm going to show you how to run Cypress in a Jenkins pipeline (via Docker) to test an Angular application.

The main focus of this tutorial is not on these individual technologies. Instead, it's how to pull them all together to give you an example of how continuous integration could look for your project.


What's needed before starting?

This tutorial assumes that you have...

  • An Angular application (built with the CLI) with Cypress installed and at least one test written.

  • Docker installed & configured on your machine

  • Experience working with these technologies before

This example was developed using macOS


Jenkins Initial Setup, Startup and Configuration

Start by creating some volumes for your Jenkins Docker container:

# from the terminal
docker volume create jenkins-data 
docker volume create npm-cache
docker volume create cypress-cache
Enter fullscreen mode Exit fullscreen mode

Next, use the docker run command to pull down the Jenkins Docker image, configure the volumes you've created and run the container serving the Jenkins app on localhost:8080:

# from the terminal
docker run -u root -d --name jenkins-tut -p 8080:8080 --ipc=host \
-v jenkins-data:/var/jenkins_home \
-v npm-cache:/root/.npm \
-v cypress-cache:/root/.cache \
-v /var/run/docker.sock:/var/run/docker.sock jenkinsci/blueocean:latest
Enter fullscreen mode Exit fullscreen mode

Once that completes, open a browser and navigate to: http://localhost:8080 and wait until you see the 'Unlock Jenkins' page appear.

This page should prompt you for an administrator password. To retrieve it, peek at the Docker logs for the jenkins-tut container you have running:

# from the terminal
docker logs jenkins-tut
Enter fullscreen mode Exit fullscreen mode

The password should appear in the log under the 'Jenkins initial setup is required...' statement. It's a long alphanumeric string - you can't miss it! It looks like this: 2f064d3663814887964b682940572567.

Copy and paste that puppy in and select 'Continue'!

After you make it past the 'Unlock Jenkins' page, you will be prompted to create a user and login for returning sessions. Create your desired user and continue on.

Next, you'll be prompted with the 'Customize Jenkins' page. Select 'Install Suggested Plugins' and continue on.

You should now see the Jenkins Dashboard!


Prepping Angular/Cypress for Deployment

Now hop over to your Angular project (that has Cypress installed and configured with at least one running test) and wire up the build process.

First, we need to install 2 packages that will allow Angular and Cypress to run together on the Jenkins node.

Inside the root directory of your Angular project, run:

# from the terminal
npm install angular-http-server --save-dev
npm install start-server-and-test --save-dev
Enter fullscreen mode Exit fullscreen mode

angular-http-server allows the built Angular code (in /dist) to be served on the Jenkins node during the Pipeline. We want it running so that Cypress can call it.

start-server-and-test enforces a sequential start up between the Angular app and the Cypress tests. This prevents the issue of Cypress trying to call the app before it's running.

Next we'll need to add some scripts to the package.json file. These scripts will make use of the new packages and orchestrate the startup of the Angular app and the Cypress test suite.

In the package.json:

...
"scripts": {
    ...
    "cy:run": "cypress run",
    "cy:verify": "cypress verify",
    "ci:start-ng": "angular-http-server --path ./dist/YOUR_APP_NAME -p 4200",
    "ci:cy-run": "start-server-and-test ci:start-ng http://localhost:4200 cy:run",
    ...
}
...
Enter fullscreen mode Exit fullscreen mode

If you're using Angular 9 you may want to add this line as well to the package.json scripts section:

    # for Angular 9 only
    "postinstall": "ngcc --properties es2015 browser module main --first-only --create-ivy-entry-points",
Enter fullscreen mode Exit fullscreen mode

This works with v9's Ivy view engine and will speed up compilation time when running ng build.

Lastly, well create a Jenkinsfile at the root of the project that Jenkins will read from and define some stages for the build to go through.

Create a Jenkinsfile

pipeline {
    agent {
        docker {
            image 'cypress/base:12.16.1' 
            args '-p 3000:3000' 
        }
    }
    stages {
        stage('Install Dependencies') { 
            steps {
                sh 'npm ci'
                sh 'npm run cy:verify'
            }
        }
        stage('Build') { 
            steps {
                sh 'npm run build'
            }
        }
        stage('Test') { 
            steps {
                sh 'npm run ci:cy-run'
            }
        }
    }
}
Enter fullscreen mode Exit fullscreen mode

The agent section will allow our Jenkins container to pull in a Cypress Docker image (with Node 12.16.1) for the duration of the build. The steps that follow will actually be run in that environment.

We also add in 3 steps that will install the projects dependencies, verify a successful Cypress installation, build the Angular project and then run the compiled Angular code with Cypress using the sequential start provided by the angular-http-server and start-server-and-test packages.

With all that added, save and commit your changes!


Create the Jenkins Pipeline Project

Now head back to your Jenkins instance...

  • From the dashboard view, click 'Create New Jobs' in the middle or 'New Item' at the top of the left menu.

  • Enter a name for your item.

  • Select 'Pipeline' as the type and click 'OK' at the bottom of the page.

  • Add an optional description for your pipeline.

  • Click the 'pipeline' tab at the top of the page to bring the 'pipeline' section into view.

  • In the 'Definition' field, select: 'Pipeline script from SCM'

  • In the 'SCM' field, select: 'Git'

  • In the 'Repository URL' field, enter the path to your project on your local system translated to the /home directory of the Jenkins container. For example, if your project is at /Users/YOUR_NAME/Development/YOUR_PROJECT then you would enter /home/Development/YOUR_PROJECT. Jenkins will perform some validation against your entry to let you know if it can reach the project.

  • Lastly, 'Save' your configuration!


Run the Build!

With your new pipeline created, it's time to run a build!

From the Jenkins dashboard view, you should see your new pipeline listed. Select the project name and then in the details view select 'Build Now'.

Once you've started the build, you should see a new entry in the 'Build History' menu to the left. If you click on the flashing dot next to the build number, you can view the log of the build.

Verify success and celebrate!


Resources

This tutorial came from a collection of resources. I encourage you to further explore these technologies and improve upon this process!

https://jenkins.io/doc/tutorials/build-a-node-js-and-react-app-with-npm/

https://www.cypress.io/blog/2019/08/02/guest-post-angular-adding-cypress-ui-tests-to-your-devops-pipeline/

https://github.com/cypress-io/cypress-example-kitchensink/blob/master/Jenkinsfile

Cover photo by JJ Ying on Unsplash

Top comments (0)