Automating ACMEv2 Certificate Management on BIG-IP
Introduction
While we often associate and confuse Let's Encrypt with ACMEv2, the former is ultimately a consumer of the latter. The "Automated Certificate Management Environment" (ACME) protocol describes a system for automating the renewal of PKI certificates. The ACME protocol can be used with public services like Let's Encrypt, but also with internal certificate management services. In this article we explore the more generic support of ACME (version 2) on the F5 BIG-IP.
Back in March of 2023, Google proposed a significant reduction in the lifespan of Internet certificates, from the average 13 months down to just 90 days. Most recently (April 2026) Apple announced a proposal to reduce certificate validity down to just 47 days. Put plainly, the industry believes (rightly so) that the longer a certificate is valid, the less reliable it becomes. But perhaps the most important takeaway from all of this is, that while shorter-lived certificates are a good thing in principle, the current, common, manual approach to managing these 90-day (and shorter) certificates will become all but untenable. There is practically no scenario where an enterprise would or should employ, nor rely, on any human person, to manage certificates that live for 3 months. This is a recipe for disaster, and all the more reason to embrace certificate automation.
In this article, I’m specifically going to address this need for the F5 BIG-IP using the most well-known of the certificate automation protocols, ACMEv2. For what it’s worth, certificate automation in principle isn’t that complicated. There are a number of protocols (i.e., ACME, SCEP, CMP, EST), a variety of implementations for each of these, and lots of documentation that makes it seem like rocket science. But in truth, these protocols and processes all do more or less three things:
- Make a request to a certificate authority for a new certificate
- Satisfy a challenge to verify ownership of the certificate/domain
- Retrieve and install the new certificate
The “client” in this case is some utility, typically within arms-reach of the TLS server, that will perform the automation functions and push the new certificate(s) to the server.
I will propose here, an ACMEv2 certificate automation solution that runs ON the BIG-IP, that performs all of the necessary and ancillary functions needed to support an enterprise BIG-IP certificate renewal strategy, and supports most of the ACMEv2 8555 RFC functionality, not just what is required for Let’s Encrypt. The project is called Kojot Acme and is basically two things:
- A wrapper for the Dehydrated ACMEv2 client.
- A utility to support all of the additional, ancillary functions, including installation, configuration, authentication, revocation, scheduling, and reporting.
Now, in lieu of repeating everything that is thoroughly documented on the project page, I thought it more fun to come at this from a different angle. In the below, we’ll set up a local ACMEv2 server, install and configure Kojot ACME on the BIG-IP, and start generating certificates. Once you see how it works, it’s a super simple manipulation of a config file to point to a public ACMEv2 server to get real certificates (ex. Let’s Encrypt, ZeroSSL, Buypass, etc.). But also, for those of you in military bunkers miles below the surface, where no sunlight or Internet is accessible to you, you’ll likely have your own ACMEv2 servers installed somewhere in a cave nearby. Kojot Acme will work for that too. Kojot is heavily influenced by Jason Rahm's pivotal ACMEv2 client work. Let’s get started!
A Quick Review of the ACMEv2 Protocol
Before we go stumbling through this experience, it is perhaps best to step back and take another look at the ACMEv2 protocol as a whole so that we know what’s happening. I’ll start with a ridiculously simple flow diagram, as described in the introduction.
The client asks for a new certificate, the server asks the client to prove ownership, and then the server issues a new certificate. That’s basic ACME protocol flow. Now let’s overlay the above with the TLS server, the thing that actually needs the cert.
Let’s unpack this.
- Some process needs to know when to renew the certificate(s). This isn’t expressly required of the ACME client, but it’s not uncommon for the ACME client to poll the TLS server’s certificate status.
- When the ACME client decides that it needs to renew a certificate, it contacts the ACME server.
- In turn, two things need to happen:
- The ACME server needs to challenge the requestor to prove that that requestor owns this domain. Without getting too deep into the weeds, ACMEv2 defines multiple challenge types: http-01, dns-01, and tls-01. The latter two are useful, but the http-01 type is predominant, so we’ll focus on this one. In this case, the ACME server sends the client a unique token.
- As a function of the http-01 challenge, the ACME server will use public DNS to resolve the IP of the TLS server stated in the original new certificate request, then make an HTTP request to that IP at a specifically defined URL. The ACME server will expect the HTTP server to respond with the token that was provided in step 3a. So then, the ACME client will use a “hook” function to push that token to the HTTP server so that it can be ready for the ACME server request. If we were talking about an Apache or NGINX web server, the hook would imply placing the token in a folder on the TLS server and potentially configuring the web server to listen on the ACME challenge URL.
 
