There is a new version of this tutorial available for Ubuntu 22.04 (Jammy Jellyfish).

How to Install Nextcloud with Nginx on Ubuntu 18.04 LTS

Nextcloud is a free (Open Source) Dropbox-like software, a fork of the ownCloud project. Nextcloud is written in PHP and JavaScript, it supports many database systems such as MySQL/MariaDB, PostgreSQL, Oracle Database, and SQLite. In order to keep your files synchronized between Desktop and your own server, Nextcloud provides applications for Windows, Linux, and Mac desktops and a mobile app for Android and iOS. Nextcloud is not just a Dropbox clone, it provides additional features like Calendar, Contacts, Schedule tasks, and streaming media with Ampache etc.

In this tutorial, we will show you how to install and configure the latest Nextcloud release (at the time of writing this, the latest release is 13.0.2) on an Ubuntu 18.04 server. We will run Nextcloud with a Nginx web server and PHP7.1-FPM and use MySQL server as the database system.

Prerequisites

  • Ubuntu 18.04
  • Root privileges

What we will do

  1. Install Nginx Web server
  2. Install and Configure PHP7.1-FPM
  3. Install and Configure MySQL Server
  4. Generate SSL Letsencrypt
  5. Download Nextcloud 13
  6. Configure Nginx Virtual Host for Nextcloud
  7. UFW Firewall Configuration
  8. Nextcloud Post-Installation

Step 1 - Install Nginx Web server

The first step we will do in this nextcloud guide is to install the Nginx web server. We will be using the Nginx web server instead of Apache web server.

Log in to the server and update the repository, then install Nginx web server using the apt command as shown below.

sudo apt update
sudo apt install nginx -y

After the installation is complete, start the Nginx service and enable the service to launch every time at system boot using systemctl.

systemctl start nginx
systemctl enable nginx

The Nginx web server has been installed on Ubuntu 18.04. Check it using the netstat or curl command below.

netstat -plntu
curl -I localhost

The Nginx web server is now running under Ubuntu on standard HTTP port 80.

Install Nginx Web server

Step 2 - Install and Configure PHP7.1-FPM

In this tutorial, we will be using the PHP7.1-FPM for nextcloud. We will be using PHP7.1-FPM from the PPA repository, so we need to add new PPA repository to our system.

Install the 'software-properties-common' package and add the 'ondrej PHP' PPA repository by running the following commands.

sudo apt install software-properties-common -y
sudo add-apt-repository ppa:ondrej/php -y

Note:

On Ubuntu 18.04, the 'add-apt-repository' command will automatically update the repository.

Now install PHP7.1 and PHP7.1-FPM with all extensions needed using the single apt command below.

sudo apt install php7.1-fpm php7.1-mcrypt php7.1-curl php7.1-cli php7.1-mysql php7.1-gd php7.1-iconv php7.1-xsl php7.1-json php7.1-intl php-pear php-imagick php7.1-dev php7.1-common php7.1-mbstring php7.1-zip php7.1-soap -y

After the installation is complete, we will configure the php.ini files for php-fpm and php-cli.

Go to the '/etc/php/7.1' directory.

cd /etc/php/7.1/

Edit the php.ini files for php-fpm and php-cli using vim.

vim fpm/php.ini
vim cli/php.ini

Uncomment the 'date.timezone' line and change the value with your own timezone.

date.timezone = Asia/Jakarta

Uncomment the 'cgi.fix_pathinfo' line and change the value to '0'.

cgi.fix_pathinfo=0

Save and exit.

Next, edit the php-fpm pool configuration 'www.conf'.

vim fpm/pool.d/www.conf

Uncomment those lines below.

env[HOSTNAME] = $HOSTNAME
env[PATH] = /usr/local/bin:/usr/bin:/bin
env[TMP] = /tmp
env[TMPDIR] = /tmp
env[TEMP] = /tmp

Save and exit.

Restart the PHP7.1-FPM service and enable it to launch every time on system boot.

systemctl restart php7.1-fpm
systemctl enable php7.1-fpm

Now check it using the netstat command.

netstat -pl | grep php

And you will get the php-fpm is now running under the sock file '/run/php/php7.1-fpm.sock'.

Install PHP

Step 3 - Install and Configure MySQL Server

