Skip to content

brett-buskirk/guide-to-deployment

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

28 Commits
 
 
 
 
 
 
 
 

Repository files navigation

Hosting and Deploying a Website

This is a comprehensive guide to hosting and deploying a front-end React website on DigitalOcean. In addition, it covers the process of registering a domain and obtaining a SSL certificate through Namecheap in order to create a professional and secure website.


Initial Notes

This project assumes some basic knowledge of or comfortablility using a terminal. The major steps are all included below, but it might behoove you to research anything you don't know. I use a Chromebook's Linux container for such development, but anything containing a Bash terminal should work just fine.

Second, this tutorial follows the steps to create and use a domain I registered for this purpose. When following along, you'll want to purchase your own domain and substitute it whenever you see indianatrektribe.online, indianatrektribe_online, or similar conventions.


Registering a Domain Name

The first step is to make sure you have a domain name secured through a registrar. There are several choices out there to choose from, but in this case, we're going to use Namecheap. This process is fairly straight-forward, and once the transaction is complete, you'll be able to view your domain. Note that when you purchase the domain, you're also going to buy an SSL certificate, as shown in the next section.

Domain Registration


Obtaining an SSL Certificate

Once you have a domain secured, you'll want to attach an SSL Certificate to it. When you purchase the domain, you will also be given an option to add an SSL certificate to your cart.

An SSL (Secure Sockets Layer) certificate is a digital certificate that authenticates a website's identity and enables an encrypted connection between a web server and a web browser. The certificate is a bit of code on a web server that provides security for online communications. When a web browser contacts a secured website, the SSL certificate enables an encrypted connection. This link ensures that all data passed between the web server and browser remains private.

A website needs an SSL certificate in order to keep user data secure, verify ownership of the website, prevent attackers from creating a fake version of the site, and gain user trust. Without SSL, site visitors and customers are at higher risk of having their data stolen or injection of malware via a man-in-the-middle (MiTM) attack.

Once you have purchased the SSL certificate, you should be able to see it on the registrar's dashboard.

SSL Certificate

We'll deal with this some more later. For now, just make sure you have an SSL certificate secured and ready for use.


Create a DigitalOcean Account

For this project, we'll be using DigitalOcean as our hosting platform. It is here that you will create your remote virtual machine (VM) in the cloud. You will then provision this machine as a web server to host your website. Setting up an account with DigitalOcean is relatively straightforward. Simply go to the link above and create your account.


Installing doctl

DigitalOcean has a handy command line interface (CLI) that you will be using for much of your work. This CLI is called doctl, and you'll want to follow this link to learn about, install, and configure the CLI for our work throughout this process. Don't forget to create an API token and store it in a safe place, such as your password manager if you have one. Once you have doctl installed and configured, you can check this by running the following command in your terminal:

doctl account get

Generate and add SSH Keys to DigitalOcean

In order to access your server remotely, you'll need to configure an SSH key pair and add the public key to DigitalOcean. You can generate a key pair by running the following commmand:

ssh-keygen -t rsa -f ~/.ssh/do-key-rsa

For this purpose, hit enter when prompted for a password in order to create a passwordless ssh key pair. This will create two files in a directory named .ssh. These files are named do-key-rsa (the private key) and do-key-rsa.pub (the public key).

To add the public key to DigitalOcean, run the following command:

doctl compute ssh-key import do-key-rsa --public-key-file ~/.ssh/do-key-rsa.pub

If successful, you should see some output confirming the key was added, along with the key's FingerPrint. You can also retrieve a list of your keys on DigitalOcean by running:

doctl compute ssh-key list

Output:

ID          Name          FingerPrint
39125477    do-key-rsa    55:5c:ce:7d:b3:6d:89:7b:37:48:bd:25:99:86:ae:eb

You'll need to remember the ID of your key, because you'll need it for the next step.


Create a Droplet on DigitalOcean

DigitalOcean calls their VMs (Virtual Machines) droplets. To create a droplet, run the following command:

doctl compute droplet create --image debian-12-x64 --size s-1vcpu-2gb --region nyc1 --enable-monitoring --ssh-keys 39125477 --tag-name webserver indianatrektribe.online

Output:

ID           Name                       Public IPv4    Private IPv4    Public IPv6    Memory    VCPUs    Disk    Region    Image            VPC UUID    Status    Tags         Features                    Volumes
370152202    indianatrektribe.online                                                  2048      1        50      nyc1      Debian 12 x64                new       webserver    monitoring,droplet_agent    