- If the ACME challenge was successful, the ACME server will let the ACME client know that it’s working on generating a new certificate, and eventually, where to download the new certificate.
- And finally, once the ACME client has the new certificate, it’ll use another hook function to update that certificate on the TLS server. Again, if talking about an Apache or NGINX web server, this might also mean restarting the web server domain to pick up the new information.
The above is of course by no means exhaustive. You can dig much deeper in the ACME Protocol Flow section on the project page or go directly to RFC 8555. In any case, this should give you a pretty good idea of how ACMEv2 works, so let’s move on to building stuff!
ACMEv2 Server Setup
The only requirements you need to satisfy for this local ACMEv2 server installations are:
- Docker and Docker Compose
- Network connectivity to the BIG-IP
The latter requirement implies that a) the ACME client running on the BIG-IP can reach the ACME server, and b) that the ACME server can reach the BIG-IP virtual server hosting the challenge response (token). We’ll use internal static Host information to simulate public DNS. To get started, log into the terminal of the Docker/Docker Compose server you’re going to use, and either:
- Git clone the project and then change directory to the acme-servers folder.
- Navigate to the acme-servers folder in the project page and copy one of the YAML files to a file.
You’ll have two ACMEv2 server options. Pebble is an open-source derivative managed by Let’s Encrypt, so will have similar functionality. Smallstep CA is an ACMEv2 server managed by Smallstep. Whichever one you choose, edit the corresponding YAML file. The ONLY thing you need to change inside this file is the set of “echo” statements near the bottom. These are setting up /etc/hosts entries in the ACME server container and should point your hostnames to the corresponding BIG-IP virtual server IPs. Recall in earlier review, this will be step 3b, where the ACME server resolves the IP of the TLS server and makes an ACME challenge request. That IP needs to be hosted by a BIG-IP virtual server. In lieu of pasting all of that YAML here, you can follow either of these links to show the example YAML:
In both cases you'll see a section near the bottom that looks like this:
## Insert local hosts DNS entries (if needed for internal testing)
echo "172.16.1.151   foo.f5labs.com" >> /etc/hosts
echo "172.16.1.152   bar.f5labs.com" >> /etc/hosts
echo "172.16.1.153   www.f5labs.com" >> /etc/hostsUpdate the echo statements to match the IP address of the BIG-IP port 80 VIP that's handling the ACME challenge, and the corresponding certificate subject name that this VIP is going to manage. In a real world scenario, you might have multiple HTTPS applications hosted on the BIG-IP, each with a unique IP address. You'll need a port 80 HTTP VIP for each of those unique applications. Again, the ACME server will do a DNS lookup for the site based on the certificate name, and then make its challenge to that IP address, on HTTP port 80. Everything else in the YAML file is configured to generate certificates from a private CA. Save the YAML file and launch the container with Docker Compose:
docker-compose -f docker-compose-smallstep-ca.yaml up -dor
docker-compose -f docker-compose-pebble-ca.yaml up -dFrom any machine in network range of this server, you can test to see if the ACME server is running:
[Pebble] curl -vk https://<server-ip>:14000/dir
[SmallStep] curl -vk https://<server-ip>:9000/acme/acme/directoryIf it’s working, you’ll get JSON response with a list of ACMEv2 protocol message types and associated URLs. Example:
{
  "newNonce": "https://172.16.1.112:9000/acme/acme/new-nonce",
  "newAccount": "https://172.16.1.112:9000/acme/acme/new-account",
  "newOrder": "https://172.16.1.112:9000/acme/acme/new-order",
  "revokeCert": "https://172.16.1.112:9000/acme/acme/revoke-cert",
  "keyChange": "https://172.16.1.112:9000/acme/acme/key-change"
}That’s it, we’re done here. Let’s move on to the BIG-IP.
Kojot Acme Installation
I don’t want to scare you, but this process is only going to take THREE quick steps:
- Install Kojot Acme – SSH to the BIG-IP console and run the following command:
curl -s https://raw.githubusercontent.com/f5devcentral/kojot-acme/main/install.sh | bashThis is going to install a few BIG-IP objects and create a working folder at /shared/acme. I realize at this phase some secret squirrel types might not be able to access this script from the BIG-IP. One option would be to download the project to ZIP file, expand on the BIG-IP, then run the install.sh script locally.
- Modify the global configuration data group – The installation creates a data group named dg_acme_config. The purpose of this data group is to store “global” configuration information. Basically, it’s a list of all of the managed certificate domain names and corresponding ACME server information. Let’s say you own a domain foo.com, you want to manage a certificate for www.foo.com, and you’re using the local Pebble ACME server to provide renewable certificates. Add the following information to the data group:
www.foo.com := --ca https://<pebble-server-ip>:14000/dirFor the sake of this demonstration, you can rely on the defaults, but you can add additional parameters here, like a specific “client” configuration file (--config), and preferred certificate algorithm (-a rsa, -a prime256v1, -a secp384r1). The default client configuration file is at /shared/acme/config, and also explained in detail in the project README page under Configuration Details.
- Create an HTTP virtual server – As previously stated, in a real world scenario you’d have a “TLS server” represented by a BIG-IP HTTPS virtual server. But for the ACMEv2 challenge to work, you only really need an HTTP VIP for now *and a corresponding ACMEv2 challenge iRule). The IP of this VIP needs to match the IP added in the YAML file, and the hostname in the global configuration data group needs to match the corresponding hostname in the YAML file. Once that’s created, you might want to test that you can ping the HTTP virtual server from the ACME server. For example:
docker exec -it pebble /bin/sh
ping www.foo.comIf you can ping the BIG-IP port 80 VIP from the ACME server, you're good to go. At this point you can trigger an ACMEv2 certificate fetch from the BIG-IP SSH console:
cd /shared/acme
./f5acmehandler.sh --verboseThe --verbose flag will let you see all of the communications between the BIG-IP and the ACME server. And when that’s done, you should have a new certificate installed on the BIG-IP. If this were a production environment where the certificate already exists and is associated with a client SSL profile, the utility uses a transaction to make sure the certificate, and potentially a new private key, are successfully applied. Finally, when you’re convinced that the ACME client is working as needed, you can modify the global configuration data group to point to production ACME servers. Additional information on scheduling, authentication, revocation, and reporting are all documented in the project page.
Both of the provided ACME servers adhere to RFC 8555 functionality, so what works for these will also work for the Internet-based ACMEv2 providers.
20 Comments
- JRahmAdmin Awesome work, Kevin_Stewart !! 
- Nice Kevin. Im using dehydrated with a self-made hook for integrate everything in bigip. 
 I don't have time, but I'd like to give it a try.
