diff --git a/docs/media/Images/access-jenkins.png b/docs/media/Images/access-jenkins.png
new file mode 100644
index 0000000..5ce6936
Binary files /dev/null and b/docs/media/Images/access-jenkins.png differ
diff --git a/docs/media/Images/jenkins-admin-password.png b/docs/media/Images/jenkins-admin-password.png
new file mode 100644
index 0000000..133b337
Binary files /dev/null and b/docs/media/Images/jenkins-admin-password.png differ
diff --git a/docs/media/Images/jenkins-module-architecture.png b/docs/media/Images/jenkins-module-architecture.png
new file mode 100644
index 0000000..010e04c
Binary files /dev/null and b/docs/media/Images/jenkins-module-architecture.png differ
diff --git a/docs/media/diagrams/jenkins-module-architecture.drawio b/docs/media/diagrams/jenkins-module-architecture.drawio
new file mode 100644
index 0000000..ab593b5
--- /dev/null
+++ b/docs/media/diagrams/jenkins-module-architecture.drawio
@@ -0,0 +1 @@
+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
\ No newline at end of file
diff --git a/mkdocs.yml b/mkdocs.yml
index ae5fe61..d8a2764 100644
--- a/mkdocs.yml
+++ b/mkdocs.yml
@@ -58,9 +58,11 @@ theme:
info: octicons/info-16
logo: media/logos/aws-logo-white.svg
favicon: media/logos/aws-logo-white.svg
-
-markdown_extensions:
+
+markdown_extensions:
- admonition
+ - pymdownx.details
+ - pymdownx.superfences
- pymdownx.snippets:
base_path: "."
check_paths: true
@@ -73,6 +75,8 @@ markdown_extensions:
- pymdownx.emoji:
emoji_index: !!python/name:material.extensions.emoji.twemoji
emoji_generator: !!python/name:material.extensions.emoji.to_svg
+ - pymdownx.tabbed:
+ alternate_style: true
nav:
- Home:
diff --git a/modules/jenkins/README.md b/modules/jenkins/README.md
index ace25df..cf4b964 100644
--- a/modules/jenkins/README.md
+++ b/modules/jenkins/README.md
@@ -1,17 +1,319 @@
# Jenkins Module
-Jenkins is an open source automation server. It helps automate the parts of software development related to building, testing, and deploying, facilitating continuous integration, and continuous delivery.
+Jenkins is an open source automation server that simplifies repeatable software development tasks such as building, testing, and deploying applications. This module deploys Jenkins on an Elastic Container Service (ECS) cluster backed by AWS Fargate using the latest Jenkins container image ([jenkins/jenkins:lts-jdk17](https://hub.docker.com/r/jenkins/jenkins)). The deployment includes an Elastic File System (EFS) volume for persisting plugins and configurations, along with an Elastic Load Balancer (ELB) deployment for TLS termination. The module also includes the deployment of an EC2 Autoscaling group to serve as a flexible pool of build nodes, however the user must configure Jenkins to use the build farm.
This module deploys the following resources:
- An Elastic Container Service (ECS) cluster backed by AWS Fargate.
- An ECS service running the latest Jenkins container ([jenkins/jenkins:lts-jdk17](https://hub.docker.com/r/jenkins/jenkins)) available.
- An Elastic File System (EFS) for the Jenkins service to use as a persistent datastore.
+- A user defined number of ZFS File Systems
- An Elastic Load Balancer (ELB) for TLS termination of the Jenkins service
- A configurable number of EC2 Autoscaling groups to serve as a flexible pool of build nodes for the Jenkins service
- Supporting resources including KMS keys for encryption and IAM roles to ensure security best practices
+## Architecture
-
+![Jenkins Module Architecture](../../docs/media/Images/jenkins-module-architecture.png)
+
+## Prerequisites
+There are a few prerequisites to the deployment of Jenkins which are not directly provided in the module. The first is a public certificate used by the Jenkins ALB for SSL termination, instructions are included for provisioning the certificates using the AWS Certificates Manager (ACM) but certificates created else where can also be uploaded into the service for later use. The second is an AWS Secrets Manager secret which is used to store sensitive information, such as SSH keys, which can then be made available to Jenkins through a Plugin.
+
+
+### Create a Public Certificate
+
+??? note "How to Create a Public Certificate Using Amazon Certificate Manager"
+
+ 1. Sign in to the AWS Management Console and open the Amazon Certificate Manager (ACM) [console](https://console.aws.amazon.com/acm/home) and choose Request a certificate.
+
+ 2. In the Domain names section, type your domain name.
+
+ a. You can use a fully qualified domain name (FQDN), such as www.example.com, or a bare or apex domain name such as example.com. You can also use an asterisk (\*) as a wild card in the leftmost position to protect several site names in the same domain. For example, \*.example.com protects corp.example.com, and images.example.com. The wild-card name will appear in the Subject field and in the Subject Alternative Name extension of the ACM certificate.
+
+ b. When you request a wild-card certificate, the asterisk (\*) must be in the leftmost position of the domain name and can protect only one subdomain level. For example, \*.example.com can protect login.example.com, and test.example.com, but it cannot protect test.login.example.com. Also note that \*.example.com protects only the subdomains of example.com, it does not protect the bare or apex domain (example.com). To protect both, see the next step.
+
+ !!! note
+ In compliance with [RFC 5280](https://datatracker.ietf.org/doc/html/rfc5280), the length of the domain name (technically, the Common Name) that you enter in this step cannot exceed 64 octets (characters), including periods. Each subsequent Subject Alternative Name (SAN) that you provide, as in the next step, can be up to 253 octets in length.
+
+ c. To add another name, choose Add another name to this certificate and type the name in the text box. This is useful for protecting both a bare or apex domain (such as example.com) and its subdomains such as *.example.com).
+ 3. In the Validation method section, choose either DNS validation – recommended or Email validation, depending on your needs.
+
+ !!! note
+ If you are able to edit your DNS configuration, we recommend that you use DNS domain validation rather than email validation. [DNS validation](https://docs.aws.amazon.com/acm/latest/userguide/dns-validation.html) has multiple benefits over email validation. See DNS validation.
+
+ a. Before ACM issues a certificate, it validates that you own or control the domain names in your certificate request. You can use either email validation or DNS validation.
+
+ b. If you choose email validation, ACM sends validation email to three contact addresses registered in the WHOIS database, and up to five common system administration addresses for each domain name. You or an authorized representative must reply to one of these email messages. For more information, see [Email validation](https://docs.aws.amazon.com/acm/latest/userguide/email-validation.html).
+
+ c. If you use DNS validation, you simply add a CNAME record provided by ACM to your DNS configuration. For more information about DNS validation, see [DNS validation](https://docs.aws.amazon.com/acm/latest/userguide/dns-validation.html).
+
+ 4. In the Key algorithm section, chose one of the three available algorithms:
+ * RSA 2048 (default)
+ * ECDSA P 256
+ * ECDSA P 384
+
+ a. For information to help you choose an algorithm, see [Key algorithms](https://docs.aws.amazon.com/acm/latest/userguide/acm-certificate.html#algorithms) and the AWS blog post [How to evaluate and use ECDSA certificates in AWS Certificate Manager](https://aws.amazon.com/blogs/security/how-to-evaluate-and-use-ecdsa-certificates-in-aws-certificate-manager/).
+
+ 5.In the Tags page, you can optionally tag your certificate. Tags are key-value pairs that serve as metadata for identifying and organizing AWS resources. For a list of ACM tag parameters and for instructions on how to add tags to certificates after creation, see [Tagging AWS Certificate Manager certificates](https://docs.aws.amazon.com/acm/latest/userguide/tags.html). When you finish adding tags, choose Request.
+
+ 6. After the request is processed, the console returns you to your certificate list, where information about the new certificate is displayed.
+ a. A certificate enters status Pending validation upon being requested, unless it fails for any of the reasons given in the troubleshooting topic [Certificate request fails](https://docs.aws.amazon.com/acm/latest/userguide/troubleshooting-failed.html). ACM makes repeated attempts to validate a certificate for 72 hours and then times out. If a certificate shows status Failed or Validation timed out, delete the request, correct the issue with [DNS validation](https://docs.aws.amazon.com/acm/latest/userguide/dns-validation.html) or [Email validation](https://docs.aws.amazon.com/acm/latest/userguide/email-validation.html), and try again. If validation succeeds, the certificate enters status Issued.
+
+ !!! note
+ Depending on how you have ordered the list, a certificate you are looking for might not be immediately visible. You can click the black triangle at right to change the ordering. You can also navigate through multiple pages of certificates using the page numbers at upper-right.
+
+[Amazon Certificate Manager Documentation](https://docs.aws.amazon.com/acm/latest/userguide/gs-acm-request-public.html)
+
+### (Optional) Upload Secrets to AWS Secrets Manager
+
+AWS Secrets Manager can be used to store sensitive information such as SSH keys and access tokens, which can then be made available to Jenkins. We recommend using the service to store the private key for the Jenkins agents which the Jenkins orchestrator uses to communicate over SSH.
+
+!!! warning
+ To grant Jenkins access to the secrets stored in the AWS Secrets Manager, the `AWS Secrets Manager Credentials Provider` Jenkins plugin is recommended. **There are requirements around tagging your secrets for the plugin to work properly**. See the [AWS Secrets Manager Credentials Provider Plugin Documentation](https://plugins.jenkins.io/aws-secrets-manager-credentials-provider/) for additional details. Step-by-step instructions for uploading SSH keys into AWS Secrets Manager can be found in the [Configure Pluings]() section.
+
+??? note "How to Upload Secrets to AWS Secrets Manager"
+
+ 1. Open the Secrets Manager [console](https://console.aws.amazon.com/secretsmanager/).
+
+ 2. Choose Store a new secret.
+
+ 3. On the Choose secret type page, do the following:
+ 1. For Secret type, choose Other type of secret.
+
+ 1. In Key/value pairs, either enter your secret in JSON Key/value pairs, or choose the Plaintext tab and enter the secret in any format (you must choose Plaintext if storing SSH keys). You can store up to 65536 bytes in the secret.
+
+ 1. For Encryption key, choose the AWS KMS key that Secrets Manager uses to encrypt the secret value. For more information, see [Secret encryption and decryption](https://docs.aws.amazon.com/secretsmanager/latest/userguide/security-encryption.html).
+
+ - For most cases, choose aws/secretsmanager to use the AWS managed key for Secrets Manager. There is no cost for using this key.
+
+ - If you need to access the secret from another AWS account, or if you want to use your own KMS key so that you can rotate it or apply a key policy to it, choose a customer managed key from the list or choose Add new key to create one. For information about the costs of using a customer managed key, see [Pricing](https://docs.aws.amazon.com/secretsmanager/latest/userguide/intro.html#asm_pricing).
+
+ - You must have [Permissions for the KMS key](https://docs.aws.amazon.com/secretsmanager/latest/userguide/security-encryption.html#security-encryption-authz). For information about cross-account access, see [Access AWS Secrets Manager secrets from a different account](https://docs.aws.amazon.com/secretsmanager/latest/userguide/auth-and-access_examples_cross.html).
+
+ 1. Choose Next
+
+ 4. On the Configure secret page, do the following:
+ 1. Enter a descriptive Secret name and Description. Secret names must contain 1-512 Unicode characters.
+
+ 1. (Optional) In the Tags section, add tags to your secret. For tagging strategies, see [Tag AWS Secrets Manager secrets](https://docs.aws.amazon.com/secretsmanager/latest/userguide/managing-secrets_tagging.html). Don't store sensitive information in tags because they aren't encrypted.
+
+ 1. (Optional) In Resource permissions, to add a resource policy to your secret, choose Edit permissions. For more information, see [Attach a permissions policy to an AWS Secrets Manager secret](https://docs.aws.amazon.com/secretsmanager/latest/userguide/auth-and-access_resource-policies.html).
+
+ 1. (Optional) In Replicate secret, to replicate your secret to another AWS Region, choose Replicate secret. You can replicate your secret now or come back and replicate it later. For more information, see [Replicate secrets across Regions](https://docs.aws.amazon.com/secretsmanager/latest/userguide/replicate-secrets.html).
+
+ 1. Choose Next.
+
+
+
+ 5. (Optional) On the Configure rotation page, you can turn on automatic rotation. You can also keep rotation off for now and then turn it on later. For more information, see [Rotate secrets](https://docs.aws.amazon.com/secretsmanager/latest/userguide/rotating-secrets.html). Choose Next.
+
+ 6. On the Review page, review your secret details, and then choose Store.
+
+ Secrets Manager returns to the list of secrets. If your new secret doesn't appear, choose the refresh button.
+
+
+[AWS Secrets Manager Documentation](https://docs.aws.amazon.com/secretsmanager/latest/userguide/create_secret.html)
+
+### (Optional) Create Amazon Machine Image (AMI) for Jenkins Agent Using Packer
+
+The CGD Toolkit provides packer templates for generating Amazon Machine Images (AMIs) for use as Jenkins Agents. The Toolkit provides both Windows and Linux options
+
+??? note "How to Generate SSH Keys"
+
+ 1. Open your preferred Command Line App.
+
+ 2. Paste the text below, replacing the email used in the example with your email
+
+ ``` bash
+ # This command creates a new SSH key, using the provided email as a label
+ ssh-keygen -t ed25519 -C "your_email@example.com"
+ ```
+
+ 3. When prompted to "Enter a file in which to save the key", enter a path in which to save the generated key, or press **Enter** to accept the default location.
+
+ 4. When prompted to "Enter passphrase", enter a password for your key, or leave empty for no password.
+
+ !!! warning
+ If using the [AWS Secrets Manager Credentials Provider](https://plugins.jenkins.io/aws-secrets-manager-credentials-provider/), leave the passphrase empty. The AWS Secrets Manager Credentials Provider plugin does **NOT** support passphrases.
+
+??? note "How to Create Linux AMI using Packer"
+
+ 1. Copy the existing example Packer configuration file located at **assets/packer/build-agents/linux/example.pkvars.hcl**
+ 2. Replace placeholder values
+ * **region** - AWS Region code to deploy the AMI into. i.e. "us-west-2"
+ * **vpc_id** - The ID of the VPC you wish to use to create the AMI.
+ * **subnet_id** - The ID of the subnet you wish to use to create the AMI.
+ * **profile** - The name of the [AWS CLI profile](https://docs.aws.amazon.com/cli/latest/userguide/cli-configure-files.html) you wish to use to create the AMI.
+ * **public_key** - The public ssh key to be used by Jenkins when communicating with its agents.
+ 3. Download the Packer dependencies
+ ``` shell
+ # This command will download all necessary dependencies to build the AMI
+ packer init
+ ```
+
+ !!! note
+ If you do not have Hashicorp Packer installed, see [Packer Installation Instructions](https://developer.hashicorp.com/packer/tutorials/docker-get-started/get-started-install-cli).
+
+ 4. Build the Image
+ ``` shell
+ # This command builds the Linux x86_64 AMI using the configurations provided in the .pkvars.hcl you created above
+ packer build -var-file your-vars.pkvars.hcl amazon-linux-2023-x86_64.pkr.hcl
+ ```
+
+ 5. Notate the AMI id (**ami-#################**) returned from the previous command.
+ ``` shell
+ ==> Wait completed after 10 minutes 23 seconds
+
+ ==> Builds finished. The artifacts of successful builds are:
+ --> jenkins-linux-packer.amazon-ebs.al2023: AMIs were created:
+ us-east-1: ami-08862...e2e
+ ```
+
+## Installation
+```shell
+terraform init
+```
+
+## Deploy
+```shell
+# Review architecture to be deployed
+terraform plan -var-file=deployment-variables.tfvars
+
+# Deploy terraform module
+# It is recommended that you use a .tfvars file to pass variables to the module when deploying
+terraform deploy -var-file=deployment-variables.tfvars
+```
+
+## Accessing Jenkins
+
+Once deployed, the Jenkins service can be accessed through its associated load balancer. The service is accessible through the ALB on ports 443 and 80 by default. The module does **NOT** allowlist any IP addresses by default. Users must edit the ALB security group to grant access to the Jenkins Service.
+
+You can find the DNS address for the ALB created by the module by running the following command:
+
+``` bash
+# This command pulls the DNS address of the ALB associated with the Jenkins ECS service
+terraform output jenkins_alb_dns_name
+```
+
+## Configuring Jenkins
+
+When accessing Jenkins for the first time, an administrators password is required. This password is auto-generated and available through the ECS logs. The administrative user will be replaced with a new user upon completion of the setup, see [Creating Users](README.md#creating-users).
+
+
+### Retrieve the Jenkins Administrator Password
+
+1. Open the AWS console and navigate to the [Elastic Container Service (ECS) console](https://console.aws.amazon.com/ecs).
+2. In the `Clusters` tab, select the name of the cluster you created
+3. Select the name of the jenkins service you created
+4. Select the `Logs` tab
+5. Scroll through the logs until you find the password, below is an example of what the password section looks like. Note that each line is shown as its own log entry in the console.
+
+![Jenkins Admin Password](../../media/Images/jenkins-admin-password.png)
+
+### Jenkins Initial Configuration
+
+1. Open the Jenkins console on your preferred browser, see [Accessing Jenkins](README.md#accessing-jenkins) for details.
+2. Paste the password you retrieved from the logs in the previous step into the text box and click **Continue**
+3. You will then be prompted to select the plugins you wish to install.
+ a. If you are unsure of which plugins to install, select `Install suggested plugins`.
+ b. Otherwise, select `Select plugins to install` and choose your preferred plugins.
+4. You are then prompted to create your first admin user.
+ a. Enter a username for your new user
+ b. Enter a password
+ c. Enter your full name
+ d. Enter you email
+ e. Click `Save and Continue`
+5. For the Jenkins URL, accept the default value by clicking `Save and Finish`.
+6. Click `Start using Jenkins`
+
+### Configuring Plugins
+
+There are 2 plugins recommended for the solutions: The [EC2 Fleet](https://plugins.jenkins.io/ec2-fleet/) Plugin and the [AWS Secrets Manager Credentials Provider](https://plugins.jenkins.io/aws-secrets-manager-credentials-provider/) Plugin. The `EC2 Fleet` Plugin is used to integrate Jenkins with AWS and allows EC2 instances to be used as build nodes through an autoscaling group. The `AWS Secrets Manager Credentials Provider` Plugin will allow users to store their credentials in AWS Secrets Manager and seamlessly access them in Jenkins.
+
+#### Install the Plugins
+
+1. Open the Jenkins console.
+2. On the left-hand side, select the `Manage Jenkins` tab.
+3. Then, under the `System Configuration` section, select `Plugins`.
+4. On the left-hand side, select ` Available plugins`.
+5. Using the search bar at the top of the page, search for `EC2 Fleet`.
+6. Select the `EC2 Fleet` plugin.
+7. Using the search bar at the top of the page, search for `AWS Secret Manager Credentials Provider`.
+8. Select the `AWS Secret Manager Credentials Provider` plugin.
+9. Click `install` on the top-right corner of the page.
+10. Once the installation is complete, Select `Go back to the top page` at the bottom of the page
+
+#### Create the Necessary Credentials
+
+!!! note ""
+ === "Using the Jenkins Console"
+ 1. From the Jenkins homepage, on the left-hand side, select `Manage Jenkins`
+ 2. Under the `Security` section, select `Credentials`
+ 3. Under `Stores scoped to Jenkins`, select `System`
+ 4. Select `Global credentials (unrestricted)`
+ 5. In the top right corner, click the `Add Credentials` button
+ 6. For the `Kind` dropdown, select `SSH Username with private key`
+ 7. For `ID` enter a name for your credentials
+ 8. For `Description` add a description of your credential
+ 9. For `Username`, enter the username to be used for the SSH connection
+ 10. For `Private Key`, select the `Enter directly` radio button.
+ 11. In the next section displayed, select the `Add` button
+ 12. Paste the Private Key created earlier into the text box.
+ 13. For `Passphrase` enter the Passphrase for the SSH key, if no passphrase was entered when creating the keys, leave this blank
+ 14. Note the `ID` of your newly created credentials. This will be referenced in the next section.
+
+ === "Using the AWS Secrets Manager Plugin"
+
+ 1. Open the Secrets Manager [console](https://console.aws.amazon.com/secretsmanager/).
+
+ 2. Choose Store a new secret.
+
+ 3. On the Choose secret type page, do the following:
+ 1. For Secret type, choose Other type of secret.
+
+ 1. Select the `Plaintext` test tab, select all text in the textbox and delete it.
+
+ 1. Paste your Private key into the textbox
+
+ 1. Choose Next
+
+ 4. On the Configure secret page, do the following:
+ 1. Enter a descriptive Secret name and Description. Note that the name chosen here will also be used as the name of the credentials within Jenkins.
+
+ 1. In the Tags section, add the 2 required tags for the [AWS Secrets Manager Credentials Provider](https://plugins.jenkins.io/aws-secrets-manager-credentials-provider/) Plugin
+ * `jenkins:credentials:type` = `sshUserPrivateKey`
+ * `jenkins:credentials:username` = ``
+
+
+ !!! info
+ The username will depend on the image being used for the build agent.
+
+ * Amazon Linux -> `ec2-user`
+ * Ubuntu -> `ubuntu`
+ * Windows -> `Administrator`
+
+
+ 1. Choose Next.
+
+ 6. On the Review page, review your secret details, and then choose Store.
+
+#### Connect Jenkins to the Build Farm
+
+1. From the Jenkins homepage, on the left-hand side, choose `Manage Jenkins`.
+2. Under the `System Configuration` section, choose `Clouds`
+3. Select `New Cloud`
+3. Enter a name for your cloud configuration
+4. Select `Amazon EC2 Fleet`
+5. Click `Create`
+6. On the `New Cloud` configuration page, change the following settings.
+ 1. **Region** - Select the region in which you deployed the module
+ 1. **EC2 Fleet** - Select the autoscaling group created by the module
+ 1. **Launcher** - Select `Launch agents via SSH`
+ 1. **Launcher** -> **Credentials** - Select the credentials created in the previous step
+ 1. **Launcher** -> **Host Key Verification Strategy** - Select `Non verifying Verification Strategy`
+ 1. **Connect to instaces via private IP instead of public IP** - Select the `Private IP` check box
+ 1. **Max Idle Minutes Before Scaledown** - Set this variable to `5` (minutes). Feel free to change this based on your needs.
+
+
## Requirements
| Name | Version |
@@ -122,7 +424,7 @@ No modules.
| [jenkins\_agent\_secret\_arns](#input\_jenkins\_agent\_secret\_arns) | A list of secretmanager ARNs (wildcards allowed) that contain any secrets which need to be accessed by the Jenkins service. | `list(string)` | `null` | no |
| [jenkins\_alb\_access\_logs\_bucket](#input\_jenkins\_alb\_access\_logs\_bucket) | ID of the S3 bucket for Jenkins ALB access log storage. If access logging is enabled and this is null the module creates a bucket. | `string` | `null` | no |
| [jenkins\_alb\_access\_logs\_prefix](#input\_jenkins\_alb\_access\_logs\_prefix) | Log prefix for Jenkins ALB access logs. If null the project prefix and module name are used. | `string` | `null` | no |
-| [jenkins\_alb\_subnets](#input\_jenkins\_alb\_subnets) | A list of subnets to deploy the Jenkins load balancer into. Public subnets are recommended. | `list(string)` | n/a | yes |
+| [jenkins\_alb\_subnets](#input\_jenkins\_alb\_subnets) | A list of subnet ids to deploy the Jenkins load balancer into. Public subnets are recommended. | `list(string)` | n/a | yes |
| [jenkins\_cloudwatch\_log\_retention\_in\_days](#input\_jenkins\_cloudwatch\_log\_retention\_in\_days) | The log retention in days of the cloudwatch log group for Jenkins. | `string` | `365` | no |
| [jenkins\_efs\_performance\_mode](#input\_jenkins\_efs\_performance\_mode) | The performance mode of the EFS file system used by the Jenkins service. Defaults to general purpose. | `string` | `"generalPurpose"` | no |
| [jenkins\_efs\_throughput\_mode](#input\_jenkins\_efs\_throughput\_mode) | The throughput mode of the EFS file system used by the Jenkins service. Defaults to bursting. | `string` | `"bursting"` | no |
diff --git a/modules/jenkins/variables.tf b/modules/jenkins/variables.tf
index 9935a98..c17021c 100644
--- a/modules/jenkins/variables.tf
+++ b/modules/jenkins/variables.tf
@@ -87,7 +87,7 @@ variable "cluster_name" {
# - Load Balancer -
variable "jenkins_alb_subnets" {
type = list(string)
- description = "A list of subnets to deploy the Jenkins load balancer into. Public subnets are recommended."
+ description = "A list of subnet ids to deploy the Jenkins load balancer into. Public subnets are recommended."
}
variable "enable_jenkins_alb_access_logs" {