How to Install a Custom SSL Certificate in UniFi Video

The default installation of Ubiquiti’s UniFi Video software uses a self signed certificate for the web interface.  As a result users will notice a certificate warning in their browsers when they connect to the page.

In UniFi Video version 3.7.0 Ubiquiti added experimental support for importing custom certificates.  This allows you to import a certificate signed by a trusted root certificate authority and make the browser certificate errors disappear.

Unfortunately the process still isn’t very easy.  Below is the process I used to install a signed certificate in UniFi Video.

Note that I’m running the UniFi Video software on Centos which isn’t officially supported by Ubiquiti.  As a result some of my paths might be slightly different from other distributions but the general process is the same.

Generating the Certificate Signing Request (CSR)

In order to make Chrome happy the certificate must contain the Subject Alternative Name field (SAN).  Chrome now ignores the CN field and will display certificate errors if the SAN field is not present.

It turns out this field is actually useful since you can list multiple different DNS names in the same certificate.  This allows you to list both an internal and external DNS name for the same server within the same certificate.

In order for the SAN fields to be present they must be specified in the CSR.  I performed these steps on a linux system with openSSL installed.  It can be done from Windows but the commands may vary and I won’t get into that in this post.

Create a new file called san.cnf.  You will need to edit 7 of the lines and replace them with your own information.  Enter your two digit country code, state name, locality, org name, DNS.1, and DNS.2.  If you only want to specify DNS.1 then that is fine, you can also use more than 2.

[ req ]
default_bits = 2048
distinguished_name = req_distinguished_name
req_extensions = req_ext
prompt = no
[ req_distinguished_name ]
countryName = YourCountryCode
stateOrProvinceName = YourState
localityName = YourCity
organizationName = YourOrg
commonName = YourFQDN
[ req_ext ]
subjectAltName = @alt_names
[alt_names]
DNS.1 = YourFQDN
DNS.2 = YourDN

Next generate the certificate request that references the san.cnf file created in the previous step.  Run this command from the same directory as the san.cnf file (or specify the full path).
openssl req -out request.csr -newkey rsa:2048 -nodes -keyout private.key -config san.cnf

When the process completes you will have two files, request.csr which contains the certificate request and private.key that contains the associated private key.

To verify that the SANS extension is in your CSR run the command below.  You should see the DNS names specified from the alt_names section of the san.cnf file.

openssl req -noout -text -in request.csr

Use the command below to convert the private key into a format that UniFi Video will accept.  If you are seeing errors in your server.log file complaining about “Failed to import private key” or “algid parse error, not a sequence” then your private key is probably in the incorrect format.  The server will only accept the key in DER format.

openssl pkcs8 -topk8 -inform PEM -outform DER -nocrypt -in private.key -out private.key.der

Getting the certificate signed

At this point you should have a properly formatted private key (private.key.der) and certificate request (request.csr).  The next step is to have the certificate request signed by a trusted root certificate authority.  In my case the trusted CA is a Microsoft Enterprise PKI server running on my network.  In your case it could be a public CA such as Namecheap or one of the many others.

For Microsoft PKI servers you can initiate a certificate request by accessing the certificate services web interface (http://yourca/certsrv/).  Click on advanced request and paste the full contents of the request.csr file in the top box.

Be sure to select web server from the certificate template drop down box.

If all goes well you will see the screen shown below which presents the option to download the certificate.  Unfi Video requires the certificate be DER encoded so select that option and download the certificate.  In some cases you may need the entire chain but in my I did not since all of my clients already trust this root CA.

Installing the certificate and private key on the server

Create the certificates folder.  Your path may be /usr/lib/UniFi-Video/ or something else depending on where your base installation lives.  On my Centos installation it’s in /opt/UniFi-Video.

mkdir -p /opt/UniFi-Video/data/certificates

Copy the signed certificate and private keyinto the certificates folder and rename them:

  • ufv-server.cert.der (signed certificate)
  • ufv-server.key.der (private key file)

Change the permissions on the newly created directory and the files inside.

chown -R unifi-video:unifi-video /opt/UniFi-Video/data/certificates/

At this point the contents of the certificates directly should look like this:

Stop the unifi-video service.  The command may very depending on your operating system.

systemctl stop unifi-video.service

Modify the system.properties file to enable custom certificates.

vi /opt/UniFi-Video/data/system.properties

Add the line below to the file and save the changes.

ufv.custom.certs.enable=true

Make a backup the existing trust store in case something goes wrong.

cp /opt/UniFi-Video/data/ufv-truststore /opt/UniFi-Video/data/ufv-truststore.backup

Next delete the existing trust store.

rm -rf /opt/UniFi-Video/data/ufv-truststore

Start the unifi-video service.

systemctl start unifi-video.service

If everything works correctly the private key and certificate inside the certificates directory will disappear and a new trust store file will be created that contains them.  The new certificate should be active in the web server at this point. From within chrome you can press Ctrl + shift +I to inspect the certificate.

If it’s not working check the log file for any errors (/opt/UniFi-Video/logs/error.log).  Even if your custom certificate is not successfully imported the UniFi software will automatically regenerate a new trust store so you’ll have to repeat part of the process in order to test a new certificate if you encounter problems.

Sam Kear

Sam graduated from the University of Missouri - Kansas City with a bachelors degree in Information Technology. Currently he works as a network analyst for an algorithmic trading firm. Sam enjoys the challenge of troubleshooting complex problems and is constantly experimenting with new technologies.

6 thoughts to “How to Install a Custom SSL Certificate in UniFi Video”

  1. I’m running the UniFi Video NVR on Debian 9 and was able to use my own certificates now! 🙂
    Thanks for sharing your knowledge!

  2. Great walk-through! Thanks!

    Just some side notes:

    1. Running on Debian 9.5 (4.9.0-7-amd64 #1 SMP Debian 4.9.110-3+deb9u2 (2018-08-13) x86_64 GNU/Linux) and UniFi 3.9.9, was exactly the same process.

    2. The location was however (as you already pointed out) different: /var/lib/unifi-video

    3. Created the certificates folder in /var/lib/unifi-video/certificates

    4. When downloading from a Cert Authority, they provided the signed cert in PEM encoding. Confirmed by running ‘type’ command against the file:
    type ca-signedcert-file.crt

    5. So I had to convert the signed CA certificate to DER before copying over to ‘/var/lib/unifi-video/certificates’ and renaming :
    openssl x509 -inform PEM -outform der -in ca-signedcert-file.crt -out ca-signedcert-file.der

    NB: Don’t forget the rename part like I did to begin with! Doh!
    ufv-server.cert.der (signed certificate)
    ufv-server.key.der (private key file)

    Basically, read the steps that Sam has kindly documented and you shouldn’t go wrong!

    As for the request for the Windows SSL walk through – There is a You Tube video about this, but I gave up on Windows updates since MS kept trashing my machines with poor quality Windows updates.

    🙂

Leave a Reply

Your email address will not be published. Required fields are marked *