- Mario_FrancoAltocumulus Excellent, but I wonder if this script could work with any CA. A few years ago, I configured a script that works with Let's Encrypt, and the entire script is on the BigIP. However, it doesn't work with other CAs, and some clients want to use ACME with other CAs, like DigiCert or Sectigo - Kevin_StewartEmployee This script is intended to work with the http-01 specification of RFC 8555, which Let's Encrypt adheres to. Other public ACMEv2 providers include ZeroSSL, BuyPass, SSL.com, Sectigo, and Google ACMEv2. The script was also tested extensively with "local" ACMEv2 servers (Pebble and SmallStep Step-CA). It's important to point out here that CAs have to support ACMEv2 for this to work, which is the list I've included above. To my knowledge, DigiCert, Symantec, Comodo, GoDaddy, GlobalSign (and I'm sure others) have yet to implement the ACMEv2 protocol. For those, you'd want to use the built-in Certificate Order Management feature in BIG-IP. 
 
- lawrencegtNimbostratus Hi, looking at implementing this at some point soon. Happy to see I can use EAB key hmac etc. One thing, on the kojot-acme page, i don't see any examples of how to include a SAN on your examples. All of our certs typically have a long organistation fqdn, then a shorter one for people to use. Think "SuperLongExampleComany.com" vs "slec.com" so it's a deal breaker to use for us to have. 
 Thanks for you work.- Kevin_StewartEmployee You make a good point. The config anchors on the domain name of the cert and intentionally copies the SANs from existing certs in the renewal. However, it doesn't have a provision for specifying SAN values for a new cert. The simplest thing would be to create a self-signed cert with all of the needed SANs and then point your config at this. The Kojot script would overwrite this with an issued cert, keeping all of the SAN values. But I can definitely look into adding this feature. - lawrencegtNimbostratus Thanks Kevin, I'll have to have a play by the sounds of it! Thanks for your response, much appreciated. 
 
 
