DEV Community

Cover image for Automation of LAMP (Linux, Apache, MySQL, PHP) stack deployment using Bash Script in Ansible
Adaeze Nnamdi-Udekwe
Adaeze Nnamdi-Udekwe

Posted on • Updated on • Originally published at clouddiadem.hashnode.dev

Automation of LAMP (Linux, Apache, MySQL, PHP) stack deployment using Bash Script in Ansible

Introduction

A Bash Script is a text file with several commands written in the Bash shell language. The Bash shell carries out operations one by one when you run the script. This allows you to automate repetitive tasks, manage files, and configure systems.

Ansible is an open-source automation tool for configuration and
management of IT infrastructures. It allows for the easy
management of numerous servers using YAML language to describe
automated tasks. Ansible operates agentlessly, meaning there is no
need to install additional software on the nodes it manages.


You will appreciate the use of Ansible as a tool if you find out about things it does to make your IT job easy. Here are some advantages:

  • Ease of use and simplicity: YAML is a human-readable language used to write playbooks (configuration files).
  • Strong and adaptable: Ansible is capable of managing challenging jobs and changing with the times.
  • Agentless: Management and security are made simpler by not requiring software to be installed on the controlled nodes.
  • Idempotent: Playbooks can be run repeatedly without producing unpleasant or unexpected results.
  • Huge support and community: Ansible has a large and active community, as well as a wealth of modules and resources.

Context

Just so we understand what we are going to be doing in this project, I will explain all that it entails.

  • We are going to automate the provisioning of two Ubuntu-based servers, named “master” and “slave”, using Vagrant.
  • On the Master node, we will create a bash script to automate the deployment of a LAMP (Linux, Apache, MySQL, PHP) stack.
  • The bash script we create above should clone a PHP application from GitHub, install all necessary packages, and configure Apache web server and MySQL.
  • We will ensure the bash script is reusable and readable.
  • Using Ansible Playbook, we will execute the bash script on the Slave node and verify that the PHP application is accessible through the Virtual Machine’s IP address.
  • We will also create a cron job to check the server’s uptime every 12 am.

Steps

Setting Up Virtual Machines

  • To create a Virtual Machine, Check these resources for Ubuntu.

  • You should have a Vagrantfile for the Virtual Machines saved in your local file folder just like mine below after you start your machine. But if you don't I can walk you through it. Remember you need to create 2 Ubuntu servers named "master" and "slave". Open your Virtual box down as you will need it open to start your Virtual Machines. Create two folders and name them "master" and "slave".

folders

  • Open the folders with Gitbash. Let us start with the "master" folder.

gitbash

  • Next is to Initialize your master Ubuntu machine. After opening the "master" folder with git bash, run:
vagrant init Ubuntu/focal64
Enter fullscreen mode Exit fullscreen mode

vagrant init master

  • Open the Vagrantfile and make some edits to the configuration file using the command nano vagrantfile. It would help if you had something like this.

vagrantfile

  • Go ahead to paste the configuration below just under Vagrant.configure("2") do |config| like the image below. Also, add config.ssh.insert_key = false under config.vm.box = ubuntu/focal64. Press "crtl+O" to save, then "enter" and "ctrl+x" to exit the vagrantfile.
# Provision master Node
config.vm.define "master" do |master|
master.vm.box = "ubuntu/focal64"
master.vm.hostname = "master"
master.vm.network "private_network", ip: "192.168.33.10", type: "dhcp"
end

Enter fullscreen mode Exit fullscreen mode

NOTE: I have circled the aspects you should the various commands under. Pay attention to them.

vagrantfile

Then run the following command to start the machine. It would be best if you had something like the image below towards the end after running the command.

vagrant up
Enter fullscreen mode Exit fullscreen mode

vagrant up master

  • Connect the virtual machine by running:
vagrant ssh
Enter fullscreen mode Exit fullscreen mode

