Compare commits
8 Commits
01e83795ad
...
38607ec437
Author | SHA1 | Date | |
---|---|---|---|
38607ec437 | |||
8fb1b29aef | |||
21c25413fc | |||
3e0e0beb0f | |||
fac9de10aa | |||
e9c31546ab | |||
be19efb887 | |||
43b28bc982 |
341
content/backup-strategy.md
Normal file
341
content/backup-strategy.md
Normal file
|
@ -0,0 +1,341 @@
|
||||||
|
Title: My Backup Strategy
|
||||||
|
Date: 2021-04-08
|
||||||
|
Category: Writing
|
||||||
|
Summary: Details about the backup system for my data.
|
||||||
|
Wide: true
|
||||||
|
|
||||||
|
[TOC]
|
||||||
|
|
||||||
|
Regularly backing up all the data I care about is very important to me. This
|
||||||
|
article outlines my strategy to make sure I never lose essential data.
|
||||||
|
|
||||||
|
## Motivation
|
||||||
|
|
||||||
|
Backups should be as automatic as possible. This ensures laziness and
|
||||||
|
forgetfulness won't interfere with the regularity.
|
||||||
|
|
||||||
|
All software used to create and store the backups should be free and open source
|
||||||
|
so I'm not depending on the survival of a company.
|
||||||
|
|
||||||
|
Backups need to be tested to ensure they are correct and happening regularly.
|
||||||
|
Multiple copies of the backups should exist, including at least one offsite to
|
||||||
|
protect against my building burning down.
|
||||||
|
|
||||||
|
Backups should also be incremental when possible (rather than mirror copies) so
|
||||||
|
an accidental deletion isn't propagated into the backups, making the file
|
||||||
|
irrecoverable.
|
||||||
|
|
||||||
|
## Strategy
|
||||||
|
|
||||||
|
I have one backup folder `/mnt/backup` on my media server at home that serves as
|
||||||
|
the destination for all my backup sources. All scheduled automatic backups write
|
||||||
|
to their own subfolder inside of it.
|
||||||
|
|
||||||
|
This backup folder is then synced to encrypted 2.5" 1 TB hard drives which I
|
||||||
|
rotate between my bag, offsite, and my parent's house.
|
||||||
|
|
||||||
|
## Backup Sources
|
||||||
|
|
||||||
|
I use the tool `rdiff-backup` extensively because it allows me to take
|
||||||
|
incremental backups locally or over SSH. It acts very similar to `rsync` and has
|
||||||
|
no configuration.
|
||||||
|
|
||||||
|
### Email
|
||||||
|
|
||||||
|
I have every email since 2010 backed up continuously in case my email provider
|
||||||
|
disappears.
|
||||||
|
|
||||||
|
I use `offlineimap` to sync my mail to the directory `~/email` on my media
|
||||||
|
server as a Maildir. Since offlineimap is only a syncing tool, the emails need
|
||||||
|
to be copied elsewhere to be backed up. I run `rdiff-backup` from a weekly cron
|
||||||
|
job:
|
||||||
|
|
||||||
|
<span class="aside">I'll explain what backup_check.txt does below</span>
|
||||||
|
|
||||||
|
```
|
||||||
|
*/15 * * * * offlineimap > /var/log/offlineimap.log 2>&1
|
||||||
|
00 12 * * 1 date -Iseconds > /home/email/email/backup_check.txt
|
||||||
|
|
||||||
|
20 12 * * 1 rdiff-backup /home/email/email /mnt/backup/local/email/
|
||||||
|
40 12 * * 1 rdiff-backup --remove-older-than 12B --force /mnt/backup/local/email/
|
||||||
|
```
|
||||||
|
|
||||||
|
Here's my `.offlineimaprc` for reference:
|
||||||
|
|
||||||
|
```
|
||||||
|
[general]
|
||||||
|
accounts = main
|
||||||
|
[Account main]
|
||||||
|
localrepository = Local
|
||||||
|
remoterepository = Remote
|
||||||
|
[Repository Local]
|
||||||
|
type = Maildir
|
||||||
|
localfolders = ~/email
|
||||||
|
[Repository Remote]
|
||||||
|
type = IMAP
|
||||||
|
readonly = True
|
||||||
|
folderfilter = lambda foldername: foldername not in ['Trash', 'Spam', 'Drafts']
|
||||||
|
remotehost = example.com
|
||||||
|
remoteuser = mail@example.com
|
||||||
|
remotepass = supersecret
|
||||||
|
sslcacertfile = /etc/ssl/certs/ca-certificates.crt
|
||||||
|
```
|
||||||
|
|
||||||
|
### Notes
|
||||||
|
|
||||||
|
I use Standard Notes to take notes and wrote the tool
|
||||||
|
[standardnotes-fs](https://github.com/tannercollin/standardnotes-fs) to mount my
|
||||||
|
notes as a file system to view and edit them as plain text files.
|
||||||
|
|
||||||
|
I take weekly backups of the mounted file system on my media server with cron:
|
||||||
|
|
||||||
|
```
|
||||||
|
00 12 * * 1 date -Iseconds > /home/notes/notes/backup_check.txt
|
||||||
|
15 12 * * 1 rdiff-backup /home/notes/notes /mnt/backup/local/notes/
|
||||||
|
```
|
||||||
|
|
||||||
|
### Nextcloud
|
||||||
|
|
||||||
|
I self-host a Nextcloud instance to store all my personal documents (non-code
|
||||||
|
projects, tax forms, spreadsheets, etc.). Since it's only a syncing software,
|
||||||
|
the files need to be copied elsewhere to be backed up.
|
||||||
|
|
||||||
|
I take weekly backups of the Nextcloud data folder with cron:
|
||||||
|
|
||||||
|
```
|
||||||
|
00 12 * * 1 rdiff-backup /var/www/nextcloud/data/tanner/files /mnt/backup/local/nextcloud/
|
||||||
|
30 12 * * 1 rdiff-backup --remove-older-than 12B --force /mnt/backup/local/nextcloud/
|
||||||
|
```
|
||||||
|
|
||||||
|
### Gitea
|
||||||
|
|
||||||
|
I self-host a Gitea instance to store all my git repositories for code-based
|
||||||
|
projects. My home folder is also a git repo so I can easily sync my config files
|
||||||
|
and password database between servers and machines.
|
||||||
|
|
||||||
|
I take weekly backups of the Gitea data folder with cron:
|
||||||
|
|
||||||
|
```
|
||||||
|
00 12 * * 1 date -Iseconds > /home/gitea/gitea/data/backup_check.txt
|
||||||
|
10 12 * * 1 rdiff-backup --exclude **data/indexers --exclude **data/sessions /home/gitea/gitea/data /mnt/backup/local/gitea/
|
||||||
|
35 12 * * 1 rdiff-backup --remove-older-than 12B --force /mnt/backup/local/gitea/
|
||||||
|
```
|
||||||
|
|
||||||
|
### Telegram
|
||||||
|
|
||||||
|
Telegram Messenger is my main app for communication. My parents, most of my
|
||||||
|
friends, and friend groups are on there so I don't want to lose those messages
|
||||||
|
in case Telegram disappears or my account gets banned.
|
||||||
|
|
||||||
|
Telegram includes a data export feature, but it can't be automated. Instead I
|
||||||
|
run the deprecated software
|
||||||
|
[telegram-export](https://github.com/expectocode/telegram-export) hourly with
|
||||||
|
cron:
|
||||||
|
|
||||||
|
```
|
||||||
|
0 * * * * bash -c 'timeout 50m /home/tanner/opt/telegram-export/env/bin/python -m telegram_export' > /var/log/telegramexport.log 2>&1
|
||||||
|
```
|
||||||
|
|
||||||
|
It likes to hang, so `timeout` kills it if it's still running after 50 minutes.
|
||||||
|
Hasn't corrupted the database yet.
|
||||||
|
|
||||||
|
### Phone
|
||||||
|
|
||||||
|
[Signal
|
||||||
|
Messenger](https://play.google.com/store/apps/details?id=org.thoughtcrime.securesms&hl=en_CA&gl=US)
|
||||||
|
automatically exports a copy of my text messages database, and
|
||||||
|
[Aegis](https://play.google.com/store/apps/details?id=com.beemdevelopment.aegis&hl=en_CA&gl=US)
|
||||||
|
allows me to export an encrypted JSON file of my two-factor authentication
|
||||||
|
codes.
|
||||||
|
|
||||||
|
I mount my phone's internal storage as a file system on my desktop using
|
||||||
|
[adbfs-rootless](https://github.com/spion/adbfs-rootless). I then rsync the
|
||||||
|
files over to my media server:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ ./adbfs ~/mntphone
|
||||||
|
$ time rsync -Wav \
|
||||||
|
--exclude '*cache' --exclude nobackup \
|
||||||
|
--exclude '*thumb*' --exclude 'Telegram *' \
|
||||||
|
--exclude 'collection.media' \
|
||||||
|
--exclude 'org.thunderdog.challegram' \
|
||||||
|
--exclude '.trashed-*' --exclude '.pending-*' \
|
||||||
|
~/mntphone/storage/emulated/0/ \
|
||||||
|
localmediaserver:/mnt/backup/files/phone/
|
||||||
|
```
|
||||||
|
|
||||||
|
Unfortunately this is a manual process because I need to plug my phone in each
|
||||||
|
time. Ideally it would happen automatically while I'm asleep and the phone is
|
||||||
|
charging.
|
||||||
|
|
||||||
|
### Miscellaneous Files
|
||||||
|
|
||||||
|
The directory `/backup/files` is a repository for any kind of files I want to
|
||||||
|
keep forever. My phone data, old archives, computer files, Minecraft worlds,
|
||||||
|
files from previous jobs, and so on.
|
||||||
|
|
||||||
|
All the files will be included in the 1 TB hard drive backup rotations.
|
||||||
|
|
||||||
|
### Web Services
|
||||||
|
|
||||||
|
Web services that I run like [txt.t0.vc](https://txt.t0.vc) and
|
||||||
|
[QotNews](https://news.t0.vc) are backed up daily, weekly, and monthly depending
|
||||||
|
on how frequently the data changes.
|
||||||
|
|
||||||
|
I run `rdiff-backup` on the remote server with cron:
|
||||||
|
|
||||||
|
```
|
||||||
|
00 14 * * * date -Iseconds > /home/tanner/tbot/t0txt/data/backup_check.txt
|
||||||
|
|
||||||
|
04 14 * * * rdiff-backup /home/tanner/tbot/t0txt/data tbotbak@remotebackup::/mnt/backup/remote/tbotbak/daily/t0txt/
|
||||||
|
14 14 * * * rdiff-backup --remove-older-than 12B --force tbotbak@remotebackup::/mnt/backup/remote/tbotbak/daily/t0txt/
|
||||||
|
|
||||||
|
24 14 * * 1 rdiff-backup /home/tanner/tbot/t0txt/data tbotbak@remotebackup::/mnt/backup/remote/tbotbak/weekly/t0txt/
|
||||||
|
34 14 * * 1 rdiff-backup --remove-older-than 12B --force tbotbak@remotebackup::/mnt/backup/remote/tbotbak/weekly/t0txt/
|
||||||
|
|
||||||
|
44 14 1 * * rdiff-backup /home/tanner/tbot/t0txt/data tbotbak@remotebackup::/mnt/backup/remote/tbotbak/monthly/t0txt/
|
||||||
|
55 14 1 * * rdiff-backup --remove-older-than 12B --force tbotbak@remotebackup::/mnt/backup/remote/tbotbak/monthly/t0txt/
|
||||||
|
```
|
||||||
|
|
||||||
|
The user `tbotbak` has write access only to the `/mnt/backup/remote/tbotbak`
|
||||||
|
directory. It has its own passwordless SSH key that's only permitted to run the
|
||||||
|
`rdiff-backup --server` command for security.
|
||||||
|
|
||||||
|
### Protospace
|
||||||
|
|
||||||
|
I run a lot of services for [Protospace](https://protospace.ca/), my city's
|
||||||
|
makerspace.
|
||||||
|
|
||||||
|
The member portal I wrote called [Spaceport](https://my.protospace.ca/) creates
|
||||||
|
an archive I download daily:
|
||||||
|
|
||||||
|
```
|
||||||
|
40 10 * * * wget --content-disposition \
|
||||||
|
--header="Authorization: secretkeygoeshere" \
|
||||||
|
--directory-prefix /mnt/backup/remote/portalbak/ \
|
||||||
|
--no-verbose --append-output=/var/log/portalbackup.log \
|
||||||
|
https://api.my.protospace.ca/backup/
|
||||||
|
```
|
||||||
|
|
||||||
|
The main website and [wiki](https://wiki.protospace.ca) that I sysadmin gets
|
||||||
|
backed up weekly:
|
||||||
|
|
||||||
|
```
|
||||||
|
0 12 * * 1 mysqldump --all-databases > /var/www/dump.sql
|
||||||
|
15 12 * * 1 date -Iseconds > /var/www/backup_check.txt
|
||||||
|
20 12 * * 1 rdiff-backup /var/www pshostbak@remotebackup::/mnt/backup/remote/pshostbak/weekly/www/
|
||||||
|
```
|
||||||
|
|
||||||
|
The Protospace [Minecraft
|
||||||
|
server](http://games.protospace.ca:8123/?worldname=world&mapname=flat&zoom=3&x=74&y=64&z=354)
|
||||||
|
I run gets backed up daily:
|
||||||
|
|
||||||
|
```
|
||||||
|
00 15 * * * date -Iseconds > /home/tanner/minecraft/backup_check.txt
|
||||||
|
00 15 * * * rdiff-backup --exclude **CoreProtect --exclude **dynmap /home/tanner/minecraft psminebak@remotebackup::/mnt/backup/remote/psminebak/
|
||||||
|
30 15 * * * rdiff-backup --remove-older-than 12B --force psminebak@remotebackup::/mnt/backup/remote/psminebak/
|
||||||
|
```
|
||||||
|
|
||||||
|
I also back up our Google Drive with rclone:
|
||||||
|
|
||||||
|
```
|
||||||
|
45 12 * * 1 rclone copy -v protospace: /mnt/backup/files/protospace/google-drive/
|
||||||
|
```
|
||||||
|
|
||||||
|
## Backup Copies
|
||||||
|
|
||||||
|
My backup folder `/mnt/backup` now looks like this:
|
||||||
|
|
||||||
|
```
|
||||||
|
/mnt/backup/
|
||||||
|
├── files
|
||||||
|
│ ├── docs
|
||||||
|
│ ├── phone
|
||||||
|
│ ├── protospace
|
||||||
|
│ ├── telegram
|
||||||
|
│ ├── usbsticks
|
||||||
|
│ └── ... and so on
|
||||||
|
├── local
|
||||||
|
│ ├── email
|
||||||
|
│ ├── gitea
|
||||||
|
│ ├── nextcloud
|
||||||
|
│ └── notes
|
||||||
|
└── remote
|
||||||
|
├── portalbak
|
||||||
|
├── pshostbak
|
||||||
|
├── psminebak
|
||||||
|
├── tbotbak
|
||||||
|
└── telebak
|
||||||
|
```
|
||||||
|
|
||||||
|
This directory tree is the master backup and I make a copy of the entire tree
|
||||||
|
every Saturday to a hard drive.
|
||||||
|
|
||||||
|
The directory is copied over with the following script:
|
||||||
|
|
||||||
|
```text
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
cryptsetup luksOpen /dev/sdf external
|
||||||
|
mount /dev/mapper/external /mnt/external
|
||||||
|
|
||||||
|
time rsync -av --delete /mnt/backup/local/ /mnt/external/backup/local/
|
||||||
|
time rsync -av --delete /mnt/backup/remote/ /mnt/external/backup/remote/
|
||||||
|
time rdiff-backup --force -v5 /mnt/backup/files/ /mnt/external/backup/files/
|
||||||
|
|
||||||
|
python3 /home/tanner/scripts/checkbackup.py
|
||||||
|
|
||||||
|
umount /mnt/external
|
||||||
|
cryptsetup luksClose external
|
||||||
|
```
|
||||||
|
|
||||||
|
I wrote a Python script `checkbackup.py` that goes through each backup and
|
||||||
|
compares the timestamp in `backup_check.txt` files to the current time. This
|
||||||
|
makes sure that the cron ran, backups were taken, and transferred over
|
||||||
|
correctly.
|
||||||
|
|
||||||
|
## Rotating Hard Drives
|
||||||
|
|
||||||
|
I rotate through 2.5" 1 TB hard drives each Saturday when I do a backup. They
|
||||||
|
are quite cheap at [$65 CAD](https://www.memoryexpress.com/Products/MX65194)
|
||||||
|
each so I can have a bunch floating around.
|
||||||
|
|
||||||
|
|
||||||
|
I keep one connected to the server, one in my bag, one offsite, one at my
|
||||||
|
mother's house, and one at my dad's house. Every Saturday I run the script above
|
||||||
|
to take a copy and then swap the drive with the one in my bag. It then gets
|
||||||
|
<span class="aside">I go back home about twice per year</span>
|
||||||
|
swapped when I visit my offsite location. Same for when I visit my parents. This
|
||||||
|
means that all hard drives eventually get rotated through with new data and
|
||||||
|
don't sit too long unpowered.
|
||||||
|
|
||||||
|
The drives are all encrypted with full-disk LUKS encryption using a password I'm
|
||||||
|
unlikely to forget.
|
||||||
|
|
||||||
|
I run the check-summing `btrfs` file system on them in RAID-1 to protect against
|
||||||
|
bitrot. This means I can only use 0.5 TB of storage for my backups, but the data
|
||||||
|
is stored redundantly.
|
||||||
|
|
||||||
|
Here's how I set up new hard drives to do this:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ sudo cryptsetup luksOpen /dev/sdf external
|
||||||
|
$ sudo mkfs.btrfs -f -m dup -d dup /dev/mapper/external
|
||||||
|
$ sudo mount /dev/mapper/external /mnt/external/
|
||||||
|
$ sudo mkdir /mnt/external/backup
|
||||||
|
$ sudo chown -R tanner:tanner /mnt/external/backup
|
||||||
|
$ sudo umount /mnt/external
|
||||||
|
$ sudo cryptsetup luksClose external
|
||||||
|
```
|
||||||
|
|
||||||
|
## Future Improvements
|
||||||
|
|
||||||
|
I'm working on a system to automatically back up all my home directories to my
|
||||||
|
media server. I need this to grab Bash histories and code that's
|
||||||
|
work-in-progress. I've been burned by not having this once when a server died.
|
||||||
|
|
||||||
|
I'd like to automate backing up my phone by connecting it to a Raspberry Pi when
|
||||||
|
I go to sleep.
|
||||||
|
|
||||||
|
I need to get better at fully testing my backups by restoring them on a blank
|
||||||
|
machine.
|
|
@ -27,4 +27,7 @@ by an AC-DC converter.
|
||||||
This entire process happens quicker than half a second, so it feels instant.
|
This entire process happens quicker than half a second, so it feels instant.
|
||||||
|
|
||||||
![a rusted metalic hand]({static}/images/light-switch/light2.jpg)
|
![a rusted metalic hand]({static}/images/light-switch/light2.jpg)
|
||||||
|
|
||||||
|
<span class="aside">Black stuff's liquid electrical tape</span>
|
||||||
|
|
||||||
![a rusted metalic hand]({static}/images/light-switch/light3.jpg)
|
![a rusted metalic hand]({static}/images/light-switch/light3.jpg)
|
||||||
|
|
|
@ -2,6 +2,7 @@ Title: Choosing a Linux Flavour
|
||||||
Date: 2020-10-31
|
Date: 2020-10-31
|
||||||
Category: Writing
|
Category: Writing
|
||||||
Summary: A recommendation on which flavour of Linux to run.
|
Summary: A recommendation on which flavour of Linux to run.
|
||||||
|
Wide: true
|
||||||
|
|
||||||
[TOC]
|
[TOC]
|
||||||
|
|
||||||
|
@ -14,6 +15,7 @@ I run Debian on my computers and servers.
|
||||||
## Linux Distributions
|
## Linux Distributions
|
||||||
|
|
||||||
When people refer to the "flavour of Linux" they are talking about a Linux
|
When people refer to the "flavour of Linux" they are talking about a Linux
|
||||||
|
<span class="aside">Interjection: it's technically called GNU/Linux</span>
|
||||||
distribution (distro). It mostly describes what software is distributed in its
|
distribution (distro). It mostly describes what software is distributed in its
|
||||||
software repository.
|
software repository.
|
||||||
|
|
||||||
|
|
|
@ -8,7 +8,7 @@ It's also great for when I'm on vacation. The plant is a year old now and
|
||||||
doesn't look as good as it used to (kinda like you). So this machine is like its
|
doesn't look as good as it used to (kinda like you). So this machine is like its
|
||||||
life support.
|
life support.
|
||||||
|
|
||||||
Update: this plant died long ago.
|
<span class="aside">Update: this plant died long ago</span>
|
||||||
|
|
||||||
![the device and pump on a 2L pop bottle with a tube running to a flowerpot]({static}/images/plant-waterer/waterer1.jpg)
|
![the device and pump on a 2L pop bottle with a tube running to a flowerpot]({static}/images/plant-waterer/waterer1.jpg)
|
||||||
|
|
||||||
|
@ -36,4 +36,6 @@ Another feature was the ability to run the pump backwards. This completely
|
||||||
eliminated the siphoning problem from before. After pumping for a set duration,
|
eliminated the siphoning problem from before. After pumping for a set duration,
|
||||||
it would run backwards until the tube was cleared of water.
|
it would run backwards until the tube was cleared of water.
|
||||||
|
|
||||||
|
<span class="aside">Also dead :(</span>
|
||||||
|
|
||||||
![the new version beside a big Min aralia plant]({static}/images/plant-waterer/waterer3.jpg)
|
![the new version beside a big Min aralia plant]({static}/images/plant-waterer/waterer3.jpg)
|
||||||
|
|
|
@ -24,22 +24,41 @@
|
||||||
|
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block info %}
|
||||||
<p><a href="/">← Return to Home</a></p>
|
Tanner Collin
|
||||||
<header>
|
<p class='theme-select'>
|
||||||
<h1>{{ article.title }}</h1>
|
<a onClick="setTheme('light')">Light</a> / <a onClick="setTheme('dark')">Dark</a>
|
||||||
<div class="summary">
|
|
||||||
{{ article.summary }}
|
|
||||||
</div>
|
|
||||||
<p class="metadata">
|
|
||||||
{{ article.locale_date }}
|
|
||||||
{% if article.modified %}
|
|
||||||
— updated {{ article.locale_modified }}
|
|
||||||
{% endif %}
|
|
||||||
</p>
|
</p>
|
||||||
</header>
|
{% endblock %}
|
||||||
<hr />
|
|
||||||
<article>
|
{% block content %}
|
||||||
{{ article.content }}
|
|
||||||
</article>
|
{% if article.wide %}
|
||||||
|
<div class="content content-wide">
|
||||||
|
{% else %}
|
||||||
|
<div class="content">
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<div class="topbar">
|
||||||
|
{{ info() }}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p><a href="/">← Return to Home</a></p>
|
||||||
|
<header>
|
||||||
|
<h1>{{ article.title }}</h1>
|
||||||
|
<div class="summary">
|
||||||
|
{{ article.summary }}
|
||||||
|
</div>
|
||||||
|
<p class="metadata">
|
||||||
|
{{ article.locale_date }}
|
||||||
|
{% if article.modified %}
|
||||||
|
— updated {{ article.locale_modified }}
|
||||||
|
{% endif %}
|
||||||
|
</p>
|
||||||
|
</header>
|
||||||
|
<hr />
|
||||||
|
<article>
|
||||||
|
{{ article.content }}
|
||||||
|
</article>
|
||||||
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
|
@ -44,48 +44,30 @@
|
||||||
</noscript>
|
</noscript>
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
{% if PROD %}
|
{% if PROD %}
|
||||||
<body class="<?php echo $themeClass; ?>">
|
<body class="<?php echo $themeClass; ?>">
|
||||||
{% else %}
|
{% else %}
|
||||||
<body>
|
<body>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<div class="container">
|
|
||||||
<div class="sidebar">
|
{%- macro info() -%}
|
||||||
<img src="/theme/me.jpg" class="me" alt="A picture of me smiling" />
|
{% block info %}{% endblock %}
|
||||||
<div class="info">
|
{%- endmacro -%}
|
||||||
<p>
|
|
||||||
Tanner Collin
|
<div class="container">
|
||||||
</p>
|
<div class="sidebar">
|
||||||
<p class="contact-icons">
|
{{ info() }}
|
||||||
<a href="mailto:site@tannercollin.com" rel="noreferrer noopener"><img alt="email icon" src="/theme/mail.svg" /></a>
|
|
||||||
<a href="https://t.me/tannercollin" target="_blank" rel="noreferrer noopener"><img alt="telegram logo" src="/theme/telegram.svg" /></a>
|
|
||||||
<a href="https://github.com/tannercollin" target="_blank" rel="noreferrer noopener"><img alt="github logo" src="/theme/github.svg" /></a>
|
|
||||||
</p>
|
|
||||||
<p class='theme-select'>
|
|
||||||
<a onClick="setTheme('light')">Light</a> / <a onClick="setTheme('dark')">Dark</a>
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="content">
|
|
||||||
<div class="topbar">
|
|
||||||
<img src="/theme/me.jpg" class="me" alt="A picture of me smiling" />
|
|
||||||
<div class="info">
|
|
||||||
Tanner Collin
|
|
||||||
<p class="contact-icons">
|
|
||||||
<a href="mailto:site@tannercollin.com" rel="noreferrer noopener"><img alt="email icon" src="/theme/mail.svg" /></a>
|
|
||||||
<a href="https://t.me/tannercollin" target="_blank" rel="noreferrer noopener"><img alt="telegram logo" src="/theme/telegram.svg" /></a>
|
|
||||||
<a href="https://github.com/tannercollin" target="_blank" rel="noreferrer noopener"><img alt="github logo" src="/theme/github.svg" /></a>
|
|
||||||
</p>
|
|
||||||
<p class='theme-select'>
|
|
||||||
<a onClick="setTheme('light')">Light</a> / <a onClick="setTheme('dark')">Dark</a>
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% block content %}
|
|
||||||
{% endblock %}
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
{% endblock %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p class="copyright">
|
||||||
|
© 2012–2021 Tanner Collin
|
||||||
|
</p>
|
||||||
|
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
function setTheme(theme) {
|
function setTheme(theme) {
|
||||||
console.log('Setting theme to', theme);
|
console.log('Setting theme to', theme);
|
||||||
|
|
|
@ -7,43 +7,64 @@
|
||||||
<meta name="summary" content="The personal website of Tanner Collin." />
|
<meta name="summary" content="The personal website of Tanner Collin." />
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block info %}
|
||||||
<p>
|
<img src="/theme/me.jpg" class="me" alt="A picture of me smiling" />
|
||||||
Hi, I'm Tanner! I do firmware and web development in Calgary.
|
<div class="info">
|
||||||
</p>
|
Tanner Collin
|
||||||
|
<p class="contact-icons">
|
||||||
<h2>Contact Info</h2>
|
<a href="mailto:site2@tannercollin.com" rel="noreferrer noopener"><img alt="email icon" src="/theme/mail.svg" /></a>
|
||||||
|
<a href="https://t.me/tannercollin" target="_blank" rel="noreferrer noopener"><img alt="telegram logo" src="/theme/telegram.svg" /></a>
|
||||||
<p>
|
<a href="https://github.com/tannercollin" target="_blank" rel="noreferrer noopener"><img alt="github logo" src="/theme/github.svg" /></a>
|
||||||
Email: <a href="mailto:site@tannercollin.com">site@tannercollin.com</a> <br />
|
</p>
|
||||||
Telegram: <a href="https://t.me/tannercollin" target="_blank" rel="noreferrer noopener">@tannercollin</a>
|
<p class='theme-select'>
|
||||||
</p>
|
<a onClick="setTheme('light')">Light</a> / <a onClick="setTheme('dark')">Dark</a>
|
||||||
|
</p>
|
||||||
<h2>Resume</h2>
|
</div>
|
||||||
|
{% endblock %}
|
||||||
<ul>
|
|
||||||
<li>Firmware Engineer at <a href="https://cabanablockchain.com" target="_blank" rel="noreferrer noopener">Cabana Blockchain</a>, 2018–</li>
|
{% block content %}
|
||||||
<li>Lead Hardware Engineer at <a href="https://criticalcontrol.com/" target="_blank" rel="noreferrer noopener">Critical Control</a>, 2016–2018</li>
|
<div class="content">
|
||||||
<li>Electrical Engineer at <a href="https://www.opener.aero/" target="_blank" rel="noreferrer noopener">Opener Aero</a>, 2016–2016</li>
|
|
||||||
<li>Electrical Engineer Intern at <a href="https://www.pason.com/" target="_blank" rel="noreferrer noopener">Pason Systems</a>, 2014–2015</li>
|
<div class="topbar">
|
||||||
<li>BSc. Electrical Engineering from University of Calgary</li>
|
{{ info() }}
|
||||||
</ul>
|
</div>
|
||||||
|
|
||||||
<h2>Projects</h2>
|
<p>
|
||||||
|
Hi, I'm Tanner! I do firmware and web development in Calgary.
|
||||||
<p>
|
</p>
|
||||||
My main hobby is working on technical projects. I typically design websites or
|
|
||||||
build tools that make my life easier. Sometimes art.
|
<h2>Contact Info</h2>
|
||||||
</p>
|
|
||||||
|
<p>
|
||||||
<p>
|
Email: <a href="mailto:site2@tannercollin.com">site2@tannercollin.com</a> <br />
|
||||||
You can find my code on <a href="https://github.com/tannercollin" target="_blank" rel="noreferrer noopener">GitHub</a>.
|
Telegram: <a href="https://t.me/tannercollin" target="_blank" rel="noreferrer noopener">@tannercollin</a>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
{% for article in articles_page.object_list if article.category.name == 'Projects' %}
|
<h2>Resume</h2>
|
||||||
<h3><a href="{{ article.url }}">{{ article.title }}</a></h3>
|
|
||||||
<div class="summary">
|
<ul>
|
||||||
{{ article.summary }}
|
<li>Firmware Engineer at <a href="https://cabanablockchain.com" target="_blank" rel="noreferrer noopener">Cabana Blockchain</a>, 2018–</li>
|
||||||
</div>
|
<li>Lead Hardware Engineer at <a href="https://criticalcontrol.com/" target="_blank" rel="noreferrer noopener">Critical Control</a>, 2016–2018</li>
|
||||||
{% endfor %}
|
<li>Electrical Engineer at <a href="https://www.opener.aero/" target="_blank" rel="noreferrer noopener">Opener Aero</a>, 2016–2016</li>
|
||||||
|
<li>Electrical Engineer Intern at <a href="https://www.pason.com/" target="_blank" rel="noreferrer noopener">Pason Systems</a>, 2014–2015</li>
|
||||||
|
<li>BSc. Electrical Engineering from University of Calgary</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<h2>Projects</h2>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
My main hobby is working on technical projects. I typically design websites or
|
||||||
|
build tools that make my life easier. Sometimes art.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
You can find my code on <a href="https://github.com/tannercollin" target="_blank" rel="noreferrer noopener">GitHub</a>.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
{% for article in articles_page.object_list if article.category.name == 'Projects' %}
|
||||||
|
<h3><a href="{{ article.url }}">{{ article.title }}</a></h3>
|
||||||
|
<div class="summary">
|
||||||
|
{{ article.summary }}
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
|
@ -15,11 +15,22 @@ a {
|
||||||
pre {
|
pre {
|
||||||
font-size: 1rem;
|
font-size: 1rem;
|
||||||
padding: 1rem;
|
padding: 1rem;
|
||||||
|
overflow-x: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
:not(pre)>code {
|
||||||
|
padding: 0 2px;
|
||||||
|
font-size: 0.9rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.container {
|
.container {
|
||||||
max-width: 56rem;
|
max-width: 56rem;
|
||||||
margin: 2rem auto 0 auto;
|
margin: 2rem auto 12rem auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.copyright {
|
||||||
|
font: 1rem/1.5 Apparatus SIL,serif;
|
||||||
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.theme-select {
|
.theme-select {
|
||||||
|
@ -124,6 +135,10 @@ pre {
|
||||||
max-width: 36rem;
|
max-width: 36rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.content-wide {
|
||||||
|
max-width: none;
|
||||||
|
}
|
||||||
|
|
||||||
.content p {
|
.content p {
|
||||||
font: 1.2rem/1.5 Apparatus SIL,serif;
|
font: 1.2rem/1.5 Apparatus SIL,serif;
|
||||||
}
|
}
|
||||||
|
@ -141,6 +156,15 @@ pre {
|
||||||
height: auto;
|
height: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.content .aside {
|
||||||
|
display: inline;
|
||||||
|
float: left;
|
||||||
|
position: relative;
|
||||||
|
width: 8rem;
|
||||||
|
margin-left: -9rem;
|
||||||
|
font-size: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
.toclink:not(:hover)::after {
|
.toclink:not(:hover)::after {
|
||||||
visibility: hidden;
|
visibility: hidden;
|
||||||
}
|
}
|
||||||
|
@ -169,7 +193,7 @@ pre {
|
||||||
|
|
||||||
|
|
||||||
body {
|
body {
|
||||||
background-color: #fff;
|
background-color: #eee;
|
||||||
color: #000;
|
color: #000;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -179,11 +203,15 @@ a {
|
||||||
}
|
}
|
||||||
|
|
||||||
pre {
|
pre {
|
||||||
background-color: #eee;
|
background-color: #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
:not(pre)>code {
|
||||||
|
background-color: #ddd;
|
||||||
}
|
}
|
||||||
|
|
||||||
.toc {
|
.toc {
|
||||||
background-color: #eee;
|
background-color: #ddd;
|
||||||
}
|
}
|
||||||
|
|
||||||
.content p.metadata {
|
.content p.metadata {
|
||||||
|
@ -192,18 +220,22 @@ pre {
|
||||||
|
|
||||||
body.dark {
|
body.dark {
|
||||||
background-color: #000;
|
background-color: #000;
|
||||||
color: #fff;
|
color: #eee;
|
||||||
}
|
}
|
||||||
|
|
||||||
body.dark a {
|
body.dark a {
|
||||||
color: #fff;
|
color: #eee;
|
||||||
border-bottom: 1px solid #fff;
|
border-bottom: 1px solid #eee;
|
||||||
}
|
}
|
||||||
|
|
||||||
body.dark pre {
|
body.dark pre {
|
||||||
background-color: #222;
|
background-color: #222;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
body.dark :not(pre)>code {
|
||||||
|
background-color: #222;
|
||||||
|
}
|
||||||
|
|
||||||
body.dark .toc {
|
body.dark .toc {
|
||||||
background-color: #222;
|
background-color: #222;
|
||||||
}
|
}
|
||||||
|
@ -222,7 +254,7 @@ body.dark .contact-icons img {
|
||||||
|
|
||||||
@media (prefers-color-scheme: dark) {
|
@media (prefers-color-scheme: dark) {
|
||||||
body.light {
|
body.light {
|
||||||
background-color: #fff;
|
background-color: #eee;
|
||||||
color: #000;
|
color: #000;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -232,11 +264,15 @@ body.dark .contact-icons img {
|
||||||
}
|
}
|
||||||
|
|
||||||
body.light pre {
|
body.light pre {
|
||||||
background-color: #eee;
|
background-color: #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
body.light :not(pre)>code {
|
||||||
|
background-color: #ddd;
|
||||||
}
|
}
|
||||||
|
|
||||||
body.light .toc {
|
body.light .toc {
|
||||||
background-color: #eee;
|
background-color: #ddd;
|
||||||
}
|
}
|
||||||
|
|
||||||
body.light .content p.metadata {
|
body.light .content p.metadata {
|
||||||
|
@ -253,18 +289,22 @@ body.dark .contact-icons img {
|
||||||
|
|
||||||
body {
|
body {
|
||||||
background-color: #000;
|
background-color: #000;
|
||||||
color: #fff;
|
color: #eee;
|
||||||
}
|
}
|
||||||
|
|
||||||
a {
|
a {
|
||||||
color: #fff;
|
color: #eee;
|
||||||
border-bottom: 1px solid #fff;
|
border-bottom: 1px solid #eee;
|
||||||
}
|
}
|
||||||
|
|
||||||
pre {
|
pre {
|
||||||
background-color: #222;
|
background-color: #222;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
:not(pre)>code {
|
||||||
|
background-color: #222;
|
||||||
|
}
|
||||||
|
|
||||||
.toc {
|
.toc {
|
||||||
background-color: #222;
|
background-color: #222;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user