2018-09-01 11:53:14 +00:00
# Installing Plume (for development or production)
2018-06-28 23:28:15 +00:00
2018-09-01 11:53:14 +00:00
## Prerequisites
In order to be installed and to work correctly, Plume needs:
2018-09-02 11:13:17 +00:00
- *Git* (to get the code)
- *Curl* (for RustUp, the Rust installer)
- *GCC* (to compile C dependencies)
- *PostgreSQL* (for the database)
- *GetText* (to manage translations)
- *Rust* and *Cargo* (to build the code)
2018-09-01 11:53:14 +00:00
All the following instructions will need a terminal.
Here are the commands to install PostgreSQL and GetText on various operating systems.
Some of them may need root permissions.
On **Debian** :
2018-06-29 17:13:58 +00:00
```bash
2018-06-28 23:28:15 +00:00
apt update
2018-09-01 11:53:14 +00:00
apt install gettext postgresql postgresql-contrib libpq-dev git
```
On **Fedora** , **CentOS** or **RHEL** :
```bash
dnf install postgresql-server postgresql-contrib mariadb-devel libsq3-devel libpqxx libpqxx-devel
# TODO: GetText + Git install
```
On **Gentoo** :
```bash
emerge --sync
emerge -av postgresql eselect-postgresql
# TODO: GetText + Git install
```
On **Mac OS X** , with [Homebrew ](https://brew.sh/ ):
```bash
brew update
brew install postgres
# TODO: GetText + Git install
```
## Creating a new user (optional)
This step is recommended if you are in a **production environment** , but it is not necessary.
```bash
2018-06-28 23:28:15 +00:00
adduser plume
su - plume
2018-09-01 11:53:14 +00:00
cd ~
```
Creating a new user will let you use systemd to manage Plume if you want (see the dedicated section below).
## Installing Rust and Cargo
We said that Plume needed Rust and Cargo to work, but we didn't installed them at the same time as PostgreSQL and GetText, because there is an universal installation method called RustUp.
You can install it on **GNU/Linux** and **Mac OS X** with:
```bash
curl https://sh.rustup.rs -sSf | sh
```
2018-09-02 11:13:17 +00:00
When asked, choose the *"1) Proceed with installation (default)"* option.
Then run this command to be able to run cargo in the current session:
```bash
export PATH="$PATH:/home/plume/.cargo/bin:/home/plume/.local/bin:/usr/local/sbin"
```
2018-09-01 17:32:31 +00:00
On **Windows** , you'll need, if you don't already have them, to download and install the [Visual C++ 2015 Build Tools ](https://www.microsoft.com/en-us/download/details.aspx?id=48159 ). Then, download the [rustup installer ](https://www.rust-lang.org/en-US/install.html ) and run it.
2018-09-01 11:53:14 +00:00
## Getting and compiling the Plume source code
Plume needs to be compiled from source.
```bash
2018-06-28 23:28:15 +00:00
git clone https://github.com/Plume-org/Plume.git
cd Plume
2018-09-01 11:53:14 +00:00
# This may take some time as RustUp will download all
# the required Rust components, and Cargo will download
# and compile all dependencies.
cargo build
2018-06-29 17:13:58 +00:00
```
2018-06-28 23:28:15 +00:00
2018-09-01 11:53:14 +00:00
We may provide precompiled packages and Docker images in the future (if you have experience in these fields and want to help, you're welcome).
2018-06-29 17:13:58 +00:00
2018-09-01 11:53:14 +00:00
## Configuring PostgreSQL
You can either run PostgreSQL from the machine that runs Plume, or from another server. We recommend you to use the first setup for development environments, or in production for small instances.
In the first case, just run this command after the PostgreSQL installation, to start it:
```
2018-06-29 19:15:10 +00:00
service postgresql start
2018-06-29 17:13:58 +00:00
```
2018-06-28 23:28:15 +00:00
2018-09-01 11:53:14 +00:00
If you want to have two separate machines, run these commands on the database server after you installed the dependencies mentionned above on both servers:
2018-06-29 17:13:58 +00:00
```bash
2018-06-28 23:34:00 +00:00
service postgresql start
2018-06-28 23:28:15 +00:00
su - postgres
createuser -d -P plume
createdb -O plume plume
2018-06-29 17:13:58 +00:00
```
2018-06-28 23:28:15 +00:00
2018-09-01 11:53:14 +00:00
## Running migrations
Migrations are scripts to update the database. They are run by a tool called Diesel, which can be installed with:
2018-06-29 17:13:58 +00:00
```bash
2018-09-01 11:53:14 +00:00
cargo install diesel_cli --no-default-features --features postgres --version '=1.2.0'
2018-06-29 17:13:58 +00:00
```
2018-06-28 23:32:15 +00:00
2018-09-01 11:53:14 +00:00
Plume should normally run migrations for you when needed, but if you want to run them manually, the command is:
2018-06-28 23:32:15 +00:00
2018-09-01 11:53:14 +00:00
```bash
diesel migration run --database-url postgres://USER:PASSWORD@IP:PORT/plume
```
2018-06-28 23:32:15 +00:00
2018-09-01 11:53:14 +00:00
This command may be useful if you decided to use a separate database server.
2018-06-28 23:32:15 +00:00
2018-09-01 11:53:14 +00:00
## Starting Plume
2018-06-28 23:32:15 +00:00
2018-09-01 11:53:14 +00:00
When you launch Plume for the first time, it will ask you a few questions to setup your instance before it actually launches. To start it, run these commands.
2018-06-29 17:13:58 +00:00
2018-09-01 11:53:14 +00:00
```
# Optional, only do it if the database URL is not
# postgres://plume:plume@localhost/plume
export DB_URL=postgres://plume:PASSWORD@DBSERVERIP:DBPORT/plume
cargo run
```
2018-06-29 19:15:10 +00:00
2018-09-01 11:53:14 +00:00
## Configuring Nginx
2018-06-29 19:09:25 +00:00
2018-09-01 17:24:35 +00:00
Here is a sample Nginx configuration for a Plume instance (replace `blog.example.com` with your domain name):
2018-09-01 11:53:14 +00:00
```nginx
2018-09-01 17:24:35 +00:00
server {
listen 80;
server_name blog.example.com;
location /.well-known/acme-challenge {}
location / {
return 301 https://$host$request_uri;
}
}
2018-09-01 11:53:14 +00:00
2018-09-01 17:24:35 +00:00
server {
server_name blog.example.org;
access_log /var/log/nginx/access.log;
listen [::]:443 ssl; # managed by Certbot
SSLCertificateFile /etc/letsencrypt/live/blog.example.com/cert.pem
SSLCertificateKeyFile /etc/letsencrypt/live/blog.example.com/privkey.pem
SSLCertificateChainFile /etc/letsencrypt/live/blog.example.com/chain.pem
# for ssl conf: https://cipherli.st/
ssl_protocols TLSv1.2 TLSv1.3;# Requires nginx >= 1.13.0 else use TLSv1.2
2018-09-01 23:34:03 +00:00
ssl_prefer_server_ciphers on;
2018-09-01 17:24:35 +00:00
ssl_dhparam /etc/letsencrypt/ssl-dhparams.pem;# openssl dhparam -out /etc/letsencrypt/ssl-dhparam.pem 4096
ssl_ciphers ECDHE-RSA-AES256-GCM-SHA512:DHE-RSA-AES256-GCM-SHA512:ECDHE-RSA-AES256-GCM-SHA384:DHE-RSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-SHA384;
ssl_ecdh_curve secp384r1; # Requires nginx >= 1.1.0
ssl_session_timeout 10m;
ssl_session_cache shared:SSL:10m;
ssl_session_tickets off; # Requires nginx >= 1.5.9
ssl_stapling on; # Requires nginx >= 1.3.7
ssl_stapling_verify on; # Requires nginx => 1.3.7
resolver 9.9.9.9 80.67.169.12 valid=300s;
2018-09-01 23:34:03 +00:00
resolver_timeout 5s;
2018-09-01 17:24:35 +00:00
add_header Strict-Transport-Security "max-age=63072000; includeSubDomains; preload";
add_header X-Frame-Options DENY;
add_header X-Content-Type-Options nosniff;
add_header X-XSS-Protection "1; mode=block";
add_header Content-Security-Policy "default-src 'self';";
add_header Content-Security-Policy "frame-ancestors 'self'";
location ~* \.(jpg|jpeg|png|gif|ico|js|pdf)$ {
add_header Cache-Control "public";
expires 7d;
}
location / {
2018-09-01 23:34:03 +00:00
proxy_pass http://localhost:7878/;
2018-09-01 17:24:35 +00:00
proxy_set_header Host $http_host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto $scheme;
client_max_body_size 10m;
}
2018-09-01 11:53:14 +00:00
}
2018-06-29 19:09:25 +00:00
```
2018-09-01 17:24:35 +00:00
## Configuring Apache
If you prefer Apache, you can use this configuration (here too replace `blog.example.com` with your domain):
```apache
< VirtualHost * :80 >
ServerName blog.example.com
Redirect / https://blog.example.com/
< / VirtualHost >
< VirtualHost * :443 >
ServerAdmin admin@example.com
ServerName blog.example.com
Header always set Referrer-Policy "strict-origin-when-cross-origin"
Header always set Strict-Transport-Security "max-age=31536000"
SSLEngine on
# for cipher conf: https://cipherli.st/
SSLCipherSuite EECDH+AESGCM:EDH+AESGCM:AES256+EECDH:AES256+EDH
SSLProtocol All -SSLv2 -SSLv3 -TLSv1 -TLSv1.1
SSLHonorCipherOrder On
Header always set Strict-Transport-Security "max-age=63072000; includeSubDomains; preload"
Header always set X-Frame-Options DENY
Header always set X-Content-Type-Options nosniff
SSLCompression off
SSLUseStapling on
SSLStaplingCache "shmcb:logs/stapling-cache(150000)"
# Requires Apache >= 2.4.11
SSLSessionTickets Off
SSLCertificateFile /etc/letsencrypt/live/blog.example.com/cert.pem
SSLCertificateKeyFile /etc/letsencrypt/live/blog.example.com/privkey.pem
SSLCertificateChainFile /etc/letsencrypt/live/blog.example.com/chain.pem
ProxyPreserveHost On
RequestHeader set X-Forwarded-Proto "https"
ProxyPass / http://127.0.0.1:7878/
ProxyPassReverse / http://127.0.0.1:7878/
< / VirtualHost >
```
2018-09-01 11:53:14 +00:00
## Systemd integration
2018-09-01 23:34:03 +00:00
If you want to manage your Plume instance with systemd, you can use the following unit file (to be saved in `/etc/systemd/system/plume.service` ):
2018-09-01 11:53:14 +00:00
```toml
[Unit]
Description=plume
[Service]
Type=simple
User=plume
WorkingDirectory=/home/plume/Plume
2018-09-01 17:18:58 +00:00
ExecStart=/home/plume/.cargo/bin/cargo run
2018-09-01 11:53:14 +00:00
TimeoutSec=30
Restart=always
[Install]
WantedBy=multi-user.target
```
2018-09-01 23:34:03 +00:00
## SysVinit integration
This script can also be useful if you are using SysVinit.
```bash
#!/bin/bash
#
# chkconfig: 35 90 12
# description: Plume
#
# Get function from functions library
. /etc/init.d/functions
# Start Plume
start() {
initlog -c "echo -n Démarrage de Plume: "
cd /home/plume/Plume & cargo run
### Create the lock file ###
touch /var/lock/subsys/plume
success $"Plume est prêt !"
echo
}
# Restart Plume
stop() {
initlog -c "echo -n Arrêt de Plume: "
killproc cargo run
### Now, delete the lock file ###
rm -f /var/lock/subsys/plume
echo
}
### main logic ###
case "$1" in
start)
start
;;
stop)
stop
;;
status)
status FOO
;;
restart|reload|condrestart)
stop
start
;;
*)
echo $"Usage: $0 {start|stop|restart|reload|status}"
exit 1
esac
exit 0
```
2018-09-01 11:53:14 +00:00
## Caveats:
- Pgbouncer is not yet supported (named transactions are used).
- Rust nightly is a moving target, dependancies can break and sometimes you need to check a few versions to find the one working (run `rustup override set nightly-2018-05-15` or `rustup override set nightly-2018-05-31` in the Plume directory if you have issues during the compilation)
- Rust nightly 2018-06-28 is known to be failing to compile diesel 1.3.2
## Acknowledgements
2018-06-29 19:09:25 +00:00
2018-09-01 17:24:35 +00:00
Most of this documentation have been written by *gled-rs* . The systemd unit file, Nginx and Apache configurations have been written by *nonbinaryanargeek* . Some parts (especially the instructions to install native dependencies) are from the [Aardwolf project ](https://github.com/Aardwolf-Social/aardwolf ).