Skip to content

MitocGroup/tutorial-ci-for-serverless

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

3 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Tutorial: Continuous Integration for Serverless Applications

Continuous Integration is the practice of merging all working copies of developer code into one shared mainline several times a day. Best practices include automation of builds and deployments, with fast and self-testing builds, as well as production-like testing environments. With serverless, the continuous integration pipeline evolves from a one-lane, one-way street into a multiple-lane, two-way highway.

In this tutorial, we take a simple serverless application and walk you through the steps to set up unit testing, end-to-end testing, code coverage, code analysis, code security, code performance, and peer code review. You can learn how to use AWS serverless components in combination with GitHub, Travis-CI, CodeClimate, Snyk and other serverless-friendly services.

Getting Started

Step 1: Your Serverless Application

Step 1 Branch: https://github.com/MitocGroup/tutorial-ci-for-serverless/tree/tutorial-step1

For this tutorial, we went with Single Page Application from todomvc.com, transformed by our team into TodoMVC Serverless App. Similar approach can be applied to serverless applications described in Serverless Single Page Apps or Serverless Stack, as well as any other serverless applications from Curated List of Awesome Serverless.

# download locally todomvc serverless application codebase
git clone [email protected]:MitocGroup/deep-microservices-todomvc.git

# download locally tutorial repository codebase
git clone [email protected]:MitocGroup/tutorial-ci-for-serverless.git

# copy serverless application into a new branch, part of tutorial repository
cd ./tutorial-ci-for-serverless
git checkout -b tutorial-step1
cp -R ../deep-microservices-todomvc/src ../deep-microservices-todomvc/bin .
git commit -a -m "initial serverless application"
git push --set-upstream origin tutorial-step1

Step 2: Setup Travis CI

Step 2 Branch: https://github.com/MitocGroup/tutorial-ci-for-serverless/tree/tutorial-step2

Travis CI takes care of running your tests and deploying your apps. Quoting official Getting Started guide:

To start using Travis CI, make sure you have all of the following:

  • GitHub login
  • Project hosted as a repository on GitHub
  • Working code in your project
  • Working build or test script

Here below is our initial .travis.yml file:

language: node_js
dist: trusty
git:
  depth: 1
cache:
  bundler: true
  directories:
    - node_modules
    - "$(npm root -g)"
    - "$(npm config get prefix)/bin"
node_js:
  - 6
  - 8
script: echo "Hello World!"

Once we have updated .travis.yml, we can commit the code back to our repository:

# assuming you're still inside the tutorial repository
git checkout -b tutorial-step2
git add .travis.yml
git commit -m "enable travis ci"
git push --set-upstream origin tutorial-step2

And there you go, our first successful build: https://travis-ci.org/MitocGroup/tutorial-ci-for-serverless/builds/283669984

Step 3: Setup Unit Testing

Step 3 Branch: https://github.com/MitocGroup/tutorial-ci-for-serverless/tree/tutorial-step3

Travis CI inspired us with its simplicity and streamlined process. On the other hand, we were surprised that triggering tests execution is not part of the service. And we couldn't find any other tool that does it well, so we wrote one: https://www.npmjs.com/package/recink

To get started with this tool, just install it on your machine and configure:

# install recink as global nodejs library
npm install -g recink

# configure recink in your repository
recink configure recink

Last command will dump an example of recink.yml (conceptually the same approach as .travis.yml), which helps to simplify and streamline the execution of unit tests, e2e tests and more:

---
$:
  npm:
    scripts: []
    dependencies:
      chai: 'latest'
  emit:
    pattern:
      - /.+\.js$/i
    ignore:
      - /^(.*\/)?bin(\/?$)?/i
      - /^(.*\/)?node-bin(\/?$)?/i
      - /^(.*\/)?node_modules(\/?$)?/i
      - /^(.*\/)?vendor(\/?$)?/i
  test:
    mocha:
      options:
        ui: 'bdd'
        reporter: 'spec'
    pattern:
      - /.+\.spec\.js$/i
    ignore: ~

deep-todomvc:
  root: 'src/deep-todomvc'

In order to let Travis CI know that we have some tests to execute, we go back to .travis.yml and change the script that will be executing from script: echo "Hello World!" to this:

before_install:
  - npm install -g recink
script: recink run unit

Once we have updated .recink.yml and .travis.yml, we can commit the code back to our repository:

# assuming you're still inside the tutorial repository
git checkout -b tutorial-step3
git add .travis.yml .recink.yml
git commit -m "enable unit testing"
git push --set-upstream origin tutorial-step3

Step 4: Setup Code Coverage

