Skip to content

Commit 8ef463e

Browse files
authored
feat!: 🏗 restructure using docker (#24)
- Docker is now used for deployment. - Route 53 functionality has been removed.
1 parent 368ec94 commit 8ef463e

15 files changed

+285
-171
lines changed

.env.sample

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
# <REQUIRED> Name used for docker images/aws resources
2+
APP_NAME=aws-certbot
3+
4+
# <REQUIRED> AWS access key
5+
AWS_ACCESS_KEY_ID=
6+
7+
# <REQUIRED> AWS secret access key
8+
AWS_SECRET_ACCESS_KEY=
9+
10+
# <REQUIRED> AWS region to use
11+
AWS_DEFAULT_REGION=us-east-1
12+
13+
# <REQUIRED> Cloudflare API key with edit.zone permissions
14+
DNS_CLOUDFLARE_API_TOKEN=
15+
16+
# <REQUIRED> A list of domains separated by commas and semicolons.
17+
# The semicolon separates groups of domains, while commas separate
18+
# individual domains. For example:
19+
# domain.com,*.domain.com;example.io,staging.example.io
20+
DOMAIN_LIST=
21+
22+
# <REQUIRED> Email to associate with SSL certificate
23+
DOMAIN_EMAIL=[email protected]
24+
25+
# <REQUIRED> Number of days before expiration to request renewal
26+
DAYS_BEFORE_EXPIRATION=20

.gitattributes

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
* text=auto eol=lf

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -2,3 +2,4 @@
22
/dist
33
*.ini
44
__pycache__
5+
.env

.releaserc

-6
This file was deleted.

Dockerfile

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
FROM ubuntu:22.04
2+
3+
# Dependencies (needed for scripts as well)
4+
RUN apt-get update && \
5+
apt-get install -y unzip jq curl
6+
7+
# Install docker
8+
RUN curl -fsSL https://get.docker.com -o get-docker.sh && \
9+
sh get-docker.sh
10+
11+
# Install AWS CLI
12+
RUN ARCH=$(dpkg --print-architecture) && \
13+
if [ "$ARCH" = "amd64" ]; then \
14+
curl "https://awscli.amazonaws.com/awscli-exe-linux-x86_64.zip" -o "awscliv2.zip" && \
15+
unzip awscliv2.zip && \
16+
./aws/install; \
17+
elif [ "$ARCH" = "arm64" ]; then \
18+
curl "https://awscli.amazonaws.com/awscli-exe-linux-aarch64.zip" -o "awscliv2.zip" && \
19+
unzip awscliv2.zip && \
20+
./aws/install; \
21+
else \
22+
echo "Unsupported architecture: $ARCH"; \
23+
fi
24+
25+
# Setup workspace
26+
WORKDIR /app
27+
28+
COPY . /app
29+
30+
# Give executable permissions
31+
RUN chmod +x /app/run.sh
32+
RUN chmod +x /app/deploy.sh
33+
34+
# Entry point to run Docker commands on the host
35+
CMD ["./run.sh"]

README.md

+59-52
Original file line numberDiff line numberDiff line change
@@ -1,66 +1,73 @@
11
# AWS-Certbot ![tests](https://github.com/j3ko/aws-certbot/actions/workflows/test.yml/badge.svg)
2-
Auto renew and update [letsencrypt.org](https://letsencrypt.org) SSL certificates provisioned on [ACM](https://aws.amazon.com/certificate-manager/).
2+
Auto renew [letsencrypt.org](https://letsencrypt.org) SSL certificates provisioned on [ACM](https://aws.amazon.com/certificate-manager/).
33

44
## Requirements
5-
- [AWS CLI v2](https://aws.amazon.com/cli/)
6-
- [PowerShell 7.1+](https://docs.microsoft.com/en-us/powershell/scripting/install/installing-powershell?view=powershell-7.1)
7-
- Registered domain ([namecheap](https://www.namecheap.com/), [godaddy](https://godaddy.com/), etc..)
8-
- DNS service ([cloudflare](https://www.cloudflare.com/), [route53](https://aws.amazon.com/route53/), etc..)
9-
10-
## Setup
11-
12-
### Introduction
13-
Certbot integrates with many popular [DNS services](https://certbot.eff.org/docs/using.html?highlight=dns#dns-plugins) to verify SSL certificate challenges automatically via API. AWS-Certbot currently handles Route53 and Cloudflare integration. Cloudflare has the advantage of offering free DNS services for personal/hobby project sites. For integrations with other services, please open a [Feature Request](https://github.com/j3ko/aws-certbot/issues/new?assignees=j3ko&labels=enhancement&template=feature_request.md&title=).
14-
15-
### Cloudflare Setup
16-
1. [Register](https://dash.cloudflare.com/sign-up) for a Cloudflare account
17-
1. Configure your domain registrar to use Cloudflare as your DNS service provider. This is specific for each registrar ([namecheap specific guide](https://www.namecheap.com/support/knowledgebase/article.aspx/9607/2210/how-to-set-up-dns-records-for-your-domain-in-cloudflare-account/)).
18-
1. [Create](https://developers.cloudflare.com/api/tokens/create) a Cloudflare API token. It is recommended to generate a token with the minimum required permissions (ie. read/write access to the DNS zone you want AWS-Certbot to handle).
19-
1. Rename `cloudflare.ini.sample` to `cloudflare.ini` and replace the placeholder `<API TOKEN HERE>` with your newly created API token.
20-
21-
### Route53 Setup
22-
1. [Create](https://docs.aws.amazon.com/Route53/latest/DeveloperGuide/CreatingHostedZone.html) a new hosted zone for your domain in AWS Route53
23-
1. Configure your domain registrar to use Route53 as your DNS service provider. This is specific for each registrar ([namecheap specific guide](https://www.namecheap.com/support/knowledgebase/article.aspx/10371/2208/how-do-i-link-my-domain-to-amazon-web-services/))
24-
1. Uncomment the following block in `cloud.yaml`:
5+
- [Docker v24+](https://docs.docker.com/engine/)
6+
7+
## Quick Start
8+
1. ```
9+
git clone [email protected]:j3ko/aws-certbot.git
10+
```
11+
12+
1. ```
13+
cd aws-certbot
14+
```
15+
16+
1. Edit `.env.sample` and fill in the [required fields](#environment-variables)
17+
18+
1. Build the docker image
19+
20+
```bash
21+
docker build -t aws-certbot-builder .
2522
```
26-
# - PolicyName: Route53ListZones
27-
# PolicyDocument:
28-
# Version: "2012-10-17"
29-
# Statement:
30-
# - Effect: Allow
31-
# Action:
32-
# - "route53:ListHostedZones"
33-
# Resource: "*"
34-
# - PolicyName: Route53ModifyZones
35-
# PolicyDocument:
36-
# Version: "2012-10-17"
37-
# Statement:
38-
# - Effect: Allow
39-
# Action:
40-
# - "route53:GetChange"
41-
# - "route53:ChangeResourceRecordSets"
42-
# Resource:
43-
# - "arn:aws:route53:::change/*"
44-
# - "arn:aws:route53:::hostedzone/<HOSTED ZONE ID HERE>"
23+
1. Run **aws-certbot** locally
24+
25+
```bash
26+
docker run -it --rm --env-file=./.env.sample -v /var/run/docker.sock:/var/run/docker.sock aws-certbot-builder
4527
```
46-
1. Replace the `<HOSTED ZONE ID HERE>` placeholder with your newly created hosted zone id
4728

48-
## Deployment
49-
Run
50-
```
51-
.\deploy.ps1
52-
```
29+
### What does it do?
30+
Running **aws-certbot** locally will:
31+
1. Check ACM to see if any domains in `DOMAIN_LIST` are expiring soon.
32+
1. If domains are missing or expiring, certbot runs and generates a new SSL certificate
33+
1. Any newly generated certificates are uploaded to ACM
5334

54-
## Configuration
55-
Several configuration variables are available on the AWS-Certbot lambda function:
35+
## Environment Variables
36+
| Variable | Description | Required? |
37+
|---|---|---|
38+
| APP_NAME | Name used for docker images/AWS resources ||
39+
| AWS_ACCESS_KEY_ID | AWS access key ||
40+
| AWS_SECRET_ACCESS_KEY | AWS secret access key ||
41+
| AWS_DEFAULT_REGION | AWS region to use ||
42+
| DOMAIN_LIST | A list of domains separated by commas and semicolons. The semicolon separates groups of domains, while commas separate individual domains. For example: `domain.com,*.domain.com;example.io,staging.example.io` ||
43+
| DOMAIN_EMAIL | Cloudflare API key with edit.zone permissions ||
44+
| DAYS_BEFORE_EXPIRATION | Number of days before expiration to request renewal ||
45+
46+
## Deploying to AWS
47+
48+
1. Edit `.env.sample` and fill in the [required fields](#environment-variables)
49+
50+
1. Build the docker image
51+
52+
```bash
53+
docker build -t aws-certbot-builder .
54+
```
55+
1. Deploy **aws-certbot** to AWS
56+
57+
```bash
58+
docker run -it --rm --env-file=./.env.sample -v /var/run/docker.sock:/var/run/docker.sock aws-certbot-builder ./deploy.sh
59+
```
5660

57-
`CERTBOT_BUCKET` - Bucket name containing AWS-Certbot code
61+
### What does it do?
5862

59-
`DOMAIN_EMAIL` - Email address to use for [letsencrypt.org](https://letsencrypt.org) registration
63+
1. The **aws-certbot** docker image is built and uploaded to [ECR](https://aws.amazon.com/ecr/).
64+
1. The cloud formation defined in `cloud.yaml` is deployed to run the docker image as a lambda function.
65+
1. A timer is defined in `cloud.yaml` to execute the lambda function once a day.
6066

61-
`DOMAIN_LIST` - Comma separated list of domains/subdomains to enlist for automatic renewal (eg. `foo.com,sub.foo.com`). Multiple domains are separated by semi-colons (eg. `foo.com,sub.foo.com;bar.com,*.bar.com`)
67+
## Known Issues
6268

63-
`CERTS_RENEW_DAYS_BEFORE_EXPIRATION` - Number of days before expiration to attempt renewal
69+
- Only Cloudflare-managed domains can currently be used.
70+
- Cloudflare API key is visible in lambda environment variables.
6471

6572
## Credits
6673
AWS-Certbot is based largely on the following amazing projects:

certbot-1.17.0.zip

-21.2 MB
Binary file not shown.

cloud.yaml

+26-52
Original file line numberDiff line numberDiff line change
@@ -2,26 +2,24 @@ AWSTemplateFormatVersion: 2010-09-09
22
Description: AWS Certbot - ACM certbot autorenewal
33

44
Parameters:
5-
BucketName:
6-
Description: S3 Bucket (Certbot configuration tree and lambda source package)
5+
EcrRepositoryUri:
76
Type: String
87

9-
ObjectVersion:
10-
Description: Generated object version
8+
DnsCloudflareApiToken:
119
Type: String
10+
NoEcho: true
1211

1312
DomainList:
14-
Description: List of managed domains (comma separated)
1513
Type: String
1614

1715
DomainEmail:
18-
Description: Email address used in cert request
1916
Type: String
2017

21-
CertRenewDays:
22-
Description: Number of days before cert expiration to request renew
18+
DaysBeforeExpiration:
19+
Type: String
20+
21+
Timestamp:
2322
Type: String
24-
Default: "20"
2523

2624
Resources:
2725
AwsCertbotRole:
@@ -47,16 +45,6 @@ Resources:
4745
- "logs:CreateLogStream"
4846
- "logs:PutLogEvents"
4947
Resource: arn:aws:logs:*:*:*
50-
- PolicyName: BucketGetPutObject
51-
PolicyDocument:
52-
Version: "2012-10-17"
53-
Statement:
54-
- Effect: Allow
55-
Action:
56-
- "s3:GetObject"
57-
- "s3:PutObject"
58-
Resource:
59-
Fn::Join: ["", ["arn:aws:s3:::", Ref: "BucketName", "/*"]]
6048
- PolicyName: ACMGetImportCerts
6149
PolicyDocument:
6250
Version: "2012-10-17"
@@ -67,51 +55,24 @@ Resources:
6755
- "acm:ListCertificates"
6856
- "acm:ImportCertificate"
6957
Resource: "*"
70-
# - PolicyName: Route53ListZones
71-
# PolicyDocument:
72-
# Version: "2012-10-17"
73-
# Statement:
74-
# - Effect: Allow
75-
# Action:
76-
# - "route53:ListHostedZones"
77-
# Resource: "*"
78-
# - PolicyName: Route53ModifyZones
79-
# PolicyDocument:
80-
# Version: "2012-10-17"
81-
# Statement:
82-
# - Effect: Allow
83-
# Action:
84-
# - "route53:GetChange"
85-
# - "route53:ChangeResourceRecordSets"
86-
# Resource:
87-
# - "arn:aws:route53:::change/*"
88-
# - "arn:aws:route53:::hostedzone/<HOSTED ZONE ID HERE>"
8958

9059
AwsCertbotFunction:
9160
Type: AWS::Lambda::Function
9261
Properties:
93-
Runtime: python3.6
9462
Description: Request and Renew Certs in ACM
95-
Handler: main.handler
63+
PackageType: Image
9664
Role:
9765
Fn::GetAtt: ["AwsCertbotRole", "Arn"]
9866
Environment:
9967
Variables:
100-
CERTBOT_BUCKET:
101-
Ref: BucketName
102-
DOMAIN_EMAIL:
103-
Ref: DomainEmail
104-
DOMAIN_LIST:
105-
Ref: DomainList
106-
CERTS_RENEW_DAYS_BEFORE_EXPIRATION:
107-
Ref: CertRenewDays
68+
DNS_CLOUDFLARE_API_TOKEN: !Ref DnsCloudflareApiToken
69+
DOMAIN_EMAIL: !Ref DomainEmail
70+
DOMAIN_LIST: !Ref DomainList
71+
DAYS_BEFORE_EXPIRATION: !Ref DaysBeforeExpiration
10872
Timeout: 120
10973
MemorySize: 512
11074
Code:
111-
S3Bucket:
112-
Ref: BucketName
113-
S3Key:
114-
Fn::Sub: "certbot-${ObjectVersion}.zip"
75+
ImageUri: !Sub "${EcrRepositoryUri}:latest"
11576

11677
AwsCertbotFunctionEvent:
11778
Type: AWS::Events::Rule
@@ -137,3 +98,16 @@ Resources:
13798
Fn::GetAtt:
13899
- "AwsCertbotFunctionEvent"
139100
- "Arn"
101+
102+
Outputs:
103+
AwsCertbotFunctionArn:
104+
Description: ARN of the AWS Certbot Lambda function
105+
Value: !GetAtt AwsCertbotFunction.Arn
106+
107+
AwsCertbotFunctionName:
108+
Description: Name of the AWS Certbot Lambda function
109+
Value: !Ref AwsCertbotFunction
110+
111+
EventRuleName:
112+
Description: Name of the AWS Certbot Lambda event rule
113+
Value: !Ref AwsCertbotFunctionEvent

cloudflare.ini.sample

-2
This file was deleted.

deploy.ps1

-36
This file was deleted.

0 commit comments

Comments
 (0)