How to install NGINX, MariaDB and multiple PHP versions in macOS

A brief introduction about the application softwares discussed in this article

  • NGINX – A high-performance web server, load balancer, reverse proxy etc. with a very low memory footprint.
  • PHP – Server-side software which handles the data processing. We use the PHP-FPM (FastCGI Process Manager) implementation.
  • MariaDB – A community developed fork of the famous MySQL relational database. It will act as our backend datastore.

At the time of writing this article, the available versions are listed below

  • macOS version 10.12.6 (Sierra)
  • NGINX version 1.15.5
  • PHP version 5.6 and 7.2
  • MariaDB version 10.3.10

We are considering that the machine does not have any software installed on the default HTTP port 80. Super user access (sudo) and brew package manager is required to complete the installation.

You can learn to setup brew package manager from the article How to install a package manager for macOS. Let us assume that your terminal shell is BASH.

NGINX

NGINX installation is the easiest. Open the terminal software and execute the following command

$ brew install nginx

Once the installation is complete, the next job would be to configure auto-start for NGINX. As we would be running the application in the default HTTP port 80, we would require root privileges (Privileged ports < 1024 can be opened only by root). Execute the following command in the terminal

$ sudo cp -v /usr/local/opt/nginx/homebrew.mxcl.nginx.plist /Library/LaunchDaemons/
$ sudo chown root:wheel /Library/LaunchDaemons/homebrew.mxcl.nginx.plist

The next step would be to change the default port of NGINX from 8080 to 80. Edit the NGINX configuration using your favourite editor. I will be using VIM. If you are a novice, I suggest that you create a backup of the configuration file. This will help rule out chances of file corruption.

$ cp /usr/local/etc/nginx/nginx.conf /tmp/nginx.conf.bak 
$ vim /usr/local/etc/nginx/nginx.conf

Search the following line inside the server directive (curly braces)

listen     8080;

Change it to

listen     80;

Start the NGINX server

$ sudo launchctl load /Library/LaunchDaemons/homebrew.mxcl.nginx.plist

You can verify the working of NGINX by executing the following command (assuming that curl is installed)

$ curl -IL http://127.0.0.1:80
HTTP/1.1 200 OK
Server: nginx/1.15.5
Date: Wed, 24 Oct 2018 14:01:29 GMT
Content-Type: text/html
Content-Length: 612
Last-Modified: Tue, 02 Oct 2018 15:13:59 GMT
Connection: keep-alive
ETag: "5bb38b37-264"
Accept-Ranges: bytes

This can also be verified by using the browser, enter http://localhost on the address bar, you will be greeted with the following image

nginx index.html

PHP

At the time of writing this article, LiteBreeze, a 14-year-old bespoke web application development company maintains multiple legacy applications. Hence PHP 5.6 is a requirement for development. However, new projects are all developed in Laravel using the latest PHP 7.2.

Let’s install PHP 5.6. Execute the following command in the terminal

$ brew install --without-apache --with-fpm --with-mysql --with-curl --with-xml \
--with-mbstring --with-bcmath --with-zip --with-gd --with-soap php56

To install PHP 7.2, execute

$ brew install php72 

Once installation of both the PHP versions are completed, we will run them in PORTS rather than FILE SOCKETS. Fire up your favourite editor and open the following files.I have used VIM

$ vim /usr/local/etc/php/7.2/php-fpm.d/www.conf

Search and replace the following line

listen = 127.0.0.1:9000

to

listen = 127.0.0.1:9072

I have set the port to 9072 – so that the last two digits indicate the PHP version running on it. Similarly for PHP 5.6 version, we will set the port to 9056 (56 denoting 5.6 version)

$ vim /usr/local/etc/php/5.6/php-fpm.conf

Search and replace the following line

listen = 127.0.0.1:9000 

to

listen = 127.0.0.1:9056

Configure the PHP CLI binary

$ echo 'export PATH="/usr/local/sbin:$PATH"' >> ~/.bash_profile && . ~/.bash_profile

Next step is to configure auto-start on system boot. Create the following directory

$ mkdir -p ~/Library/LaunchAgents

Create soft links for configuration files

$ ln -sfv /usr/local/opt/php\@5.6/homebrew.mxcl.php\@5.6.plist ~/Library/LaunchAgents/
$ ln -sfv /usr/local/opt/php\@7.2/homebrew.mxcl.php.plist ~/Library/LaunchAgents/

Start PHP-FPM services

$ launchctl load -w ~/Library/LaunchAgents/homebrew.mxcl.php\@5.6.plist
$ launchctl load -w ~/Library/LaunchAgents/homebrew.mxcl.php.plist

Before proceeding further, let’s ensure that PHP-FPM is running. We can verify by checking for open listeners on ports 9056 & 9072:

$ lsof -Pni4 | grep LISTEN | grep php

The output should look something like this(ports 9056 & 9072 are being listed):

php-fpm    8955 litebreeze    7u IPv4 0xf78171cd083fc4eb      0t0 TCP 127.0.0.1:9056 (LISTEN)
php-fpm    8956 litebreeze    0u IPv4 0xf78171cd083fc4eb      0t0 TCP 127.0.0.1:9056 (LISTEN)
php-fpm    8957 litebreeze    0u IPv4 0xf78171cd083fc4eb      0t0 TCP 127.0.0.1:9056 (LISTEN)
php-fpm   17269 litebreeze    8u IPv4 0xf78171cd1256dbf3      0t0 TCP 127.0.0.1:9072 (LISTEN)
php-fpm   17270 litebreeze    9u IPv4 0xf78171cd1256dbf3      0t0 TCP 127.0.0.1:9072 (LISTEN)
php-fpm   17271 litebreeze    9u IPv4 0xf78171cd1256dbf3      0t0 TCP 127.0.0.1:9072 (LISTEN)

Let us configure two virtual host files to mock projects of PHP versions 5.6 and 7.2 respectively

$ cd /usr/local/var/www/
$ mkdir php56
$ cd php56

Create an index.php file

$ vim index.php

Insert the following content

<?php
phpinfo();

Create another directory php72 for PHP 7.2 version project and copy the index.php file

$ cd ..
$ mkdir php72
$ cp php56/index.php php72

Next step is to create two virtual hosts for the two PHP projects

$ cd /usr/local/etc/nginx/servers

Use your favourite editor, create a file php56.conf for PHP 5.6 version project

$ vim php56.conf

Insert the following contents

server {
        listen 80;
        listen [::]:80;
        root /usr/local/var/www/php56;

        # Add index.php to the list if you are using PHP
        index index.php index.html index.htm index.nginx-debian.html;

        server_name php56.test;

        location / {
                # First attempt to serve request as file, then
                # as directory, then fall back to displaying a 404.
                try_files $uri $uri/ /index.php?$query_string;
        }

        # pass the PHP scripts to FastCGI server listening on 
        #
        location ~ \.php$ {
                fastcgi_pass 127.0.0.1:9056;
                fastcgi_split_path_info ^(.+\.php)(/.+)$;
                fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
                include fastcgi_params;
        }
}

Create a file php72.conf for PHP 7.2 version project

$ vim php72.conf

Add the following contents, notice the difference in root directory (/usr/local/var/www/php72), server name (php72.test) and PHP port (9072)

server {
        listen 80;
        listen [::]:80;
        root /usr/local/var/www/php72;

        # Add index.php to the list if you are using PHP
        index index.php index.html index.htm index.nginx-debian.html;

        server_name php72.test;

        location / {
                # First attempt to serve request as file, then
                # as directory, then fall back to displaying a 404.
                try_files $uri $uri/ /index.php?$query_string;
        }

        # pass the PHP scripts to FastCGI server listening on 
        #
        location ~ \.php$ {
                fastcgi_pass 127.0.0.1:9072;
                fastcgi_split_path_info ^(.+\.php)(/.+)$;
                fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
                include fastcgi_params;
        }
}

Edit the /etc/hosts file to add the virtual hosts

$ sudo vim /etc/hosts

Change the line

127.0.0.1       localhost

to

127.0.0.1       localhost php56.test php72.test

The end result would be

##
# Host Database
#
# localhost is used to configure the loopback interface
# when the system is booting.  Do not change this entry.
##
127.0.0.1       localhost php56.test php72.test
255.255.255.255 broadcasthost
::1             localhost

Restart NGINX

$ sudo launchctl unload /Library/LaunchDaemons/homebrew.mxcl.nginx.plist
$ sudo launchctl load /Library/LaunchDaemons/homebrew.mxcl.nginx.plist

Fire up your browser, enter the following in the address bar

http://php56.test

php 5.6 info

http://php72.test

php 7.2 info

MariaDB

To install MariaDB, execute

$ brew install mariadb

After the installation is complete, configure the start/stop service, so the MySQL server gets automatically stopped and started when the Mac is shutdown/powered-up respectively:

$ ln -sfv /usr/local/opt/mariadb\@10.3/homebrew.mxcl.mariadb.plist ~/Library/LaunchAgents

To launch the service before reboot, execute

$ launchctl load ~/Library/LaunchAgents/homebrew.mxcl.mariadb.plist

Verify the working using the following command.

$ sudo lsof -PiTCP -sTCP:LISTEN | grep mysqld

The output would be similar to this

mysqld  9574 litebreeze   22u IPv6 0xf78171cd085b6d7b      0t0 TCP *:3306 (LISTEN)