In this step, we will install the latest MySQL version and create a new database for the nextcloud installation. The latest version MySQL packages are available on the repository by default.

Install MySQL server latest version using the apt command below.

sudo apt install mysql-server mysql-client -y

After the installation is complete, start the MySQL service and enable it to launch everytime at system boot.

systemctl start mysql
systemctl enable mysql

Now we will configure the MySQL root password using the 'mysql_secure_installation' command.

Run the following command.

mysql_secure_installation

At this MySQL 5.8 version, there is a security improvement for the MySQL password policy. You need to choose the password policy - 0 for the LOW policy, 1 for the MEDIUM policy, and 2 for a STRONG password policy.

For this guide, we will be using the 'MEDIUM' password policy, and it's recommended to use the 'STRONG' password policy on the production server.

Choose number '1' and press Enter, then type your new MySQL 'root' password.

Install MySQL

Remove anonymous users? (Press y|Y for Yes, any other key for No) : Y
Disallow root login remotely? (Press y|Y for Yes, any other key for No) : Y
Remove test database and access to it? (Press y|Y for Yes, any other key for No) : Y
Reload privilege tables now? (Press y|Y for Yes, any other key for No) : Y

And the MySQL root password has been set up.

Next, we will create a new database for nextcloud installation. We will create a new database named 'nextcloud_db' with the user 'nextclouduser' and password '[email protected]'.

Login to the MySQL shell as a root user with mysql command.

mysql -u root -p
TYPE THE MYSQL ROOT PASSWORD

Now create the database and user with the password by running following MySQL queries.

create database nextcloud_db;
create user [email protected] identified by '[email protected]';
grant all privileges on nextcloud_db.* to [email protected] identified by '[email protected]';
flush privileges;

And the new database and user for the nextcloud installation has been created.

Configure MySQL database

The MySQL installation and configuration for nextcloud has been completed.

Step 4 - Generate SSL Letsencrypt

In this tutorial, we will secure nextcloud using free SSL from Letsencrypt, and we will generate certificates files using the letsencrypt tool.

If you do not have a domain name or install nextcloud on the local computer, you can generate the Self-Signed certificate using OpenSSL.

Install the 'letsencrypt' tool using the apt command below.

sudo apt install letsencrypt -y

After the installation is complete, stop the nginx service.

systemctl stop nginx

Next, we will generate the SSL certificates for our domain name 'nextcloud.hakase-labs.io' using the cerbot command line. Run the command below.

certbot certonly --standalone -d nextcloud.hakase-labs.io

You will be asked for the email address, and it's used for the renew notification. For the Letsencrypt TOS agreement, type 'A' to agree and for the share email address, you can type 'N' for No.

Generate Let's encrypt SSL certificate

When it's complete, you will get the result as shown below.

SSL cert created

The SSL certificates Letsencrypt for the netxcloud domain name has been generated, all located at the '/etc/letsencrypt/live/domain' directory.

Step 5 - Download Nextcloud

Before downloading the nextcloud source code, make sure the unzip package is installed on the system. If you don't have the package, install it using the apt command below.

sudo apt install wget unzip zip -y

Now go to the '/var/www' directory and download the nextcloud-13.0.2.zip file.

cd /var/www/
wget https://download.nextcloud.com/server/releases/latest.zip

Extract the zip file and you will get the 'nextcloud' directory, then create a new 'data' directory.

unzip latest.zip
mkdir -p nextcloud/data/

Now change the owner of 'nextcloud' directory to the 'www-data' user and group.

chown -R www-data:www-data /var/www/nextcloud/

Nextcloud has been downloaded under the '/var/www/nextcloud' directory, and it will be the web root directory.

Step 6 - Configure Nginx Virtual Host for Nextcloud

In this step, we will configure the nginx virtual host for nextcloud. We will configure nextcloud to run under the HTTPS connection and will force the HTTP connection automatically to the secure HTTPS connection.

Now go to the '/etc/nginx/sites-available' directory and create a new virtual host file 'nextcloud'.

cd /etc/nginx/sites-available/
vim nextcloud

There, paste the following nextcloud virtual host configuration.

upstream php-handler {
    #server 127.0.0.1:9000;
    server unix:/run/php/php7.1-fpm.sock;
}

