Skip to content
This repository has been archived by the owner on Mar 30, 2023. It is now read-only.
/ cnp-module-webapp Public archive

A terraservice module that creates an application service plan and a web app

License

Notifications You must be signed in to change notification settings

hmcts/cnp-module-webapp

Repository files navigation

cnp-module-webapp

A module that lets you create a Web App and its associated App Service Plan, and depending on the environment you are targeting, the module will automatically deploy to the correct Application Service Environment. Refer to the following links for a detailed explanation of an App Service Plan, Web App and Application Service Environment in Azure.

App Service Plan
Web Apps
Application Service Environment

As part of the web app creation, an SSL cert for it is created and whitelisted at the application gateway provided in the appGateway var provided in the declaration of the module

Variables

This module lets you host Java 8, Spring Boot, and NodeJs applications.

Name Type Required Default description
source String Yes this is the location source for the moj-module-webapp, the example implies a github repo containing the moj-module-webapp source
product String Yes this is the name of the product or project i.e. probate, divorce etc.
location String No UK South this is the azure region for this service
appinsights_location String No West Europe the Azure region for App Insights instance, previously limited to 'West Europe' but now avaiable in 'UK South'. Current default is kept simply to prevent data loss. New instances should set 'UK South'.
env String Yes this is used to differentiate the environments e.g dev, prod, test etc
app_settings String Yes this is the key valued pairs of application settings used by the application at runtime
is_frontend Boolean No False Indicates that this app could be routable from the public internet
additional_host_name String No A custom domain name for your web application
https_only String No "false" Configures a web site to accept only https requests. Issues redirect for http requests. NB this is a string value that accepts values "true" or "false" - the string type is required to work around issues with Terraform and ARM template handling of boolean value.
common_tags Map Yes tags that need to be applied to every resource group, passed through by the jenkins-library
asp_rg String Yes Name of resource group where app service plan resides
asp_name String Yes this is the name of the shared service plan to be deployed to. Name should follow ${product}-${env} format
capacity Integer No 2 Target number of instances of the application to run. Note that there may be more or fewer instances actually running. This should not be used to guarantee singleton (capacity=1) instances
instance_size String No I2 The SKU size for app service plan instances. Valid values are I1 (small), I2 (medium) and I3 (large). Larger instances cost more - specs for Isolated Service Plan instances can be found here https://azure.microsoft.com/en-gb/pricing/details/app-service/windows/.
shared_infra String No false If set to true, it will not create the TM profile
deployment_target String No Name of the Deployment Target. If deployment_target is empty (legacy mode) the env/core-infra-{env} will be: core-infra-{env} otherwise will be: env-infra-{env}
certificate_key_vault_id String No The id of the key vault to retrieve the ssl certificate for
certificate_name String No The name of the certificate in key vault to use

Usage

Following is an example of provisioning a NodeJs, SpringBoot, and Java enabled web app, the following code fragment shows how you could use the cnp-module-webapp to provision the infrastructure for a typical frontend. To provision a backend Java, or SpringBoot infrastructure the code is exactly the same except 'is_frontend' must be set to false. 'capacity' is optional value as by default is set to '2'

module "frontend" {
	source               = "[email protected]:contino/cnp-module-webapp?ref=master"
	product              = "${var.product}-frontend"
	location             = "${var.location}"
	appinsights_location = "${var.location}"
	env                  = "${var.env}"
	capacity             = "${var.capacity}"
	is_frontend          = true
	asp_name             = "${var.product}-${var.env}"
	asp_rg               = "${var.product}-shared-infrastructure-${var.env}"
	subscription         = "${var.subscription}"
	common_tags          = "${var.common_tags}"
	app_settings         = {
		WEBSITE_NODE_DEFAULT_VERSION = "8.8.0"
	}
}
module "backend" {
	source                 = "[email protected]:contino/cnp-module-webapp?ref=master"
	product                = "${var.product}-api"
	location               = "${var.location}"
	appinsights_location   = "${var.location}"
	env                    = "${var.env}"
	capacity               = "${var.capacity}"
	asp_name               = "${var.product}-${var.env}"
	asp_rg                 = "${var.product}-shared-infrastructure-${var.env}"
	subscription           = "${var.subscription}"
	common_tags            = "${var.common_tags}"
	java_version           = "11"
	java_container_type    = "TOMCAT"
	java_container_version = "9.0"
	app_settings           = {
		WEBSITE_NODE_DEFAULT_VERSION = "8.8.0"
	}
}