vagrant ssh

  • Now that you have logged into your "masters" virtual machine, follow the same procedure for the "slave" virtual machine. You should have something like the screenshot below by the time you ssh into your "slave" virtual machine.

slave machine

NOTE: Every other thing is the same for the procedure except for the configuration in the vagrant file which should be under Vagrant.configure("2") do |config|. Use the command below under the Vagrant.configure("2") do |config|.

# Provision Slave Node
config.vm.define "slave" do |slave|
slave.vm.box = "ubuntu/focal64"
slave.vm.hostname = "slave"
slave.vm.network "private_network", ip: "192.168.33.11", type: "dhcp"
end
Enter fullscreen mode Exit fullscreen mode

Connect the "master" and "slave" machine by _ssh-ing _into each of them

  • Use the command ip a to check for the IP address of each of the virtual machines. Below is that of the "master's" virtual machine. The circled one is the IP address for the "master" virtual machine. Do the same for the "slave" virtual machine. I have underlined each one for you to know what virtual machine owns each IP address. Also, note that your IP address will be different from mine.

IP address for master

IP address for slave

  • On your "master" Virtual Machine, write ssh-keygen to generate a private & public key. When you run the command you will see Enter file in which to save the key (/home/vagrant/.ssh/id_rsa): Enter passphrase (empty for no passphrase): Enter same passphrase again: Keep pressing the "enter" key when those 3 lines appear as seen below.

master key

  • We will repeat the above step for the "slave" virtual machine too and should look like what is below.

slave key

  • In the "master" virtual machine, change the directory into ssh with the command cd .ssh and then ls to list all the files in the ssh directory. You will see something like what is below.

cd .ssh

  • Open the content of id_rsa.pub and copy it to save it somewhere for later use. That is where your public key is and that is what you are copying. Use this command to open it cat id_rsa.pub. The key would be long and look like something below.

public key

  • Change the directory to .ssh with the command cd .ssh just as we did in the "master" virtual machine. Create and open a file in the .ssh directory of the "slave" machine at once so that the public key you copied from the "masters" machine and saved somewhere, you will paste it in the created file. I named my file 'public'. Run the command nano public and paste the public key from the "master" virtual machine here. After pasting all the public keys correctly, press 'crtl+O' to save, press 'enter', and then press 'ctrl+X' to exit.

  • Then run the command cat public >> authorized_keys to copy the public keys into the authorized_keys file from the file named public in the "slave" machine. Run the command cat authorized_keys to see the content of the copied public keys of the "masters" virtual machine and the "slave" virtual machine. Below is an image of both the "masters" and "slave" public keys.

public keys

  • Connect the "master" virtual machine to the "slave" virtual machine by using ssh to connect the IP address of the "slave" machine to the "master" machine. First of all, run the command ip a to know the IP addresses in each machine. Then run the command ssh vagrant@slave ip in the "master" virtual machine. It should look like this ssh vagrant@192.168.33.9 Make sure to replace the IP address with your own "slave" IP address. Whatever it prompts you, type 'yes'. From the screenshot, you can see that we can connect to our "slave" machine from the "master" machine.

master sshing into slave


Creating the Bash Script and Ansible to automate the provisioning of the two Ubuntu-based Virtual Machines we've created.

  • In your "master" virtual machine, create your script that will automate the deployment of LAMP (Linux, Apache, MySQL, PHP) stack.

  • Name your script anything of your choice with the .sh extension. I will be naming mine LAMP.sh. Use the command touch LAMP.sh to create your script. run ls to see if the script file you ran is showing.

create script

  • Run nano LAMP.sh to open the file script and write your script. Please copy and paste this script inside it. Each line of the command has a comment with a brief explanation with a '#' symbol in front of it. I made sure to pick each line of commands one by one to run before compiling it into a full script below.
#!/bin/bash

# Define variables for paths and IP address
LARAVEL_DIR="/var/www/html/laravel"
LARAVEL_CONF="/etc/apache2/sites-available/laravel.conf"
VIRTUAL_HOST="192.168.33.9"