server {
    listen 80;
    listen [::]:80;
    server_name nextcloud.hakase-labs.pw;
    # enforce https
    return 301 https://$server_name$request_uri;
}

server {
    listen 443 ssl http2;
    listen [::]:443 ssl http2;
    server_name nextcloud.hakase-labs.pw;

    ssl_certificate /etc/letsencrypt/live/nextcloud.hakase-labs.pw/fullchain.pem;
    ssl_certificate_key /etc/letsencrypt/live/nextcloud.hakase-labs.pw/privkey.pem;

    # Add headers to serve security related headers
    # Before enabling Strict-Transport-Security headers please read into this
    # topic first.
    # add_header Strict-Transport-Security "max-age=15552000;
    # includeSubDomains; preload;";
    #
    # WARNING: Only add the preload option once you read about
    # the consequences in https://hstspreload.org/. This option
    # will add the domain to a hardcoded list that is shipped
    # in all major browsers and getting removed from this list
    # could take several months.
    add_header X-Content-Type-Options nosniff;
    add_header X-XSS-Protection "1; mode=block";
    add_header X-Robots-Tag none;
    add_header X-Download-Options noopen;
    add_header X-Permitted-Cross-Domain-Policies none;

    # Path to the root of your installation
    root /var/www/nextcloud/;

    location = /robots.txt {
        allow all;
        log_not_found off;
        access_log off;
    }

    # The following 2 rules are only needed for the user_webfinger app.
    # Uncomment it if you're planning to use this app.
    #rewrite ^/.well-known/host-meta /public.php?service=host-meta last;
    #rewrite ^/.well-known/host-meta.json /public.php?service=host-meta-json
    # last;

    location = /.well-known/carddav {
      return 301 $scheme://$host/remote.php/dav;
    }
    location = /.well-known/caldav {
      return 301 $scheme://$host/remote.php/dav;
    }

    # set max upload size
    client_max_body_size 512M;
    fastcgi_buffers 64 4K;

    # Enable gzip but do not remove ETag headers
    gzip on;
    gzip_vary on;
    gzip_comp_level 4;
    gzip_min_length 256;
    gzip_proxied expired no-cache no-store private no_last_modified no_etag auth;
    gzip_types application/atom+xml application/javascript application/json application/ld+json application/manifest+json application/rss+xml application/vnd.geo+json application/vnd.ms-fontobject application/x-font-ttf application/x-web-app-manifest+json application/xhtml+xml application/xml font/opentype image/bmp image/svg+xml image/x-icon text/cache-manifest text/css text/plain text/vcard text/vnd.rim.location.xloc text/vtt text/x-component text/x-cross-domain-policy;

    # Uncomment if your server is built with the ngx_pagespeed module
    # This module is currently not supported.
    #pagespeed off;

    location / {
        rewrite ^ /index.php$uri;
    }

    location ~ ^/(?:build|tests|config|lib|3rdparty|templates|data)/ {
        deny all;
    }
    location ~ ^/(?:\.|autotest|occ|issue|indie|db_|console) {
        deny all;
    }

    location ~ ^/(?:index|remote|public|cron|core/ajax/update|status|ocs/v[12]|updater/.+|ocs-provider/.+)\.php(?:$|/) {
        fastcgi_split_path_info ^(.+\.php)(/.*)$;
        include fastcgi_params;
        fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
        fastcgi_param PATH_INFO $fastcgi_path_info;
        fastcgi_param HTTPS on;
        #Avoid sending the security headers twice
        fastcgi_param modHeadersAvailable true;
        fastcgi_param front_controller_active true;
        fastcgi_pass php-handler;
        fastcgi_intercept_errors on;
        fastcgi_request_buffering off;
    }

    location ~ ^/(?:updater|ocs-provider)(?:$|/) {
        try_files $uri/ =404;
        index index.php;
    }

    # Adding the cache control header for js and css files
    # Make sure it is BELOW the PHP block
    location ~ \.(?:css|js|woff|svg|gif)$ {
        try_files $uri /index.php$uri$is_args$args;
        add_header Cache-Control "public, max-age=15778463";
        # Add headers to serve security related headers (It is intended to
        # have those duplicated to the ones above)
        # Before enabling Strict-Transport-Security headers please read into
        # this topic first.
        # add_header Strict-Transport-Security "max-age=15768000; includeSubDomains; preload;";
        #
        # WARNING: Only add the preload option once you read about
        # the consequences in https://hstspreload.org/. This option
        # will add the domain to a hardcoded list that is shipped
        # in all major browsers and getting removed from this list
        # could take several months.
        add_header X-Content-Type-Options nosniff;
        add_header X-XSS-Protection "1; mode=block";
        add_header X-Robots-Tag none;
        add_header X-Download-Options noopen;
        add_header X-Permitted-Cross-Domain-Policies none;
        # Optional: Don't log access to assets
        access_log off;
    }

    location ~ \.(?:png|html|ttf|ico|jpg|jpeg)$ {
        try_files $uri /index.php$uri$is_args$args;
        # Optional: Don't log access to other assets
        access_log off;
    }
}