To secure the MariaDB installation, follow the instructions after executing the command

$ mysql_secure_installation

Test the connection

$ mysql -u root -p
Enter password:
Welcome to the MariaDB monitor.  Commands end with ; or \g.
Your MariaDB connection id is 14
Server version: 10.3.10-MariaDB Homebrew
Copyright (c) 2000, 2018, Oracle, MariaDB Corporation Ab and others.
Type 'help;' or '\h' for help. Type '\c' to clear the current input statement.

MariaDB [(none)]> exit
Bye

Shortcuts

Starting and stopping services in OSX is a tedious process. Switching between PHP CLI binary is another uphill task. Hence, to make the development easier, add the following content to your ~/.bash_profile file. Please note that the following section assumes that PHP versions 7.2 and 5.6 are the only installed versions.

alias php-fpm72.start="launchctl load -w ~/Library/LaunchAgents/homebrew.mxcl.php.plist"
alias php-fpm72.stop="launchctl unload -w ~/Library/LaunchAgents/homebrew.mxcl.php.plist"
alias php-fpm72.restart='php-fpm.stop && php-fpm.start'
alias php-fpm56.start="launchctl load -w ~/Library/LaunchAgents/homebrew.mxcl.php@5.6.plist"
alias php-fpm56.stop="launchctl unload -w ~/Library/LaunchAgents/homebrew.mxcl.php@5.6.plist"
alias php-fpm56.restart='php-fpm56.stop && php-fpm56.start'
alias nginx.error_logs='tail -100f /usr/local/var/log/nginx/error.log'
alias nginx.access_logs='tail -100f /usr/local/var/log/nginx/access.log'
alias nginx.start='sudo launchctl load /Library/LaunchDaemons/homebrew.mxcl.nginx.plist'
alias nginx.stop='sudo launchctl unload /Library/LaunchDaemons/homebrew.mxcl.nginx.plist'
alias nginx.restart='nginx.stop && nginx.start'
alias mariadb.stop='sudo /usr/local/Cellar/mariadb/10.3.10/bin/mysql.server stop'
alias mariadb.start='sudo /usr/local/Cellar/mariadb/10.3.10/bin/mysql.server start'
alias mariadb.restart='sudo /usr/local/Cellar/mariadb/10.3.10/bin/mysql.server restart'

alias php56='brew unlink php72 && brew link php56 --force'
alias php72='brew unlink php56 && brew link php72 --force'

Reinvoke the SHELL using the updated file

$ source .bash_profile
$ php -v
PHP 7.2.11 (cli) (built: Oct 11 2018 16:23:06) ( NTS )
Copyright (c) 1997-2018 The PHP Group
Zend Engine v3.2.0, Copyright (c) 1998-2018 Zend Technologies
    with Zend OPcache v7.2.11, Copyright (c) 1999-2018, by Zend Technologies

As you can figure out from the output, PHP 7.2 is the current CLI version. Let’s change it to 5.6

$ php56
Unlinking /usr/local/Cellar/php/7.2.11... 24 symlinks removed
Linking /usr/local/Cellar/php@5.6/5.6.38... 25 symlinks created
$ php -v
PHP 5.6.38 (cli) (built: Sep 14 2018 22:32:34)
Copyright (c) 1997-2016 The PHP Group
Zend Engine v2.6.0, Copyright (c) 1998-2016 Zend Technologies
    with Zend OPcache v7.0.6-dev, Copyright (c) 1999-2016, by Zend Technologies

Let us switch again back to 7.2

$ php72
Unlinking /usr/local/Cellar/php@5.6/5.6.38... 25 symlinks removed
Linking /usr/local/Cellar/php/7.2.11... 24 symlinks created
$ php -v
PHP 7.2.11 (cli) (built: Oct 11 2018 16:23:06) ( NTS )
Copyright (c) 1997-2018 The PHP Group
Zend Engine v3.2.0, Copyright (c) 1998-2018 Zend Technologies
    with Zend OPcache v7.2.11, Copyright (c) 1999-2018, by Zend Technologies

From the above examples, switching PHP CLI versions is a cinch with custom commands php56 and php72. I have listed the explanations for the following commands which were added to ~/.bash_profile file

nginx.restart - Restart the NGINX server
nginx.error_logs - Display the NGINX error log in the terminal
nginx.access_logs - Display the NGINX access log in the terminal
php-fpm72.restart - Restart the PHP 7.2 version FPM
php-fpm56.restart - Restart the PHP 5.6 version FPM
mariadb.restart - Restart the MariaDB server

As you can see, using the above approach enables us to set up our Mac to simultaneously run different projects, each using a different version of PHP. At LiteBreeze, we always strive to improve and share our technical knowledge.

We hope this article help improve your web application development experience with macOS, especially when dealing with legacy applications.