- EPISODE 107 #meteor - Meteor Cloud Is a Full Service Hosting Solution for Meteor Apps- with Filipe Névola 
- EPISODE 106 #rails - CompanyCam Helps Contractors Document Their Job- with Chad Wilken 
- EPISODE 105 #flask - TinyPilotKVM Lets You Remote Control Your Server from Your Browser- with Michael Lynch 
- EPISODE 104 #rails - uscreen Is a Platform That Helps Content Creators Build a Business- with Nick Savrov 
- EPISODE 103 #rails - Great Question Makes It Easy for Teams to Perform Customer Research- with PJ Murray 
- EPISODE 102 #rails - Robot Accounts AI Provides a System to Let You Categorize Invoices- with Josh Kinabrew 
- EPISODE 101 #rails - Fundraze Is a Flexible Fund Raising and Campaign Management Service- with CJ Avilla 
- EPISODE 100 #jekyll - Running in Production Is a Podcast Where Devs Chat about Tech Stacks- with Nick Janetakis 
- EPISODE 099 #rails - A Custom Electronic Medical Record System for an Ophthalmology Clinic- with Jason Swett 
- EPISODE 098 #rails - Games Directory Lets You Sync Your Games and Achievements in 1 Place- with Vlad Radulescu 
- EPISODE 097 #rails - School Bus Hero Helps School Bus Drivers and Aides Find a Job- with Dieter Lunn 
- EPISODE 096 #rails - Hitobito Helps You Manage Communities with Complex Group Hierarchies- with Matthias Viehweger 
- EPISODE 095 #koa - Submotion Helps You Manage Access Control for Your SAAS Subscriptions- with Kristian Dupont 
- EPISODE 094 #django - HorseRecords Lets You Record Everything about Your Horses- with Andy Ide 
- EPISODE 093 #rails - Politico Europe Is a Business to Business News and Data Service- with Karl Roos 
- EPISODE 092 #akka - Couchmate Is a Social Chat Platform for Viewers of Live TV- with Matt Oliver 
- EPISODE 091 #ansible - Managing 40+ Servers in a Data Center at a Medical University- with Maciej Delmanowski 
- EPISODE 090 #rails - Avo Is a Framework for Creating Ruby on Rails Admin Panels- with Adrian Marin 
- EPISODE 089 #express - Optidash Is an Image Processing and Optimization API- with Przemek Matylla 
- EPISODE 088 #fastapi - CourseMaker Is an Online Course Builder for Technical Makers- with Chris Samiullah 
- EPISODE 087 #phoenix - Statuspal Is a Service for Hosted Status Pages and Monitoring- with Eduardo Messuti 
- EPISODE 086 #node - QA Wolf Helps You Create Automated Browser Tests as You Use Your Site- with Jon Perl 
- EPISODE 085 #rails - ListenAddict Lets You Subscribe to a Person Like You Would to a Show- with David Parker 
- EPISODE 084 #phoenix - PriceTable Mixes in Sales Automation, Project Management and Invoicing- with Ege Ersoz 
- EPISODE 083 #express - SongRender Lets You Create Audio Visualizer Videos from Audio Clips- with Jake Lazaroff 
- EPISODE 082 #pylons - Dropbox Gives You Secure Access to All of Your Files- with Utsav Shah 
- EPISODE 081 #jupyter - Mito Is a JupyterLab Extension to Make Python Data Analysis Easy- with Nate Rush 
- EPISODE 080 #laravel - 10Web Is an Automated WordPress Hosting Platform- with Tigran Nazaryan 
- EPISODE 079 #golang - NanoVMs Let You Run Your Apps Faster and Safer with Unikernels- with Ian Eyberg 
- EPISODE 078 #flask - AbstractCRE Helps Extract Key Data from Real Estate Property Documents- with Cole Simmons 
- EPISODE 077 #rails - Podia Has Everything You Need to Sell Online Courses- with Jason Charnes 
- EPISODE 076 #django - Building an Internal App to Track 1,200+ VMs and Servers at REI- with Sean Callaway 
- EPISODE 075 #django - Password Space Is a Password Manager for Families- with Nick Hnatiw 
- EPISODE 074 #laravel - Pod Hunt Helps You Find Great Podcast Episodes to Listen To- with Mubashar Iqbal 
- EPISODE 073 #django - Joyful Gifts Is an Automated Gift Giving Service- with Jonathan Adly 
- EPISODE 072 - This Episode Was Removed Because It's under Review- with Anders Hovmöller 
- EPISODE 071 #rails - DNSimple Is a Simple and Secure Domain Management Service- with Anthony Eden 
- EPISODE 070 #django - Buttondown Lets You Build, Grow and Launch Your Email Newsletter- with Justin Duke 
- EPISODE 069 #django - An Internal Financial Planning Service for Season Ticket Holders- with Denis Stepanenko 
- EPISODE 068 #phoenix - TextDB Is a Simple Way to Share Small Amounts of Data- with Ian Davidson 
- EPISODE 067 #django - Muze Is a Freeform iOS Chat Application- with Dash Winterson 
- EPISODE 066 #flask - MixedCRM Is a Sales and CRM Tool for Real Estate Property Developers- with Nidal Alhariri 
- EPISODE 065 #django - CareerVillage Is a Community Where Students Can Get Career Advice- with Jared Chung 
- EPISODE 064 #phoenix - Stacker Is an Internal Tool to Stitch Together Microservices Responses- with Sascha Wolf 
- EPISODE 063 #phoenix - Changelog.com Is a News and Podcast Platform for Developers- with Jerod Santo & Gerhard Lazu 
- EPISODE 062 #express - Browserless Gives You Fast, Scalable and Reliable Browser Automation- with Joel Griffith 
- EPISODE 061 #phoenix - Monitor Oban Jobs in Real Time with Oban Web Pro- with Parker Selbert 
- EPISODE 060 #flask - Fantasy Football Data Pros Teaches You Python through Fantasy Football- with Ben Dominguez 
- EPISODE 059 #django - Creative Hire Helps Match Designers and Companies to Various Jobs- with Kshitij Sinha 
- EPISODE 058 #django - Brightpath Is a Formative School Assessment Tool- with Kye Russell 
- EPISODE 057 #django - Crowdbotics Is a React Native + Django App Builder / Deployer- with Anand Kulkarni 
- EPISODE 056 #django - Divio Is a Cloud Platform for Managing Web Applications- with Daniele Procida 
- EPISODE 055 #django - Daughterly Care Is a CRM to Help Care for the Elderly- with Barton Ip 
- EPISODE 054 #django - Paparazzi Accessories Is an E-commerce Store That Sells Jewelry- with Dan Purcell 
- EPISODE 053 #django - An Inventory Management System for an E-commerce Platform- with Galen Rice 
- EPISODE 052 #express - A Video Course Platform Built for a Course Called Modern DevTools- with Umar Hansa 
- EPISODE 051 #django - An E-commerce Site That Lets Folks Buy Online Courses- with Dalton Polhill 
- EPISODE 050 #django - Tracking Games Played on a Flight Simulator- with Paul Cappaert 
- EPISODE 049 #django - Collecting and Processing Genomic Data to Help Cure Rare Diseases- with Dan Kolbman 
- EPISODE 048 #django - Social Hiring Platform for the Food and Beverage Industry- with Colin McFaul 
- EPISODE 047 #django - A Platform to Buy Gift Cards for Eco-Friendly Products- with Antonin Grêlé 
- EPISODE 046 #flask - Custom Hardware to Provide Affordable Email to the African Congo- with Clemens Wolff & Shaun Bathgate 
- EPISODE 045 #django - Skillwell Is a Video Platform That Helps You Unlock Physical Skills- with John Debs 
- EPISODE 044 #django - An Internal Employee Management App Focused on Schools- with Chris Goodwin 
- EPISODE 043 #django - DataWellness Helps Organizations Stay Safe and Compliant- with Dave Merwin 
- EPISODE 042 #golang - Creating a Video Course Hosting Platform to Learn Go- with Jon Calhoun 
- EPISODE 041 #django - Litt NYC Lets You Easily Find Friends and Things to Do- with Harry Moreno 
- EPISODE 040 #django - A Site to View 3D Scans of Cars with Twinner- with Jesse Hunt 
- EPISODE 039 #django - Place Card Me Lets You Create Printable Place Cards Online- with Cory Zue 
- EPISODE 038 #django - The CMS Front End for Lionel Trains' Products- with Mark Miscavage 
- EPISODE 037 #django - A Real Estate Order and Appraisal System for a Small Business- with Austin Lewis 
- EPISODE 036 #django - Passiv Is a Portfolio Management and Automation Platform- with Brendan Wood 
- EPISODE 035 #static-site - Determine What Your Toilet Paper Supply Is Based on Your Usage- with Ben Sassoon 
- EPISODE 034 #django - Confectionery Connect Is an E-commerce Video Course Marketplace- with Sean Parsons 
- EPISODE 033 #django - Zego Lets You Easily Buy Insurance by the Hour- with Stuart Kelly 
- EPISODE 032 #django - Building a Site Around Thousands of Diary Entries from Samuel Pepys- with Phil Gyford 
- EPISODE 031 #phoenix - Mux Is an API Based Platform That Lets You Process and Stream Videos- with Dylan Jhaveri 
- EPISODE 030 #flask - TradeRev Is a Machine Learning Vehicle Appraisal / Auctioning System- with Amit Jain 
- EPISODE 029 #flask - Cover Tuner Uses NLP to Help Improve Your Cover Letters- with Saad Malik 
- EPISODE 028 #dotnet - Easily Find, Reproduce and Track Your JavaScript Errors with TrackJS- with Todd Gardner 
- EPISODE 027 #python - A College Professor Self Hosts Jupyter Hub for His Students- with Peter Kazarinoff 
- EPISODE 026 #flask - Contextualise Is a Tool That Lets You Easily Organize Information- with Brett Kromkamp 
- EPISODE 025 #flask - A Cryptocurrency Powered E-commerce Store Called Strmline- with Ty Cooper 
- EPISODE 024 #rails - CovidNearMe Tracks Cases and Has Info for Citizens / Health Workers- with Scott Johnson 
- EPISODE 023 #flask - Taleas Is a Cute, Quirky, Random and Slightly Weird Webcomic- with Seth Black 
- EPISODE 022 #flask - Creating a Non-profit Food Ordering Service for Inner City Kids In LA- with Francisco Barcena 
- EPISODE 021 #phoenix - Get to Know Your Coworkers through Simple QA with This Slack App- with Adam Conrad 
- EPISODE 020 #flask - A Weather Analysis Service for Regular People and Meteorologists- with Nick Gregory 
- EPISODE 019 #phoenix - Hex.pm Is Elixir's Official Package Manager- with Eric Meadows-Jönsson 
- EPISODE 018 #phoenix - 6DOS Helps You Explore Your Personal Network- with Henry Popp 
- EPISODE 017 #rails - Smart Music Helps Musicians Practice More Efficiently- with Julien Blanchard 
- EPISODE 016 #rails - VA.gov Provides an API to Get Information about Veterans- with Charley Stran 
- EPISODE 015 #express - Kernl.us Helps WordPress Plugin and Theme Developers Manage Updates- with Jack Slingerland 
- EPISODE 014 #phoenix - ScrollKeeper Is a Collaboration Tool for Researchers- with Ian Butler 
- EPISODE 013 #phoenix - Remote.com Helps You Find Remote Jobs Anywhere- with Marcelo Lebre 
- EPISODE 012 #rails - Learn Ruby on Rails through Screencast Tutorials on GoRails- with Chris Oliver 
- EPISODE 011 #phoenix - Logflare Is a Log Management and Event Analytics Platform- with Chase Granberry 
- EPISODE 010 #flask - ScholarPack Runs 10% of the UK's Primary Schools and Gets Huge Traffic- with Gareth Thomas 
- EPISODE 009 #golang - Load Balance, Secure and Observe Your Web Applications with Nova ADC- with Dave Blakey 
- EPISODE 008 #koa - Openship Is a Shopify App for Drop Shipping and Order Fulfillment- with Junaid Kabani 
- EPISODE 007 #dotnet-core - Discworld Disorganizer Is a Discworld Book Series Search Engine- with Jamie Taylor 
- EPISODE 006 #golang - Qvault Is an Open Source Tool to Manage Passwords and Crypto Keys- with Lane Wagner 
- EPISODE 005 #flask - EarlyBrd Alerts You of New Job Board Postings So You Can Bid Faster- with Stetson Blake 
- EPISODE 004 #django - Real Python Is One of the Largest Python Learning Platforms Around- with Dan Bader 
- EPISODE 003 #phoenix - Learn Elixir and Phoenix by Building Things Over at Alchemist Camp- with Mark 
- EPISODE 002 #rails - A Video Based AWS Certificate Training Platform Called Exam Pro- with Andrew Brown 
- EPISODE 001 #phoenix - Serving Medical University Employees, Health Officials and Students- with Gabriel "OvermindDL1" Robertson 
- EPISODE 000 - Preview of What's to Come
Are you running a site in production? I'd love to hear your story, become a guest.
Taleas Is a Cute, Quirky, Random and Slightly Weird Webcomic
- #flask
- #python
- #aws
- #digitalocean
- #digitalocean-spaces
- #docker
- #github-actions
- #lambda
- #lets-encrypt
- #mysql
- #nginx
- #redis
- #serverless
- #swarm
- #ubuntu
In this episode of Running in Production, Seth Black goes over building a webcomic platform with Flask and Python. It runs on a multi-node Docker Swarm cluster on DigitalOcean and has been up and running since 2010. It gets up to 20,000+ visitors a day during traffic spikes.
Seth talks about the value of using nginx to cache Flask responses to keep the site from getting a hug of death from Reddit. He also talks about using nginx as a load balancer, running Docker Swarm, maintaining a web app that has been running for 10+ years, creating a flexible and robust deployment pipeline with GitHub Actions as a solo developer and a whole lot more.
Topics Include
- 2:07 – Experiences with using Flask to build the site for the last 10+ years
- 2:43 – Traffic ranges from 20,000+ visitors a day to hundreds per month
- 4:54 – A simple 5 second cache can save your server from a huge traffic spike
- 5:34 – Motivation for using Flask after having used many different frameworks / languages
- 6:48 – It’s split into 2 services, one for the main app and one for analytics
- 7:51 – The main app is about 1,000 lines of Python code with 15 database tables
- 8:35 – Flask-Cache is used to cache the response of a few high traffic views
- 8:55 – Each service has its own git repo that is independently deployed
- 9:31 – Responses are sent out as server rendered templates using Jinja 2
- 10:30 – A custom admin dashboard was built up over the years and it was worth it
- 13:14 – Reducing page load speeds by lazy loading comic images as you scroll down
- 14:29 – Websockets are being used in the admin to auto-save content in a real-time editor
- 16:27 – MariaDB is being used as the main database and here’s why
- 18:15 – Search isn’t implemented yet but it’s the number 1 user feature request
- 19:25 – Having search allows you to get good insights on what folks are looking for
- 20:07 – Full page caching is done with the free and open source version of nginx
- 21:42 – gunicorn is being used for the Flask app server and it works fantastically
- 21:57 – Everything is inside of Docker, running in a Swarm cluster
- 22:47 – Docker Swarm was chosen over Kubernetes because Kubernetes was overkill
- 25:08 – 2 web app servers that are load balanced, 2 DB servers, Redis and a load balancer
- 25:34 – Celery isn’t being used but Redis is doing a few things
- 26:27 – Why Seth moved from AWS to DigitalOcean over the years
- 29:17 – The load balancer is on a 2 GB of memory / 50 GB SSD / 2 virtual CPUs server
- 29:36 – The app servers have 4 GB of memory / 4 virtual CPUs and a bit more SSD disk space
- 29:45 – The database server has 8 GB of memory / 4 virtual CPUs and a 128 GB SSD
- 29:53 – Everything is running smoothly and purs like a kitten with various amounts of traffic
- 30:18 – Seth uses the same database server for a few of his other sites
- 31:15 – A combo of Ubuntu 16.04 and Ubuntu 18.04 are running with intent to upgrade
- 32:04 – Seth’s upgrade process for when it comes time to upgrading Ubuntu to 20.04
- 33:50 – DigitalOcean’s load balancer wasn’t around when all of this was set up
- 35:06 – Everything was installed by hand on the servers but Seth likes Terraform and Puppet
- 36:32 – You don’t always need to upgrade to the latest tool that came out today
- 37:02 – Over time, the app was updated from Python 2.6 to 3.4+ which was painful
- 37:58 – How do you stay motivated working on a project that’s been going for 10+ years?
- 40:49 – Walking us through deploying code from development to production
- 43:14 – GitHub Actions are used for CI / CD and it’s all automated with a single git push
- 44:06 – Secrets are stored in DigitalOcean Spaces and then are built into images on demand
- 46:17 – Error reporting and logging are handled through flat file logs that get backed up
- 46:50 – Newsletters are sent through a semi-manual mail merge strategy with Google Sheets
- 48:43 – An RSS feed is also available and Pinterest provides a decent amount of traffic
- 49:45 – The database is backed up nightly and 30 days worth of backups are saved
- 50:24 – All of the comic assets are backed up on DropBox and the code is on GitHub
- 50:44 – If the servers were to get forcefully rebooted, some of them recover automatically
- 51:36 – Various DigitalOcean alarms are set up to do email notifications if things go bad
- 52:22 – Reddit is weird sometimes (the hug of death or down voted into oblivion)
- 53:11 – Using DigitalOcean and his comic reader community for determining site healthiness
- 54:55 – Best tips? Focus on doing something you’re passionate about, it really helps
- 55:39 – Picking the right tool for the job is an art form and lean on trusted tools
- 56:27 – One of Seth’s biggest mistakes was trying to build something fancy initially
- 57:34 – Another thing was chasing / building weird things that his users didn’t ask for
- 59:04 – Check out Seth’s webcomic - he’s also on GitHub and InstaGram
Links
📄 References
- https://theoatmeal.com/
- https://xkcd.com/
- https://www.nginx.com/products/nginx/caching/
- https://developers.google.com/gsuite/solutions/mail-merge
- https://rss.com/blog/how-do-rss-feeds-work/
⚙️ Tech Stack
- flask →
- python →
- aws →
- digitalocean →
- digitalocean-spaces →
- docker →
- github-actions →
- lambda →
- lets-encrypt →
- mysql →
- nginx →
- redis →
- serverless →
- swarm →
- ubuntu →
🛠 Libraries Used
Support the Show
This episode does not have a sponsor and this podcast is a labor of love. If you want to support the show, the best way to do it is to purchase one of my courses or suggest one to a friend.
- Dive into Docker is a video course that takes you from not knowing what Docker is to being able to confidently use Docker and Docker Compose for your own apps. Long gone are the days of "but it works on my machine!". A bunch of follow along labs are included.
- Build a SAAS App with Flask is a video course where we build a real world SAAS app that accepts payments, has a custom admin, includes high test coverage and goes over how to implement and apply 50+ common web app features. There's over 20+ hours of video.
Questions
-  Want to discuss this episode on Twitter? Tag @nickjanetakis or use #RunningInProduction
Mar 23, 2020
✏️ Edit on GitHub