Save and exit.

Enable the virtual host and test the configuration, and make sure there is no error.

ln -s /etc/nginx/sites-available/nextcloud /etc/nginx/sites-enabled/
nginx -t

Now restart PHP7.1-FPM service and nginx service using the systemctl command below.

systemctl restart nginx
systemctl restart php7.1-fpm

The Nginx virtual host configuration for nextcloud has been created.

Configure Nginx for Nextcloud

Step 7 - Configure UFW Firewall

In this tutorial, we will turn on the firewall, and we will be using the UFW firewall for Ubuntu.

Enable the ufw firewall.

ufw enable

Type 'y' and press Enter to start and enable the UFW firewall.

Now add new SSH, HTTP and HTTPS to the UFW firewall list.

ufw allow ssh

ufw allow http
ufw allow https

Check the allowed list port on the UFW firewall using the command below.

ufw status

And you will get the HTTP port 80 and HTTPS port 443 is on the list.

Configure UFW Firewall

Step 8 - Nextcloud Post-Installation

Open your web browser and type the nextcloud URL address.

http://nextcloud.hakase-labs.io/

And you will be redirected to the secure HTTPS connection.

On the Top page, we need to create the admin user for nextcloud, type the admin user password. On the 'Data folder' configuration, type the full path of the 'data' directory '/var/www/nextcloud/data'.

Login to nextCloud

Scroll the page to the bottom, and you will get the database configuration. Type the database info that we've created in step 3 and then click the 'Finish Setup' button.

Database configuration

And after the installation is complete, you will get the Nextcloud Dashboard as below.

Nextcloud dashboard

The Nextcloud 13 installation with Nginx web server and MySQL database on Ubuntu 18.04 has been completed successfully.

Reference

Share this page:

19 Comment(s)

Add comment

Please register in our forum first to comment.

Comments

By: Quentin

Hi,

 

Is there anyway to do it without https ? I'd like to test on a local virtual server first but I'm stuck at the :

certbot certonly --standalone -d nextcloud.hakase-labs.io

 

Thanks

By: mohammad

If you're using local environment - you can generate self signed certificate files using openssl.

look at the link below.

https://www.howtoforge.com/tutorial/how-to-install-nextcloud-with-nginx-and-php-fpm-on-centos-7/#step-generate-a-selfsigned-ssl-certificate-for-nextcloud

By: Marcelo Oliveira

On step 6, after #nginx -t, returns:

 

[email protected]:/etc/nginx/sites-available# nginx -tnginx: [emerg] BIO_new_file("/etc/letsencrypt/live/nextcloud.hakase-labs.pw/fullchain.pem") failed (SSL: error:02001002:system library:fopen:No such file or directory:fopen('/etc/letsencrypt/live/nextcloud.hakase-labs.pw/fullchain.pem','r') error:2006D080:BIO routines:BIO_new_file:no such file)nginx: configuration file /etc/nginx/nginx.conf test failed

I followed step by step and am receiving this. Any advice? Only difference is I've utilized the latest version of nextcloud download of 13.0.5 instead of 13.0.2.zip from the previous steps without any other errors.

 

Thanks

By: Eimantas

It is not mentioned anywhere in tutorial, but you need to replace nextcloud.hakase-labs.pw in the given configuration with your vps hostname of your VPS.

By: Jayden

You should update the download URL in your wget command to: https://download.nextcloud.com/server/releases/latest.zip

By: till

That's a good idea, I've changed that in the tutorial.