# Update all package index and upgrade packages
sudo apt update
sudo apt upgrade -y
echo "Done with upgrade package"

# Update the PHP repository
sudo add-apt-repository -y ppa:ondrej/php
echo "Done with php repo update"

# Install Apache
sudo apt install -y apache2
sudo systemctl enable apache2
echo "Done with Apache installation"

# Install MySQL
sudo debconf-set-selections <<< 'mysql-server mysql-server/root_password password'
sudo debconf-set-selections <<< 'mysql-server mysql-server/root_password_again password'
sudo apt install -y mysql-server
echo "Done with MySQL installation"

# Install PHP 8.2 and 8.3 with necessary extensions
sudo apt install -y php libapache2-mod-php php-mysql php8.2 php8.2-curl php8.2-dom php8.2-xml php8.2-mysql php8.2-sqlite3 php8.3 php8.3-curl php8.3-dom php8.3-xml php8.3-mysql php8.3-sqlite3
echo "Done with PHP 8.2 and 8.3 installation"

# Run MySQL secure installation
expect <<EOF
spawn sudo mysql_secure_installation
expect "Would you like to setup VALIDATE PASSWORD component?"
send "y\r"
expect {
    "Please enter 0 = LOW, 1 = MEDIUM and 2 = STRONG" {
        send "1\r"
        exp_continue
    }
    "Remove anonymous users?" {
        send "y\r"
        exp_continue
    }
    "Disallow root login remotely?" {
        send "n\r"
        exp_continue
    }
    "Remove test database and access to it?" {
        send "y\r"
        exp_continue
    }
    "Reload privilege tables now?" {
        send "y\r"
        exp_continue
    }
}
EOF
echo "Done with MySQL secure installation"

# Restart Apache
sudo systemctl restart apache2
echo "Done with Apache restart"

# Install Git
sudo apt install -y git
echo "Done with Git installation"

# Clone Laravel repository
sudo git clone https://github.com/laravel/laravel $LARAVEL_DIR
echo "Done with cloning Laravel repository"

# Change directory to Laravel folder
cd $LARAVEL_DIR
echo "Changed directory to $LARAVEL_DIR"

# Install Composer
sudo apt install -y composer
echo "Done with installing Composer"

# Upgrade Composer to version 2
sudo php -r "copy('https://getcomposer.org/installer', 'composer-setup.php');"
sudo php -r "if (hash_file('sha384', 'composer-setup.php') === 'dac665fdc30fdd8ec78b38b9800061b4150413ff2e3b6f88543c636f7cd84f6db9189d43a81e5503cda447da73c7e5b6') { echo 'Installer verified'; } else { echo 'Installer corrupt'; unlink('composer-setup.php'); } echo PHP_EOL;"
sudo php composer-setup.php --install-dir /usr/bin --filename composer
echo "Done with upgrading Composer to version 2"

# Use Composer to install dependencies
yes | sudo composer install
echo "Done installing Composer dependencies"

# Copy Laravel configuration file and set permissions
sudo cp .env.example .env
sudo chown www-data:www-data .env
sudo chmod 640 .env
echo "Done copying Laravel configuration file and setting permissions"

# Create virtual host in /etc/apache2/sites-available
sudo tee $LARAVEL_CONF >/dev/null <<EOF
<VirtualHost *:80>
    ServerName $VIRTUAL_HOST
    ServerAlias *
    DocumentRoot $LARAVEL_DIR/public

    <Directory $LARAVEL_DIR>
        AllowOverride All
    </Directory>
</VirtualHost>
EOF
echo "Done creating virtual host in /etc/apache2"

# Generate application key
sudo php artisan key:generate
echo "Done generating application key"

# Run migrations
sudo php artisan migrate --force
echo "Done running migrations"

# Change ownership permissions
sudo chown -R www-data:www-data $LARAVEL_DIR/database/ $LARAVEL_DIR/storage/logs/ $LARAVEL_DIR/storage $LARAVEL_DIR/bootstrap/cache
echo "Done changing ownership permissions"