- ArturoEmployee Thank you very much Kevin. I have modified the script to avoid using http-0 challenge as associating one iRule per VS in all LTMs are not viable for a lot of customers. I am using an external DNS (F5 DNS with an iRule) and I successfully implemented the solution with the challenge dns-0 and request the challenge from one VS in the LTM where the FQDN is allocated ;) Although I think that I can even improve the solution sending the challenge from the ACME Client to the VS in the F5 DNS. 
 I will keep you informed.
- Awesome work Kevin_Stewart! 
 One minor suggestion - in 3. Create an HTTP virtual server you might want to mention, that you have to add the iRule to the this virtual.- KarimBenyelloulCirrostratus Thanks for the tip Daniel_Wolf . I at first forget about attaching the iRule, after figuring that out, I wanted to add the same comment you did... should've read the comment section before trying to understand what was wrong in my config 😊 
 
- KarimBenyelloulCirrostratus Hello, is there a way to automate the renewal of the management BIG-IP device certificate in a similar manner? - Kevin_StewartEmployee By virtue of the http-01 validation process, the cert subject has to be resolvable by public DNS, and that resolved IP has to have an HTTP:80 listener on the public Internet that the ACMEv2 server can get to. And since a BIG-IP mgmt. UI should never be exposed to the Internet, it really isn’t practical to renew device certs this way. That said, there are at least two options worth mentioning: - Manage the cert in the traffic certs section, do dns-01 validation (doesn’t require an Internet HTTP:80 listener), and then use a separate script on the BIG-IP to move that updated cert from traffic certs section to device cert. Kojot would behave the same, you’d just need this additional script to move the new cert to device.
- Host an internal ACMEv2 service and dns-01 – this solves for the public Internet access issue with http-01. You’d still need the script to move from traffic certs to device.
 
- ArturoEmployee As Kevin rightly pointed out, using the ACME protocol—particularly the http-01 challenge—for renewing the BIG-IP management certificate is not practical. While the dns-01 challenge offers a workaround (by validating ownership via DNS rather than HTTP), it still doesn’t directly solve the problem. Lot of devices does not have the hostname provided on the DNS server. That said, there is a more suitable solution available: A better and more secure alternative is to use a script-based approach that leverages the EST (Enrollment over Secure Transport) protocol. This method allows you to: - Authenticate securely to your internal certificate authority.
- Request and retrieve new certificates over HTTPS.
- Automatically install the renewed certificate on the BIG-IP management interface.
 This avoids public exposure and can be fully automated in environments where an internal PKI with EST support is available (e.g., Microsoft ADCS with EST proxy, EJBCA, etc.). If you’re managing BIG-IP devices at scale or aiming for a hands-off certificate lifecycle, this approach is recommended. - IchnafiCirrostratus Hello Arturo, Are there any examples or additional articles that show how a script-based solution using EST would look like? 
 
 
- Frank_ReiningaNimbostratus Hi Kevin, Thanks for all the updates. Tested this nice tool on a lab license and worked very well. Now moving to production with HA pair. Here I run into an issue that iFiles are not created and config file is not synced to the standby unit. Is there a way to see what goes wrong here? Regards, Frank - Kevin_StewartEmployee Please submit an issue in the Github repo. Working on a fix.