As you can see, you'll get some output showing that the droplet was successfully created. You can also verify this by going to your DigitalOcean dashboard, where you will see the newly-created droplet displayed.

Digital Ocean Droplet

Okay, that's a pretty gnarly command, so let's break it down:

  • doctl compute droplet create: the basic command infrastructure
  • --image debian-12-x64: The image created (Debian 12 AMD64)
  • --size s-1vcpu-2g: The size (1 virtual CPU and 2 GB of memory)
  • --region nyc1: The region where the cloud server is located
  • --enable-monitoring: Enables monitoring through DigitalOcean for analytics
  • --ssh-keys 39125477: The ID of your SSH key to be added to the VM
  • --tag-name: A tag to identify this droplet as a webserver
  • indianatrektribe.online: The name of the droplet

Once the droplet has been created, you can connect to it remotely by running:

doctl compute ssh indianatrektribe.online --ssh-key-path ~/.ssh/do-key-rsa

You'll want to type yes when asked if you want to continue. Afterward, you should be connected to your VM.

Linux indianatrektribe 6.1.0-9-amd64 #1 SMP PREEMPT_DYNAMIC Debian 6.1.27-1 (2023-05-08) x86_64

The programs included with the Debian GNU/Linux system are free software;
the exact distribution terms for each program are described in the
individual files in /usr/share/doc/*/copyright.

Debian GNU/Linux comes with ABSOLUTELY NO WARRANTY, to the extent
permitted by applicable law.
root@indianatrektribe:~# 

Once you have confirmed this, you can exit the remote host by typing exit at the terminal.


Adding a Firewall to your Droplet

Now that you have a droplet spun up, it's time to secure it! You only want to allow certian kinds of traffic in and out of your server. This is accomplished by creating a firewall and attaching it to your droplet.

doctl compute firewall create --name webserver-firewall \
 --droplet-ids 370152202 \
 --inbound-rules "protocol:tcp,ports:22,address:0.0.0.0/0,address:::/0 protocol:tcp,ports:80,address:0.0.0.0/0,address:::/0 protocol:tcp,ports:443,address:0.0.0.0/0,address:::/0" \
 --outbound-rules "protocol:icmp,address:0.0.0.0/0,address:::/0 protocol:tcp,ports:all,address:0.0.0.0/0,address:::/0 protocol:udp,ports:all,address:0.0.0.0/0,address:::/0"

Output:

ID                                      Name                  Status     Created At              Inbound Rules                                                                                                                                                      Outbound Rules                                                                                                                                          Droplet IDs    Tags    Pending Changes
23fd8e14-4bdf-42b5-a0c4-c35ce5272a42    webserver-firewall    waiting    2023-08-16T02:27:19Z    protocol:tcp,ports:22,address:0.0.0.0/0,address:::/0 protocol:tcp,ports:80,address:0.0.0.0/0,address:::/0 protocol:tcp,ports:443,address:0.0.0.0/0,address:::/0    protocol:icmp,address:0.0.0.0/0,address:::/0 protocol:tcp,ports:0,address:0.0.0.0/0,address:::/0 protocol:udp,ports:0,address:0.0.0.0/0,address:::/0    370152202              droplet_id:370152202,removing:false,status:waiting

If you can't remember your droplet's ID, you can run:

doctl compute droplet list

You can view your newly-created firewall by running:

doctl compute firewall list

Now, let's go over some of these firewall rules. The settings above allow for the following traffic:

  • All inbound SSH (port 22)
  • All inbound HTTP (port 80)
  • All inbound HTTPS (port 443)
  • All outbound ICMP
  • All outbound TCP
  • All outbound UDP

These rules can be further refined by adding specific port ranges, but such is beyond the scope of this tutorial.


Add Sudo User to the Droplet

It's not a good idea to remote into your VM as a root user. Root access is powerful and should only be used with extreme caution. In place of this, you'll want to create a new user and give it sudo (priveleged) access. First log into your droplet:

doctl compute ssh indianatrektribe.online --ssh-key-path ~/.ssh/do-key-rsa

Create the new user, giving it whatever name you want, by running the following command:

sudo adduser brett

You'll need to supply a password (make sure you remember it!) and answer a few questions (which I typically leave blank), then you'll be prompted if the information is correct.

Next, give your new user sudo priveleges:

usermod -aG sudo brett

Verify these changes by using the following command to switch to your new user:

su brett

Your prompt should change as follows:

root@indianatrektribe:~# su brett
brett@indianatrektribe:/root$ 

Now you can use this user to run root commands by prepending the word sudo to them. Run the following:

sudo apt update && sudo apt upgrade -y

You'll have to enter your password the first time you use sudo in a session. The command above will update your server and apply any upgrades that might be available.

Now you'll need to transfer the ssh keys from the root user's profile to your new user's profile. To do this you need to run the following commands.

First you need to create the .ssh directory on the user's root directory:

mkdir /home/$USER/.ssh

Make the directory only executable by the user:

chmod 700 /home/$USER/.ssh

Copy the authorized_keys file that contains your public key:

sudo cp /root/.ssh/authorized_keys /home/$USER/.ssh/authorized_keys

Make everything in .ssh owned by the user:

sudo chown -R $USER:$USER /home/$USER/.ssh

Make it readable only by the user:

sudo chmod 600 /home/$USER/.ssh/authorized_keys

Now exit the current sessions by typing and entering the following command twice:

exit

Now when you remote into the remote host, use:

doctl compute ssh indianatrektribe.online --ssh-key-path ~/.ssh/do-key-rsa --ssh-user brett

You should now be logged in as your new user.


Install nginx on the Droplet

Before you can actually host a website on your droplet, you must first turn it into a webserver. You can do this with nginx. In order to install nginx, first connect to the remote host with the user you created above. Next, run the following command:

sudo apt install nginx

Now check to make sure the nginx service is running:

systemctl status nginx

You should see some output like the following:

● nginx.service - A high performance web server and a reverse proxy server
     Loaded: loaded (/lib/systemd/system/nginx.service; enabled; preset: enabled)
     Active: active (running) since Thu 2023-08-17 04:22:50 UTC; 45s ago
       Docs: man:nginx(8)
    Process: 24046 ExecStartPre=/usr/sbin/nginx -t -q -g daemon on; master_process on; (code=exited, status=0/SUCCESS)
    Process: 24047 ExecStart=/usr/sbin/nginx -g daemon on; master_process on; (code=exited, status=0/SUCCESS)
   Main PID: 24070 (nginx)
      Tasks: 2 (limit: 2315)
     Memory: 1.6M
        CPU: 26ms
     CGroup: /system.slice/nginx.service
             ├─24070 "nginx: master process /usr/sbin/nginx -g daemon on; master_process on;"
             └─24071 "nginx: worker process"

This informs you that the service is up and running.

If everything is configured correctly, you can type your droplet's IP address in a browser and see the following screen:

Nginx landing page

Note: If you don't know what your droplet's IPv4 address is, use the following command:

doctl compute droplet get indianatrektribe.online --format PublicIPv4

Setting up the Nginx Server Blocks (Part 1)

Next you need to set up the Nginx server blocks. When using the Nginx web server, server blocks can be used to encapsulate configuration details. You'll start this process by first creating the necessary directory structure:

sudo mkdir -p /var/www/indianatrektribe.online/html

Next, make sure the current user is the owner of the directory:

sudo chown -R $USER:$USER /var/www/indianatrektribe.online/html

Now set the appropriate permissions for the directory:

sudo chmod -R 755 /var/www/indianatrektribe.online

Next you'll need to pull down a set of website files to serve, which we'll cover in the next section.


Pull Down a Website to the Remote Host from GitHub

Now you're going to pull down a repository from GitHub that contains a basic front-end website. Before you can do this, however, you'll need to install git on the remote host. You can do this with the following command:

sudo apt install git

Unless you have a repo of your own in mind, you can pull down the basic-website project from my GitHub page by running the following command:

git clone https://github.com/brett-buskirk/basic-website.git

Now you need to move all the basic website files to the location from which they will be served:

sudo mv -v ~/basic-website/* /var/www/indianatrektribe.online/html

Setting up the Nginx Server Blocks (Part 2)

Next you'll set up a server block with the correct directives that point to your custom web root:

sudo nano /etc/nginx/sites-available/indianatrektribe.online

This will open the nano editor, where you'll want to copy the following content:

server {
    listen 80;
    listen [::]:80;

    root /var/www/indianatrektribe.online/html;
    index index.html;

    server_name indianatrektribe.online www.indianatrektribe.online;

    location / {
        try_files $uri $uri/ =404;
    }
#    return 301 https://$server_name$request_uri;
}

Next, enable this server block by creating a symbolic link to your custom configuration file inside the sites-enabled directory, which Nginx reads during startup:

sudo ln -s /etc/nginx/sites-available/indianatrektribe.online /etc/nginx/sites-enabled/

You'll also want to delete the default enabled site:

sudo rm /etc/nginx/sites-enabled/default

To avoid a possible hash bucket memory problem that can arise from adding additional server names to your configuration, it is necessary to adjust a single value in the /etc/nginx/nginx.conf file. Open the file:

sudo nano /etc/nginx/nginx.conf

Find the server_names_hash_bucket_size directive and remove the # symbol to uncomment the line:

...
http {
    ...
    server_names_hash_bucket_size 64;
    ...
}
...

Next, test to make sure that there are no syntax errors in any of your Nginx files:

sudo nginx -t

If everything is okay, you'll see the following output:

nginx: the configuration file /etc/nginx/nginx.conf syntax is ok
nginx: configuration file /etc/nginx/nginx.conf test is successful

Once your configuration test passes, restart Nginx to enable your changes:

sudo systemctl restart nginx

Link Domain to DigitalOcean

The next step is to link your domain from Namecheap to DigitalOcean. To do this, first login to your Namecheap account and select Domain List from the left main menu, then click the Manage button next to your domain on the right:

Manage Domain

Now select the Custom DNS option from the Nameservers dropdown menu, and enter the following nameservers:

  • ns1.digitalocean.com
  • ns2.digitalocean.com
  • ns3.digitalocean.com

Once you're done, click the green checkmark to save your changes:

Set Custom DNS

Note: This requires the nameservers to propogate, so it could take up to 24-48 hours for the DNS changes to take effect (though sometimes it's much quicker).

Once this is done, head on over to your DigitalOcean dashboard and open a terminal on your local machine so we can add the domain and update the DNS records.

To add the domain to DigitalOcean, run the following command:

doctl compute domain create indianatrektribe.online

You should see the following output:

Domain                     TTL
indianatrektribe.online    1800

Next you'll add a couple of new A records, which maps an IPv4 address to a domain name. This determines where to direct any requests for a domain name. To map an apex domain (@), run the following command:

doctl compute domain records create indianatrektribe.online --record-type A --record-name @ --record-data 192.241.131.184

Output:

ID            Type    Name    Data               Priority    Port    TTL     Weight
1695060856    A       @       192.241.131.184    0           0       1800    0

The next A record will add an subdomain prefix (e.g. www). To create this record, run the following command:

doctl compute domain records create indianatrektribe.online --record-type A --record-name www --record-data 192.241.131.184

Output:

ID            Type    Name    Data               Priority    Port    TTL     Weight
1695061173    A       www     192.241.131.184    0           0       1800    0

Once this is done, you can visit your DigitalOcean dashboard to see the changes. You'll need to expand the Manage section on the main menu to the left, then select Networking from the menu. From there, click on the Domains tab and click on the name of your domain. You should be able to scroll down and see the list of DNS Records there:

DNS Records

Now, once the DNS has fully propogated from Namecheap to DigitalOcean, you'll be able to enter your domain name into the address bar of a browser and see your website:

Live HTTP Website

Notice the Not secure warning in the address bar to the left of your domain name. You'll fix this in the next section when you add the SSL certificate to the droplet.


Activate and set up SSL Certificate

In order to secure your droplet, you'll need to activate and install an SSL Certificate. Before you begin with the SSL activation, you'll first need to have a CSR code generated on your server.

CSR code (Certificate Signing Request) is a specific code and an essential part for the SSL activation. It contains information about website name and the company contact details. For many reasons, the code should be created on the hosting server end. On some servers, it is the obligatory condition.

In order to generate a CSR, you must remote into your droplet and run the following commands.

Switch to the root user:

sudo su

Create an ssl directory and move into it:

mkdir /root/ssl && cd /root/ssl

Use RSA key algorithm to generate CSR code:

openssl req -new -newkey rsa:2048 -nodes -keyout indianatrektribe.online -out indianatrektribe.online

After some weird output, you'll be asked some questions:

You are about to be asked to enter information that will be incorporated
into your certificate request.
What you are about to enter is what is called a Distinguished Name or a DN.
There are quite a few fields but you can leave some blank
For some fields there will be a default value,
If you enter '.', the field will be left blank.
-----
Country Name (2 letter code) [AU]:US
State or Province Name (full name) [Some-State]:Indiana
Locality Name (eg, city) []:Indianapolis
Organization Name (eg, company) [Internet Widgits Pty Ltd]:NA
Organizational Unit Name (eg, section) []:NA
Common Name (e.g. server FQDN or YOUR name) []:indianatrektribe.online
Email Address []:myemail@gmail.com

Please enter the following 'extra' attributes
to be sent with your certificate request
A challenge password []:
An optional company name []:

If you run an ls command, you can see there's a file named after your domain. Inside this file is the CSR code

cat indianatrektribe.online

Output:

-----BEGIN PRIVATE KEY-----
[ a bunch of random letters, numbers, and symbols ]
-----END PRIVATE KEY-----
-----BEGIN CERTIFICATE REQUEST-----
[ a bunch of random letters, numbers, and symbols ]
-----END CERTIFICATE REQUEST-----

You'll need to copy everything between the BEGIN and END CERTIFICATE REQUEST lines, including those two lines themselves. Login to your Namecheap account and click on SSL Certificates on the main menu to the left. Then click on the Activate button for you SSL certificate:

Activate SSL Certificate

On the page that pops up, paste what you copied above into the Enter CSR box:

CSR Code Paste

If your code is correct, you'll see the name of your domain pop up in the Primary Domain box. After this, click the Next button to go to the next step. Here you'll select the Add CNAME record for the DCV option option. Then click the Next button again:

DCV Option

On the next page, verify your email address and click Next:

PositiveSSL Email Verification

Confirm the details are correct, then click the Submit button:

PositiveSSL Confirmation

You'll be redirected back to your Namecheap dashboard with information about your SSL certificate. As of now, it's pending validation.

PositiveSSL Finished

Now click on the link that says Get a CNAME record from this page (Edit methods). You'll be redirected to a page where you can get the DCV records you'll need to complete the activation process. On this page scroll down to where you see the EDIT METHODS button. Click the arrow next to it and select the Get Record button that pops down.

Get DCV Record

You'll see a pop-up that shows two values you'll need to copy for the next step. These two values are Host and Target. Once you have these values copied somewhere, go ahead and click the Done button.

Now that you have these two values, you'll use the doctl command to create a CNAME record for your droplet. From your local machine run the following command:

doctl compute domain records create indianatrektribe.online --record-type CNAME --record-name [Host (but without .indianatrektribe.online)] --record-data [Target (with a . added at the end)] --record-ttl 30

Notice the two values used in this command:

  • [Host (but without .indianatrektribe.online)]: This is the Host value you copied a minute ago. Hoewever, do not include the .indianatrektribe.online part.
  • [Target (with a . added at the end)]: This is the full Target value you copied a minute ago, but add a . at the end.

You can go to https://mxtoolbox.com/CnameLookup.aspx to see if the CNAME record is published or not. In the Domain Name box, paste the full Host value from above. You should see that the DNS record is published.

Once your DNS record is published, you'll eventually get an email from Sectigo with a .zip file attachment that you need to download. This file needs to be copied over to your droplet. Navigate a terminal to the directory where the .zip file is located and enter the following command:

scp -i ~/.ssh/do-key-rsa indianatrektribe_online.zip root@indianatrektribe.online:~/ssl

Now remote into your droplet as the root user again and navigate to the ssl directory, where you need to extract the files:

doctl compute ssh indianatrektribe.online --ssh-key-path ~/.ssh/do-key-rsa
apt install unzip && unzip indianatrektribe_online.zip

Next you'll need to combine the files:

cat indianatrektribe_online.crt > indianatrektribe_online_chain.crt ; echo >> indianatrektribe_online_chain.crt ; cat indianatrektribe_online.ca-bundle >> indianatrektribe_online.crt

Now you'll need to add a server block to the nginx.conf file, inside the http block. Open the file with:

nano /etc/nginx/nginx.conf

Add the following block at the bottom of the http block, before the closing bracket:

	server {

		listen 443 ssl;

		ssl_certificate /root/ssl/indianatrektribe_online_chain.crt;

		ssl_certificate_key /root/ssl/indianatrektribe.online;

		root /var/www/indianatrektribe.online/html;

		server_name www.indianatrektribe.online;

		keepalive_timeout 70;

	}

Lastly, you'll need to uncomment a line in /etc/nginx/sites-available/indianatrektribe.online:

nano /etc/nginx/sites-available/indianatrektribe.online

Uncomment (remove the # symbol) the line, return 301 https://$server_name$request_uri;

Now restart the Nginx service:

systemctl restart nginx

If everything goes smoothly, you should now be able to point your browser to your domain and see the lock symbol next to the address bar. Your site is now secure and protected by HTTPS. Your work here is now done!

Live HTTP Site

About

Comprehensive guide to front-end website deployment

Topics

Resources

Stars

Watchers

Forks