# Set file permissions
sudo chmod -R 775 $LARAVEL_DIR/database/ $LARAVEL_DIR/storage/logs/ $LARAVEL_DIR/storage
echo "Done setting file permissions"

# Disable default configuration file
sudo a2dissite 000-default.conf
echo "Done disabling default configuration file"

# Enable Laravel configuration file
sudo a2ensite laravel.conf
echo "Done enabling Laravel configuration file"

# Restart Apache
sudo systemctl restart apache2
echo "Done restarting Apache"


uptime > /var/log/uptime.log

Enter fullscreen mode Exit fullscreen mode
  • Press 'crtl+O' to save, press 'enter' and then 'crtl+X' to exit the script.

The variables created in the first three lines of the script allow for its reusability, as mentioned in our context.

Leave the repository as it is in the script, as it is the same one linked earlier in this article.

Here is a snippet of my script

script

Below are detailed explanations of what each line of command does:

i. Defining variables for paths and IP address:

-LARAVEL_DIR="/var/www/html/laravel": This variable specifies the directory where the Laravel application will be located.

-LARAVEL_CONF="/etc/apache2/sites-available/laravel.conf": This variable specifies the path for the Apache virtual host configuration file for Laravel.

-VIRTUAL_HOST="192.168.33.9": This variable specifies the IP address for the virtual host.
ii. Updating package index and upgrading packages:

-sudo apt update: Updates the list of available packages and their versions.
-sudo apt upgrade -y: Upgrades all installed packages to the latest versions. The -y option automatically answers "yes" to prompts.
iii. Updating the PHP repository:

-sudo add-apt-repository -y ppa:ondrej/php: Adds the PHP PPA (Personal Package Archive) maintained by Ondřej Surý, which provides the latest PHP versions.

iv. Installing Apache:

  • sudo apt install -y apache2: Installs the Apache web server.
  • sudo systemctl enable apache2: Enables Apache to start automatically on system boot.

v. Installing MySQL:

  • sudo debconf-set-selections <<< 'mysql-server mysql-server/root_password password': Configures the root password for MySQL server installation.
  • sudo apt install -y mysql-server: Installs the MySQL server.

vi. Installing PHP 8.2 and 8.3 with necessary extensions:

sudo apt install -y php libapache2-mod-php php-mysql php8.2 ...: Installs PHP 8.2 and 8.3 with necessary extensions such as curl, dom, xml, mysql, and sqlite3.

vii. Running MySQL secure installation:

  • The expect block automates the mysql_secure_installation process, answering prompts automatically to set up MySQL securely.

viii. Restarting Apache:

  • sudo systemctl restart apache2: Restarts the Apache web server.

ix. Installing Git:

  • sudo apt install -y git: Installs Git, a version control system.

x. Cloning Laravel repository:

xi. Changing directory to Laravel folder:

  • cd $LARAVEL_DIR: Changes the current working directory to the Laravel folder.

xii. Installing Composer:

  • sudo apt install -y composer: Installs Composer, a dependency management tool for PHP.

xiii. Upgrading Composer to version 2:

  • These commands download and verify the Composer installer, then install Composer version 2.

xiv. Installing Composer dependencies:

  • yes | sudo composer install: Installs Composer dependencies for the Laravel project, automatically answering "yes" to prompts.

xv. Copying Laravel configuration file and setting permissions:

  • Copies the Laravel .env.example configuration file to .env and sets appropriate ownership and permissions.

xvi. Creating virtual host in /etc/apache2/sites-available:

  • sudo tee $LARAVEL_CONF >/dev/null <<EOF: Creates a new virtual host configuration file for Laravel.
  • This block of code sets up the virtual host with the specified server name, alias, and document root.

xvii. Generating application key:

  • sudo php artisan key:generate: Generates a unique application key for the Laravel project.

