NOTE: This fork of Sovereign is my own. It incorporates changes that the upstream maintainers were either too slow to implement or else decided not to merge for reasons unknown. These include features like:
- Roundcube from source (since it's missing from Debian Jessie)
- Giving users the ability to change their password via Roundcube
- Fixes for race condition in dropping/reloading the entire mailserver db on every run (WTF)
- Upgrade to Ansible v2.x
- Storage of passwords in Ansible Vault instead of in the vars files (hashed or otherwise)
- Lint cleanup for Ansible v2
- Fixes for OpenDKIM scripts that are broken in the upstream
- Postfix Admin
- Teardown of roles/tasks previously installed
Since I've gone off on my own, you can expect this to divurge even further from the original sovereign project. I'll likely remove things that are unnecessary, like newebe
and wallabag
simply because I don't use them and they aren't worth the time to maintain.
To be clear - this is up here for you to use if you want. It reflects what I'm running in production. Things are mostly tested, but as you can imagine, it's difficult to make changes to a production configuration and roll those up and sanitize them for release. If you find bugs or things that don't work as advertised, please open an issue. Since this is production-ready, if it's a component I use, I'll likely get it fixed right away.
The rest of this README is from the original project, modified by me to incorporate the Vault and some other niceties. I'll carve it up as I remove or modify stuff.
Sovereign is a set of Ansible playbooks that you can use to build and maintain your own personal cloud based entirely on open source software, so you’re in control.
If you’ve never used Ansible before, you might find these playbooks useful to learn from, since they show off a fair bit of what the tool can do.
The original author's background and motivations might be of interest. tl;dr: frustrations with Google Apps and concerns about privacy and long-term support.
Sovereign offers useful cloud services while being reasonably secure and low-maintenance. Use it to set up your server, SSH in every couple weeks, but mostly forget about it.
What do you get if you point Sovereign at a server? All kinds of good stuff!
- IMAP over SSL via Dovecot, complete with full text search provided by Solr.
- POP3 over SSL, also via Dovecot
- SMTP over SSL via Postfix, including a nice set of DNSBLs to discard spam before it ever hits your filters.
- Webmail via Roundcube.
- Mobile push notifications via Z-Push.
- Email client automatic configuration.
- Jabber/XMPP instant messaging via Prosody.
- An RSS Reader via Selfoss.
- Virtual domains for your email, backed by PostgreSQL.
- Secure on-disk storage for email and more via EncFS.
- Spam fighting via DSPAM and Postgrey.
- Mail server verification via OpenDKIM, so folks know you’re legit.
- CalDAV and CardDAV to keep your calendars and contacts in sync, via ownCloud.
- Your own private Dropbox, also via ownCloud.
- Your own VPN server via OpenVPN.
- An IRC bouncer via ZNC.
- Monit to keep everything running smoothly (and alert you when it’s not).
- collectd to collect system statistics.
- Web hosting (ex: for your blog) via Apache.
- Firewall management via Uncomplicated Firewall (ufw).
- Intrusion prevention via fail2ban and rootkit detection via rkhunter.
- SSH configuration preventing root login and insecure password authentication
- RFC6238 two-factor authentication compatible with Google Authenticator and various hardware tokens
- Nightly backups to Tarsnap.
- Git hosting via cgit and gitolite.
- Newebe, a social network.
- Read-it-later via Wallabag
- A bunch of nice-to-have tools like mosh and htop that make life with a server a little easier.
Don’t want one or more of the above services? Comment out the relevant role in site.yml
. Or get more granular and comment out the associated include:
directive in one of the playbooks.
- A VPS (or bare-metal server if you wanna ball hard). My VPS is hosted at Linode. You’ll probably want at least 512 MB of RAM between Apache, Solr, and PostgreSQL. Mine has 1024.
- 64-bit Debian 7 or an equivalent Linux distribution such as Ubuntu 14.04 LTS. (You can use whatever distro you want, but deviating from Debian will require more tweaks to the playbooks. See Ansible’s different packaging modules.) Support for Debian 8 and Ubuntu 16.04 is underway in the "jessie" branch.
- A wildcard SSL certificate. You can either buy one or self-sign if you want to save money.
- A Tarsnap account with some credit in it. You could comment this out if you want to use a different backup service. Consider paying your hosting provider for backups or using an additional backup service for redundancy.
Generate a private key and a certificate signing request (CSR):
openssl req -nodes -newkey rsa:2048 -keyout /tmp/wildcard_private.key -sha256 -out mycert.csr
Purchase a wildcard cert from a certificate authority, such as Positive SSL or AlphaSSL. You will provide them with the contents of your CSR, and in return they will give you your signed public certificate.
Place the certificate in /tmp/wildcard_public_cert.crt
.
Download your certificate authority’s combined cert to /tmp/wildcard_ca.pem
.
Lastly, test your certificate:
openssl verify -verbose -CAfile /tmp/wildcard_ca.pem /tmp/wildcard_public_cert.crt
Purchasing SSL certs, and wildcard certs specifically, can be a significant financial burden. It is possible to generate a self-signed SSL certificate (i.e. one that isn’t signed by a Certificate Authority) that is free of charge by nature. However, since a self-signed cert has no CA chain that can confirm its authenticity, some services might behave erratically when using such a certificate.
To create a self-signed SSL cert, run the following commands:
openssl req -nodes -newkey rsa:2048 -keyout /tmp/wildcard_private.key -sha256 -out mycert.csr
openssl x509 -req -days 365 -in mycert.csr -signkey /tmp/wildcard_private.key -out /tmp/wildcard_public_cert.crt
cp /tmp/wildcard_public_cert.crt /tmp/wildcard_ca.pem
If you haven’t already, download and install Tarsnap, or use brew install tarsnap
if you use Homebrew.
Create a new machine key for your server:
tarsnap-keygen --keyfile roles/tarsnap/files/decrypted_tarsnap.key --user me@example.com --machine example.com
For goodness sake, change the root password:
passwd
Create a user account for Ansible to do its thing through:
useradd deploy
passwd deploy
mkdir /home/deploy
Authorize your ssh key if you want passwordless ssh login (optional):
mkdir /home/deploy/.ssh
chmod 700 /home/deploy/.ssh
nano /home/deploy/.ssh/authorized_keys
chmod 400 /home/deploy/.ssh/authorized_keys
chown deploy:deploy /home/deploy -R
Create the file /etc/sudoers.d/deploy
with the following content:
Defaults:deploy !requiretty
deploy ALL=(ALL) NOPASSWD: ALL
Your new account will be automatically set up for passwordless sudo
.
NOTE: The option !requiretty
is necessary to enable pipelining, which is, in turn, required by portions of the configuration that use become
to become another unprivileged user (such as the Google Authenticator component). Pipelining will dramatically speed up Ansible if run remotely over SSH.
This fork of Sovereign takes advantage of the secure vault provided by Ansible. Sensitive variables have been moved out of user.yml
and into private.yml
. This includes SSL certificates, passwords (hashed and unhashed), and anything else that shouldn't be left laying around in a plaintext file. Doing this means that you can safely put your copy of Sovereign into a repository and benefit from source code control without having to worry about passwords being exposed to the world.
Most of the variables have been moved over verbatim, so if something used to be in user.yml
, you'll find it now in private.yml
in the same format. Defaults are all still contained in defaults.yml
for reference. The vault is loaded after user.yml
, so even if you put something in there, variable values will be updated when private.yml
is loaded.
See the README in vars
for specific information about using Ansible Vault.
Modify the settings in vars/user.yml
and vars/private.yml
to your liking. If you want to see how they’re used in context, just search for the corresponding string in the playbooks.
For your SSL certificate, copy the key, certificate, and concatenated CA certificates to the corresponding sections in private.yml
. When pasting the information, maintain indentation, and keep the lines that say 'BEGIN' and 'END' for each piece.
Setting password_hash
for your mail users is a bit tricky. You can generate one using doveadm-pw.
# doveadm pw -s SHA512-CRYPT
Enter new password: foo
Retype new password: foo
{SHA512-CRYPT}$6$drlIN9fx7Aj7/iLu$XvjeuQh5tlzNpNfs4NwxN7.HGRLglTKism0hxs2C1OvD02d3x8OBN9KQTueTr53nTJwVShtCYiW80SGXAjSyM0
Remove {SHA512-CRYPT}
and insert the rest as the password_hash
value.
Alternatively, if you don’t already have doveadm
installed, Python 3.3 or higher on Linux will generate the appropriate string for you (assuming your password is password
):
python3 -c 'import crypt; print(crypt.crypt("password", salt=crypt.METHOD_SHA512))'
On OS X and other platforms the passlib package may be used to generate the required string:
python -c 'import passlib.hash; print(passlib.hash.sha512_crypt.encrypt("password", rounds=5000))'
Same for the IRC password hash…
# znc --makepass
[ ** ] Type your new password.
[ ?? ] Enter Password: foo
[ ?? ] Confirm Password: foo
[ ** ] Kill ZNC process, if it's running.
[ ** ] Then replace password in the <User> section of your config with this:
<Pass password>
Method = sha256
Hash = 310c5f99825e80d5b1d663a0a993b8701255f16b2f6056f335ba6e3e720e57ed
Salt = YdlPM5yjBmc/;JO6cfL5
</Pass>
[ ** ] After that start ZNC again, and you should be able to login with the new password.
Take the strings after Hash =
and Salt =
and insert them as the value for irc_password_hash
and irc_password_salt
respectively.
Alternatively, if you don’t already have znc
installed, Python 3.3 or higher on Linux will generate the appropriate string for you (assuming your password is password
):
python3 -c 'import crypt; print("irc_password_salt: {}\nirc_password_hash: {}".format(*crypt.crypt("password", salt=crypt.METHOD_SHA256).split("$")[2:]))'
On OS X and other platforms the passlib:https://pythonhosted.org/passlib/ package may be used to generate the required string:
python -c 'import passlib.hash; print("irc_password_salt: {}\nirc_password_hash: {}".format(*passlib.hash.sha256_crypt.encrypt("password", rounds=5000).split("$")[2:]))'
For Git hosting, copy your public key into place:
cp ~/.ssh/id_rsa.pub roles/git/files/gitolite.pub
Finally, replace the TODOs in the file hosts
. If your SSH daemon listens on a non-standard port, add a colon and the port number after the IP address. In that case you also need to add your custom port to the task Set firewall rules for web traffic and SSH
in the file roles/common/tasks/ufw.yml
.
First, make sure you’ve got Ansible 2.0+ installed.
To run the whole dang thing:
ansible-playbook -i ./hosts site.yml
To run just one or more piece, use tags. I try to tag all my includes for easy isolated development. For example, to focus in on your firewall setup:
ansible-playbook -i ./hosts --tags=ufw site.yml
You might find that it fails at one point or another. This is probably because something needs to be done manually, usually because there’s no good way of automating it. Fortunately, all the tasks are clearly named so you should be able to find out where it stopped. I’ve tried to add comments where manual intervention is necessary.
The dependencies
tag just installs dependencies, performing no other operations. The tasks associated with the dependencies
tag do not rely on the user-provided settings that live in vars/user.yml
. Running the playbook with the dependencies
tag is particularly convenient for working with Docker images.
If you find that you no longer wish to be running a particular component, or if you know up front that you don't need it, edit main.yml
for that role and change the include line to reference the yml file that starts with no_
:
- include: no_znc.yml tags=znc
Not everything can be excluded. For example, nothing from mailserver
or common
can be excluded - these are core components of the system. If there is no no_X.yml
version of X, it can't safely be excluded.
The tasks that perform exclusion will actually undo any installation of those components or entirely prevent them from being installed. The nice thing about Ansible is that if you decide to use them later, you can just change it back, and all of the configuration for them is contained within these files.
If you’ve just bought a new domain name, point it at Linode’s DNS Manager or similar. Most VPS services (and even some domain registrars) offer a managed DNS service that you can use for this at no charge. If you’re using an existing domain that’s already managed elsewhere, you can probably just modify a few records.
Create A
records which point to your server's IP address:
example.com
mail.example.com
autoconfig.example.com
(for email client automatic configuration)read.example.com
(for Wallabag)news.example.com
(for Selfoss)cloud.example.com
(for ownCloud)git.example.com
(for cgit)
Create a MX
record for example.com
which assigns mail.example.com
as the domain’s mail server.
To ensure your emails pass DKIM checks you need to add a txt
record. The name field will be default._domainkey.EXAMPLE.COM.
The value field contains the public key used by OpenDKIM. The exact value needed can be found in the file /etc/opendkim/keys/EXAMPLE.COM/default.txt
it’ll look something like this:
v=DKIM1; k=rsa; p=MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDKKAQfMwKVx+oJripQI+Ag4uTwYnsXKjgBGtl7Tk6UMTUwhMqnitqbR/ZQEZjcNolTkNDtyKZY2Z6LqvM4KsrITpiMbkV1eX6GKczT8Lws5KXn+6BHCKULGdireTAUr3Id7mtjLrbi/E3248Pq0Zs39hkDxsDcve12WccjafJVwIDAQAB
For DMARC you'll also need to add a txt
record. The name field should be _dmarc.EXAMPLE.COM
and the value should be v=DMARC1; p=none
. More info on DMARC can be found here
Set up SPF and reverse DNS as per this post. Make sure to validate that it’s all working, for example by sending an email to check-auth@verifier.port25.com and reviewing the report that will be emailed back to you.
Sign in to the ZNC web interface and set things up to your liking. It isn’t exposed through the firewall, so you must first set up an SSH tunnel:
ssh deploy@example.com -L 6643:localhost:6643
Then proceed to http://localhost:6643 in your web browser.
Similarly, to access the server monitoring page, use another SSH tunnel:
ssh deploy@example.com -L 2812:localhost:2812
Again proceeding to http://localhost:2812 in your web browser.
Finally, sign into ownCloud to set it up. You should select PostgreSQL as the configuration backend.
We’re collecting known-good client setups on our wiki.
If you run into an errors, please check the wiki page. If the problem you encountered, is not listed, please go ahead and create an issue. If you already have a bugfix and/or workaround, just put them in the issue and the wiki page.
You will need to manually enter the password for any encrypted volumes on reboot. This is not Sovereign-specific, but rather a function of how EncFS works. This will necessitate SSHing into your machine after reboot, or accessing it via a console interface if one is available to you. Once you're in, run this:
encfs /encrypted /decrypted --public
It is possible that some daemons may need to be restarted after you enter your password for the encrypted volume(s). Some services may stall out while looking for resources that will only be available once the /decrypted
volume is available and visible to daemon user accounts.
Ask questions and provide feedback in #sovereign
on Freenode.