Step 4 Branch: https://github.com/MitocGroup/tutorial-ci-for-serverless/tree/tutorial-step4

No matter what unit testing tool you might be using, most of them include code coverage capability, which is a measurement in percentage used to describe the degree to which the source code is executed when a particular test suite runs. Higher percentage in code coverage has lower probability of bugs and undetected issues, especially when you need to make changes to the code.

To enable code coverage, you need to simply add the following config block to .recink.yml, after test config block:

---
$:
  [...]
  test:
    [...]
  coverage:
    reporters:
      text-summary: ~
    pattern:
      - /.+\.js$/i
    ignore:
      - /.+\.spec\.js$/i
      - /.+\.e2e\.js$/i
      - /^(.*\/)?node_modules(\/?$)?/i

deep-todomvc:
  root: 'src/deep-todomvc'

No changes are necessary to .travis.yml.

Once we have updated .recink.yml, we can commit the code back to our repository:

# assuming you're still inside the tutorial repository
git checkout -b tutorial-step4
git add .travis.yml .recink.yml
git commit -m "enable code coverage"
git push --set-upstream origin tutorial-step4

Step 5: Setup Code Climate

Step 5 Branch: https://github.com/MitocGroup/tutorial-ci-for-serverless/tree/tutorial-step5

Code Climate empowers organizations to take control of their code quality by incorporating fully configurable test coverage and maintainability data throughout the development workflow. This service allowed us to:

  1. Report code analysis over time and enforce industry accepted coding standards
  2. Report code coverage over time and enforce qualitative processes like
    • fail when code coverage percentage drops lower than X percent
    • fail when code coverage difference between two consecutive reports drops more than Y percentage points

In order to do that, we need to update both .travis.yml and .recink.yml like we did in previous step. Travis CI will store securely Code Climate token, while recink allow us to cache code coverage reports and enforce above described quality metrics.

We imported GitHub repository into Code Climate (follow this link for detailed instructions) and retrieved the code coverage token (follow this link for detailed instructions).

Update .recink.yml with below configuration and replace with your token:

---
$:
  [...]
  coverage:
    [...]
  codeclimate:
    token: '[replace_with_your_token]'

deep-todomvc:
  root: 'src/deep-todomvc'

Support for Code Climate was implemented as an independent recink component, therefore we need to update .travis.yml by changing before_install and script lines, as shown below:

before_install:
  - npm install -g recink
  - npm install -g recink-codeclimate
script: recink run unit -c recink-codeclimate

Once we have updated .recink.yml and .travis.yml, we can commit the code back to our repository:

# assuming you're still inside the tutorial repository
git checkout -b tutorial-step5
git add .travis.yml .recink.yml
git commit -m "enable code climate"
git push --set-upstream origin tutorial-step5

Step 6: Setup Encryption

Step 6 Branch: https://github.com/MitocGroup/tutorial-ci-for-serverless/tree/tutorial-step6

Hopefully you felt very uncomfortable to put your code climate token as plain text into GitHub. Don't even want to think that you didn't.

Well, Travis CI offers encryption capability, that allows you to store your tokens, credentials and any other secrets. We have integrated this feature into recink as follows:

# encrypt Code Climate token
recink travis encrypt -x "CODECLIMATE_REPO_TOKEN=[replace_with_your_token]"

This command automatically updates your .travis.yml. All that you need to do it to update your .recink.yml as follows:

---
$:
  [...]
  coverage:
    [...]
  codeclimate:
    token: 'process.env.CODECLIMATE_REPO_TOKEN'
  preprocess:
    '$.codeclimate.token': 'eval'

deep-todomvc:
  root: 'src/deep-todomvc'

Once we have updated .recink.yml and .travis.yml, we can commit the code back to our repository:

# assuming you're still inside the tutorial repository
git checkout -b tutorial-step6
git add .travis.yml .recink.yml
git commit -m "enable token encryption"
git push --set-upstream origin tutorial-step6

Step 7: Setup Code Coverage Compare

Step 7 Branch: https://github.com/MitocGroup/tutorial-ci-for-serverless/tree/tutorial-step7

Going back to code coverage, in theory, you want to have 100% code coverage. In practice, though, this is really difficult to achieve and sustainably maintain. That is why we have adopted a relative comparision process, which allows engineers NOT to drop lower than couple of percentage points for each commit or pull request.

First, in order to store persistently each code coverage report and reuse it for code coverage compare, we are engaging Amazon S3 as our caching layer. To work with S3, we need AWS credentials and S3 bucket region. Similar to Code Climate token, let's encrypt these secrets:

# encrypt aws access key id
recink travis encrypt -x "AWS_ACCESS_KEY_ID=[replace_with_your_access_key]"

# encrypt aws secret access key
recink travis encrypt -x "AWS_SECRET_ACCESS_KEY=[replace_with_your_secret_key]"

# encrypt aws region
recink travis encrypt -x "AWS_DEFAULT_REGION=[replace_with_your_region]"

Second, in order to allow code coverage reports to be stored in Amazon S3, update .recink.yml as shown below:

---
$:
  [...]
  coverage:
    [...]
    compare:
      negative-delta: 1
      storage:
        driver: 's3'
        options:
          - 's3://[replace_with_your_s3_bucket]/[replace_with_your_s3_key_space]'
          -
            region: 'process.env.AWS_DEFAULT_REGION'
            accessKeyId: 'process.env.AWS_ACCESS_KEY_ID'
            secretAccessKey: 'process.env.AWS_SECRET_ACCESS_KEY'
  codeclimate:
    token: 'process.env.CODECLIMATE_REPO_TOKEN'
  preprocess:
    '$.codeclimate.token': 'eval'
    '$.cache.options.1.region': 'eval'
    '$.cache.options.1.accessKeyId': 'eval'
    '$.cache.options.1.secretAccessKey': 'eval'

deep-todomvc:
  root: 'src/deep-todomvc'

Once we have updated .recink.yml and .travis.yml, we can commit the code back to our repository:

# assuming you're still inside the tutorial repository
git checkout -b tutorial-step7
git add .travis.yml .recink.yml
git commit -m "enable code coverage compare"
git push --set-upstream origin tutorial-step7

Step 8: Setup Snyk

Step 8 Branch: https://github.com/MitocGroup/tutorial-ci-for-serverless/tree/tutorial-step8

Snyk continuously finds and fixes vulnerabilities in your dependencies. This service allowed us to:

  1. Automatically test our applications dependencies
  2. Fix security risks through upgrades and patches
  3. Prevent us from adding vulnerable dependencies
  4. Stay alert about new vulnerabilities

Similar to Code Climate, we need to update both .travis.yml and .recink.yml to enable this feature as part of our Continuous Integration process.

Again, like in Code Climate step, first retrieve the token from your account (follow this link for detailed instructions). Next, encrypt Snyk token, which will update your .travis.yml file:

# encrypt snyk token
recink travis encrypt -x "SNYK_API_TOKEN=[replace_with_your_token]"

Because Snyk is built as an independent recink component, we need to enable this feature in .travis.yml by changing before_install and script lines, as shown below:

before_install:
  - npm install -g recink
  - npm install -g recink-codeclimate
  - npm install -g recink-snyk
script: recink run unit -c recink-codeclimate -c recink-snyk

After updating .recink.yml with below several lines, we'll be able to collect security reports from Snyk service:

---
$:
  [...]
  codeclimate:
    [...]
  snyk:
    token: 'process.env.SNYK_API_TOKEN'
  preprocess:
    [...]
    '$.snyk.token': 'eval'

deep-todomvc:
  root: 'src/deep-todomvc'

Once we have updated .recink.yml and .travis.yml, we can commit the code back to our repository:

# assuming you're still inside the tutorial repository
git checkout -b tutorial-step8
git add .travis.yml .recink.yml
git commit -m "enable snyk"
git push --set-upstream origin tutorial-step8

Step 9: Setup S3 Caching

Step 9 Branch: https://github.com/MitocGroup/tutorial-ci-for-serverless/tree/tutorial-step9

To be updated.

Once we have updated .recink.yml and .travis.yml, we can commit the code back to our repository:

# assuming you're still inside the tutorial repository
git checkout -b tutorial-step9
git add .travis.yml .recink.yml
git commit -m "enable s3 caching"
git push --set-upstream origin tutorial-step9

Final Step: Summary

In this tutorial, we walked you through how to:

  1. Setup Continuous Integration (using GitHub and Travis CI)
  2. Setup Unit Testing (using Mocha and REciNK)
  3. Setup Code Coverage and Code Analysis (using Code Climate and REciNK)
  4. Setup Continuous Security (using Snyk and REciNK)

At the time of writing, our team is working on several more features that will be updated in this tutorial later:

  1. Setup Continuous Deployment (using Travis CI and DevOps tools)
  2. Setup End to End Testing (using TestCafe and REciNK)
  3. Setup Continuous Performance (using PageSpeed and REciNK)

Any feedback is welcome! Feel free to reach us at www.mitocgroup.com, report bugs by filing issues here or contribute directly as described here. Thank you!