xviii. Running migrations:

  • sudo php artisan migrate --force: Runs database migrations for the Laravel project.

xix. Changing ownership permissions:

  • sudo chown -R www-data:www-data $LARAVEL_DIR/...: Changes ownership of specified directories to the www-data user and group.

xx. Setting file permissions:

  • sudo chmod -R 775 $LARAVEL_DIR/...: Sets file permissions to 775 (read, write, execute for owner and group; read and execute for others).

xxi. Disabling default configuration file:

  • sudo a2dissite 000-default.conf: Disables the default Apache configuration file.

xxii. Enabling Laravel configuration file:

  • sudo a2ensite laravel.conf: Enables the Laravel virtual host configuration file.

xxiii. Restarting Apache:

  • sudo systemctl restart apache2: Restarts the Apache web server to apply changes.
    xxiv. Logging system uptime:

  • uptime > /var/log/uptime.log: Records the current system uptime to a log file.

  • Test run the script to see if it works by running the command ./LAMP.sh


  • Still in your "master" machine create an Ansible directory with the command
mkdir Ansible
Enter fullscreen mode Exit fullscreen mode
  • Change into the Ansible directory so it becomes your present working directory.
cd Ansible
Enter fullscreen mode Exit fullscreen mode
  • Run this command to update any dependencies. From the image below, you should have something like the image below at the end after the update.
sudo apt-get update
Enter fullscreen mode Exit fullscreen mode

apt-get update

  • Then to install Ansible run the command below. It would be best if you had something like the image at the end of the installation.
sudo apt install ansible
Enter fullscreen mode Exit fullscreen mode

install ansibe


  • Still, in your Ansible directory, create your inventory. The inventory file is used to store the address of the slave nodes to be configured which will enable the playbook to run in the "slave" machine even if it is created in the "master's" machine.
touch inventory 
Enter fullscreen mode Exit fullscreen mode
  • Open your inventory file to put in the IP address of your "slave" machine. It should look like the image below, but use your IP address. Press 'crtl+O' to save, press 'enter', and then 'crtl+X' to exit the text editor.

IP address


  • Create your playbook with the .yml extension.
touch playbook.yml
Enter fullscreen mode Exit fullscreen mode
  • To ping your inventory run the command in your "master" machine. It should look like the screenshot below. Press 'crtl+C' to cancel it afterwards.
ansible all -m ping -i inventory
Enter fullscreen mode Exit fullscreen mode

pinging

  • Change the directory back to Ansible with the command cd Ansible. Then open the playbook with the command nano playbook.yml. Paste the command below inside your playbook but change the IP address to your "slave" machine IP. Press 'crtl+O' to save, press 'enter', and then 'crtl+X' to exit the playbook.
---

- hosts: slave
  become: yes
  tasks:

    # Copy LAMP.sh script to the slave node
    - name: Copy LAMP.sh script to slave node
      copy:
        src: /home/vagrant/Ansible/LAMP.sh
        dest: /home/vagrant/LAMP.sh
        mode: 0755  

      # Make the script executable

    # Execute the LAMP.sh script on the slave node
    - name: Execute LAMP.sh script on slave node
      shell: /home/vagrant/LAMP.sh


- name: Create a cron job to check server's uptime every 12 am
  hosts: slave
  become: yes
  tasks:

    # Create an uptime script that logs the server's uptime to a file
    - name: Create the uptime script
      copy:
        content: |

          #!/bin/bash

          # Log the server's uptime to a file
          uptime > /var/log/uptime.log
        dest: /usr/local/bin/check_uptime.sh
        mode: '0755'  # Make the script executable

    # Create a cron job that runs the uptime script every day at 12 am (midnight)

    - name: Create a cron job to check uptime every 12 am
      cron:
        name: Check uptime every 12 am
        job: "/usr/local/bin/check_uptime.sh"

        # Run at 12:00 am
        minute: '0'  
        hour: '0'
        state: present

    # Display the contents of the uptime log file
    - name: Display uptime log file
      shell: cat /var/log/uptime.log
      register: uptime_output

    # Display the server's uptime as captured in the uptime log file
    - name: Display the server's uptime
      debug:
        msg: "{{ uptime_output.stdout }}"

    # Task to fetch the content of the PHP application using curl
    - name: Fetch PHP application content using curl
      command: "curl -s http://192.168.33.9"  # Perform a silent HTTP GET request to the target IP address
      register: php_application  # Register the output of the curl command
      ignore_errors: true  # Continue playbook execution even if this task fails

    # Task to display the content fetched from the PHP application
    - name: Display content of PHP application
      debug:
        msg: "PHP Application Content:\n{{ php_application.stdout }}"  # Display the output of the curl command