By: mzorzy

"nextcloud virtual host configuration" can be done under ISPconfig-site-webdomani-option-directive?

how? tnx

 

By: till

No, this won't work. The tutorial above is not for an ISPConfig server. The config shown above is a complete vhost, so you would add it outside of ISPConfig in the way described above. There should be no conflict unless you try to add a website with the same domain name in ISPConfig. The alternative would be to create a website in ISPConfig, upload nextcloud into that website and then add just the location rules to the nginx directives field. If I remember correctly, there is a thread about that in the forum, probably in the priority support sub-forum.

By: stardust

Getting the following error. Ran all updates aslo rebooted. Any tips?

 

Note, selecting 'php7.1-common' instead of 'php7.1-iconv'Some packages could not be installed. This may mean that you haverequested an impossible situation or if you are using the unstabledistribution that some required packages have not yet been createdor been moved out of Incoming.The following information may help to resolve the situation:The following packages have unmet dependencies: php7.1-mcrypt : Depends: libmcrypt4 but it is not installable php7.1-zip : Depends: libzip4 (>= 1.0) but it is not installableE: Unable to correct problems, you have held broken packages.

 

Thanks

 

By: Maarten

I'm wondering why you are installing version 7.1 of PHP, when version 7.2 of PHP is the default in Ubuntu 18.04 LTS.

By: stardust

I was simply following the instructions. Why? Look at Step 2 - Install and Configure PHP7.1-FPM etc

 

By: Name

Got login loop after installation. There are no errors in the nginx or nextcloud logs. Any idea why?

By: Alex Sharp

I have installed nextcloud and all worked great, running it on ubuntu 18.04 (lubuntu).

however, I hace docker installed and on it I have a container with OnlyOffice, since it runs on the same ip as the NC i have a problem. Since every port that I assign to the Docker Container is getting redirected to nextcloud, please help me to solve and or separate this problem.

Router configuration and port forwarding all okay, but evething opens NC login page..

Thank you!

By: ArturD

Thank you very much for this post!

It took les than 5 minutes to have Nextcloud running on VPS that already had PHP, MySQL and Nginx (one I;m using for Wordpress already)! You saved me a lot of time :-)

By: Ken Dyson

Hi, thanks for the tutorial.

Everything seems to go smoothy for my until I attempt to open the web browser and type the nextcloud URL address.

I configured nginx with the server name as dysoncloud.net.  I am installing it on the local machin with managed DNS (No-Ip).  I ran the configuration client and dysoncloud.net should rediect to 70.82.208.228:3200.  I use port 3200 because my ISP blocks port 80.  I set up the nginx config to listen 3200 default_server listen [::]:3200 default_server.

I set up port forwarding to have my router send 3200 and 443 to the local IP, in this case 192.168.0.29

nextcloud config looks like this:

upstream php-handler {    #server 127.0.0.1:9000;    server unix:/run/php/php7.1-fpm.sock;}server {    listen 3200;    listen [::]:3200;    server_name dysoncloud.net;    # enforce https   return 301 https://$server_name$request_uri;}server {    listen 443 ssl http2;    listen [::]:443 ssl http2;    server_name dysoncloud.net;    ssl_certificate /etc/ssl/certs/nginx-selfsigned.crt;    ssl_certificate_key /etc/ssl/private/nginx-selfsigned.key;

 

I didn't get any errors anywhere along the way.

 

When i go to dysonsphere.net I get a webpage that says:

The site you have requested is currently unavailable, please try back again later.

Any help would be greatly appeciated.

 

Thanks

By: gho

Hi.. thanks for the tutorial..

I have read your tutorial and also interesting for this way. But, before I go to start all of this stuff, I need to know something and let me ask for it. These are the reasons and the questions:

I have a domain name that available on diferent server. I want to try it for Google Cloud VM. In this case, I assume that the name_server will confused me if I do it as your instructions (my.domain.com). Because I have tried the way like this to install phpMyadmin and get fail.  So please give me a clue or something that should I know. Thanks.

 

By: ted

Thanks for the great tutorial. It works seamless.

By: Alex

Hello. Can I change all the things in the command from 7.1 to 7.3 or 7.4?

Updated. I change to 7.4 and there was an error :/

By: koxy

After installation i get 404 error instead of dashboard