In the example above, you can set the variables using terraform variables, so you can set these values in a .tfvars file, or pass them in from a Jenkins file.

Creating a web app to host your application will create a Resource Group containing a Web App and Deployment Slot.

Each of the aforementioned resources will be named the same, using the convention product-env, so if I provide the values for product as "probate", and env as "dev" then the resulting resource group and web app will be called probate-dev.

If is_frontend is set to true, an application gw and traffic manager profile is created. To leverage these, and functionailty such as the shutter page, you will need to set the following dns records for your app and set the additional_hostname param:

  • cname pointing fqdn of your app to hmcts-<app_name>-.trafficmanager.net
  • A record pointing tm<additional_hostname> to the IP of the application gw

SSL

If you wish to have a custom SSL certificate on your app you will need some additional configuration:

data "azurerm_key_vault" "cert_vault" {
  name = "infra-vault-${var.subscription}"
  resource_group_name = "${var.env == "prod" ? "core-infra-prod" : "cnp-core-infra"}"
}

variable "certificate_name" {
  default = "STAR-sandbox-platform-hmcts-net"
}

module "backend" {
	# ... copy other values from above
	certificate_name 		 = "${var.certificate_name}"
	certificate_key_vault_id = "${data.azurerm_key_vault.cert_vault.id}"
}

Prerequisites

Before deploying you webapp, ensure you have created a shared infrastructure repo with an app service plan as demonstrated in https://github.com/hmcts/cnp-rhubarb-shared-infrastructure

Testing

There's a library of unit tests and integration tests in this repository. In the root of this repository is a tests folder. Inside that are two folders named int and unit. Folder int contains the integration tests and fixtures, the obviously named folder called unit contains the unit tests. These tests are here to give quality assurance and should be added to and modified if changes are made to moj-module-webapp. Every commit to the moj-module-webapp will result in all the unit and integration tests being executed against it, if all of this succeeds it's verisoned and released in github. This so exisiting code that uses older versions of the moj-module-webapp will not break, and new infrastructure code can reference later releases.

##Testing Dependencies: TODO. write about sandbox and other dependencies for tests to work

Consider the following code fragment:-

source   = "git::https://yourgithubrepo/cnp-module-webapp?ref=0.0.67"

the 'ref=0.0.67' in the example code fragment suggests that it is using version 0.0.67 of the moj-module-webapp.

Unit Testing

The unit tests are written in Python, they contain many examples of how to test different aspects of the moj-module-webapp terraform code.

The following line of code from the tests.py file in the unit folder enforces the naming convention for the web app, as explained earlier the convention is product-env, the following code fragment enforces this in a unit test:-

self.v.resources('azurerm_template_deployment').property('name').should_equal('${var.product}-${var.env}')

You can find the complete set of tests in the file tests.py

Integration Testing

The int folder contains a test folder in which there are seperate folders for fixtures code, and integration tests. the file moj_azure_fixtures.tf contains the terraform code that spins up the dependencies that the webapp would need in order to function in production. The suite of integration tests run against this ephemeral infrastructure to validate the web app, once all the tests are succesfully executed the infrastructure is destroyed. The resources created use random names so that the integration tests don't conflict with each other, if more that one person is working on the repository. If any tests fail, the infrastructure is not automatically destroyed, this is so you can investigate the fixtures and the webapp through the Azure Portal, to help debug the unit test.

All of this automation is driven by Chef Kitchen, the configuration of all this is in the file called .kitchen.yml at the root of the int folder. The actual integration tests are in the default/controls sub folder in the integration folder. All the code used to drive the integration tests are written in Ruby. The folder default/libraries contain the ruby libraries that gather information required for the tests to execute.

Terraform

All infrastructure provisioning is done using Terraform native azurerm provider where possible. You can find the documentation for this at the following link:-

Terraform azurerm

At the time of writing the web app does not have native azurerm provider support in terraform at version 0.0.9, so an ARM template has been used for creation of the Web App and App Service Plan. The template can be found in the templates folder at the repository root.

The ARM template is wrapped in azurerm_template_deployment provider in terraform, this is the provider used to run any custom ARM templates using Terraform.