Enter fullscreen mode Exit fullscreen mode

Here is a snippet of my playbook

playbook

  • Created a cron job to check the server’s uptime every 12 am in my playbook. You can find the task in the playbook above but this is a snippet of it.

cron job

This is the content of my log file where the server's uptime check is stored. Since my master and my slave are connected, I was able to check this from my "slave" machine. I used the command sudo crontab -l to check.

uptime check

  • Run the command below to execute the playbook:
ansible-playbook playbook.yml -i inventory
Enter fullscreen mode Exit fullscreen mode
  • I could execute the bash script on the Slave node and verify that the PHP application is accessible through the VM’s IP address. I have circled the slave IP address in the screenshot.

execution 1
Execution 2

  • Paste your "slave" IP address on your web browser. This is the output with the slave IP address. A Laravel PHP application.

laravel page


Conclusion

In summary, one of the effective and dependable methods for provisioning and controlling servers has been demonstrated in this project by the automation of the LAMP (Linux, Apache, MySQL, PHP) stack deployment utilizing a Bash script in Ansible. The deployment of the LAMP stack on Ubuntu-based servers is quite fast and less labor-intensive than with manual setup because of the script automation and it is even reusable.

The Bash script can be easily adapted to many environments and use cases due to its reusability, which is attained through defining variables that can be passed as parameters and error handling.

The deployment process was further automated and made uniform and repeatable by integrating the script into an Ansible playbook. In addition to establishing cron jobs to track server uptime and confirming that PHP applications could be accessed, the playbook carried out actions on both the master and slave nodes, giving important information about server performance.

Even though there were a few difficulties along the road, like script errors and log file problems, these could be fixed with thorough debugging and code modifications. The result showed that the PHP application was successfully deployed and verified on the slave node's IP address.

All things considered, the project demonstrates how Ansible and Bash scripting can be used to automate intricate deployment procedures. This detailed documentation of the stages and solutions makes it useful for future deployments and projects utilizing the LAMP stack.

Top comments (4)

Collapse
 
collinshenry profile image
collins-henry

Hi Adaeze,

I just wanted to drop you a quick note to express my sincere gratitude for your incredibly helpful article. Your clear explanation and step-by-step guide made it so much easier for me to automate and deploy my server.

Your expertise and generosity in sharing your knowledge are truly commendable. Thanks to your article. It's contributions like yours that make the online community such a valuable resource for people like me.

Once again, thank you for taking the time to write and share such an informative piece. I greatly appreciate it!

Collapse
 
clouddiadem profile image
Adaeze Nnamdi-Udekwe

I am very delighted to read that the article was helpful. Thank you for being kind enough to drop a comment to express how much value it offered you. Thank you again!

Collapse
 
deyemijoshua profile image
Joshua

What an interesting article to work on Adaeze. I appreciate your response and guide when I encountered a problem while working on the project. Now I know more about using ansible as an automation tool and you taught me the ability to be able to apply ip address to my vagrant vms.

Collapse
 
clouddiadem profile image
Adaeze Nnamdi-Udekwe

Hi Joshua,

Thank you for your wholesome feedback. It was my pleasure to help you troubleshoot when you got stuck. I appreciate you having to try out this project, using my article as a guide.

I hope you stick around.