diff --git a/.circleci/config.yml b/.circleci/config.yml
index d83e7122e10a8a04d2bd343c4d43cfb07d45f939..3df3f5de9b082af76246486540ae4e621c5d45a8 100644
--- a/.circleci/config.yml
+++ b/.circleci/config.yml
@@ -1,13 +1,12 @@
 version: 2.1
 jobs:
-  test:
+  test-ruby27:
     docker:
       - image: cimg/ruby:2.7-node
       - image: cimg/mysql:8.0
         command: [--default-authentication-plugin=mysql_native_password]
         environment:
           MYSQL_ROOT_HOST: '%'
-          MYSQL_USER: 'root'
           MYSQL_ROOT_PASSWORD: 'root'
           MYSQL_DATABASE: 'qpixel_test'
       - image: cimg/redis:7.0
@@ -23,21 +22,157 @@ jobs:
       - checkout
       - restore_cache:
           keys:
-            - qpixel-{{ checksum "Gemfile" }}
-            - qpixel-
+            - qpixel-ruby27-{{ checksum "Gemfile.lock" }}
+            - qpixel-ruby27-
       - run:
           name: Install Bundler & gems
           command: |
             gem install bundler
             bundle install --path=~/gems
+      - run:
+          name: Clean unnecessary gems
+          command: |
+            bundle clean --force
+      - save_cache:
+          key: qpixel-ruby27-{{ checksum "Gemfile.lock" }}
+          paths:
+            - ~/gems
+      - run:
+          name: Copy key
+          command: |
+            if [ -z "$MASTER_KEY" ]; then rm config/credentials.yml.enc; else echo "$MASTER_KEY" > config/master.key; fi
+      - run:
+          name: Prepare config & database
+          environment:
+            RAILS_ENV: test
+          command: |
+            cp config/database.sample.yml config/database.yml
+            cp config/storage.sample.yml config/storage.yml
+            bundle exec rails db:create
+            bundle exec rails db:schema:load
+            bundle exec rails db:migrate
+            bundle exec rails test:prepare
+      - run:
+          name: Current revision
+          command: |
+            git rev-parse $(git rev-parse --abbrev-ref HEAD)
+      - run:
+          name: Coveralls token
+          command: |
+            if [ -z "$COVERALLS_REPO_TOKEN" ]; then echo "Skipping coveralls"; else echo "repo_token: $COVERALLS_REPO_TOKEN" > .coveralls.yml; fi
+      - run:
+          name: Test
+          command: |
+            bundle exec rails test
+      - store_test_results:
+          path: "~/qpixel/test/reports"
+  system-test-ruby27:
+    docker:
+      - image: cimg/ruby:2.7-browsers
+      - image: cimg/mysql:8.0
+        command: [--default-authentication-plugin=mysql_native_password]
+        environment:
+          MYSQL_ROOT_HOST: '%'
+          MYSQL_ROOT_PASSWORD: 'root'
+          MYSQL_DATABASE: 'qpixel_test'
+      - image: cimg/redis:7.0
+
+    working_directory: ~/qpixel
+
+    steps:
+      - run:
+          name: Install packages
+          command: |
+            sudo apt-get --allow-releaseinfo-change -qq update
+            sudo apt-get -y install git libmariadb-dev libmagickwand-dev
+      - checkout
+      - restore_cache:
+          keys:
+            - qpixel-ruby27-{{ checksum "Gemfile.lock" }}
+            - qpixel-ruby27-
+      - run:
+          name: Install Bundler & gems
+          command: |
+            gem install bundler
+            bundle install --path=~/gems
+      - run:
+          name: Clean unnecessary gems
+          command: |
+            bundle clean --force
+      - save_cache:
+          key: qpixel-ruby27-{{ checksum "Gemfile.lock" }}
+          paths:
+            - ~/gems
+      - run:
+          name: Copy key
+          command: |
+            if [ -z "$MASTER_KEY" ]; then rm config/credentials.yml.enc; else echo "$MASTER_KEY" > config/master.key; fi
+      - run:
+          name: Prepare config & database
+          environment:
+            RAILS_ENV: test
+          command: |
+            cp config/database.sample.yml config/database.yml
+            cp config/storage.sample.yml config/storage.yml
+            bundle exec rails db:create
+            bundle exec rails db:schema:load
+            bundle exec rails db:migrate
+            bundle exec rails test:prepare
+      - run:
+          name: Current revision
+          command: |
+            git rev-parse $(git rev-parse --abbrev-ref HEAD)
+      - run:
+          name: Test
+          command: |
+            bundle exec rails test:system
+      - store_test_results:
+          path: "~/qpixel/test/reports"
+      - store_artifacts:
+          path: "~/qpixel/tmp/screenshots"
+          when: on_fail
+
+  test-ruby31:
+    docker:
+      - image: cimg/ruby:3.1-node
+      - image: cimg/mysql:8.0
+        command: [ --default-authentication-plugin=mysql_native_password ]
+        environment:
+          MYSQL_ROOT_HOST: '%'
+          MYSQL_ROOT_PASSWORD: 'root'
+          MYSQL_DATABASE: 'qpixel_test'
+      - image: cimg/redis:7.0
+
+    working_directory: ~/qpixel
+
+    steps:
+      - run:
+          name: Install packages
+          command: |
+            sudo apt-get --allow-releaseinfo-change -qq update
+            sudo apt-get -y install git libmariadb-dev libmagickwand-dev
+      - checkout
+      - restore_cache:
+          keys:
+            - qpixel-ruby31-{{ checksum "Gemfile.lock" }}
+            - qpixel-ruby31-
+      - run:
+          name: Install Bundler & gems
+          command: |
+            gem install bundler
+            bundle install --path=~/gems
+      - run:
+          name: Clean unnecessary gems
+          command: |
+            bundle clean --force
       - save_cache:
-          key: qpixel-{{ checksum "Gemfile" }}
+          key: qpixel-ruby31-{{ checksum "Gemfile.lock" }}
           paths:
             - ~/gems
       - run:
           name: Copy key
           command: |
-            echo "$MASTER_KEY" > config/master.key
+            if [ -z "$MASTER_KEY" ]; then rm config/credentials.yml.enc; else echo "$MASTER_KEY" > config/master.key; fi
       - run:
           name: Prepare config & database
           environment:
@@ -56,17 +191,82 @@ jobs:
       - run:
           name: Coveralls token
           command: |
-            echo "repo_token: $COVERALLS_REPO_TOKEN" > .coveralls.yml
+            if [ -z "$COVERALLS_REPO_TOKEN" ]; then echo "Skipping coveralls"; else echo "repo_token: $COVERALLS_REPO_TOKEN" > .coveralls.yml; fi
       - run:
           name: Test
           command: |
             bundle exec rails test
       - store_test_results:
           path: "~/qpixel/test/reports"
+  system-test-ruby31:
+    docker:
+      - image: cimg/ruby:3.1-browsers
+      - image: cimg/mysql:8.0
+        command: [ --default-authentication-plugin=mysql_native_password ]
+        environment:
+          MYSQL_ROOT_HOST: '%'
+          MYSQL_ROOT_PASSWORD: 'root'
+          MYSQL_DATABASE: 'qpixel_test'
+      - image: cimg/redis:7.0
+
+    working_directory: ~/qpixel
+
+    steps:
+      - run:
+          name: Install packages
+          command: |
+            sudo apt-get --allow-releaseinfo-change -qq update
+            sudo apt-get -y install git libmariadb-dev libmagickwand-dev
+      - checkout
+      - restore_cache:
+          keys:
+            - qpixel-ruby31-{{ checksum "Gemfile.lock" }}
+            - qpixel-ruby31-
+      - run:
+          name: Install Bundler & gems
+          command: |
+            gem install bundler
+            bundle install --path=~/gems
+      - run:
+          name: Clean unnecessary gems
+          command: |
+            bundle clean --force
+      - save_cache:
+          key: qpixel-ruby31-{{ checksum "Gemfile.lock" }}
+          paths:
+            - ~/gems
+      - run:
+          name: Copy key
+          command: |
+            if [ -z "$MASTER_KEY" ]; then rm config/credentials.yml.enc; else echo "$MASTER_KEY" > config/master.key; fi
+      - run:
+          name: Prepare config & database
+          environment:
+            RAILS_ENV: test
+          command: |
+            cp config/database.sample.yml config/database.yml
+            cp config/storage.sample.yml config/storage.yml
+            bundle exec rails db:create
+            bundle exec rails db:schema:load
+            bundle exec rails db:migrate
+            bundle exec rails test:prepare
+      - run:
+          name: Current revision
+          command: |
+            git rev-parse $(git rev-parse --abbrev-ref HEAD)
+      - run:
+          name: Test
+          command: |
+            bundle exec rails test:system
+      - store_test_results:
+          path: "~/qpixel/test/reports"
+      - store_artifacts:
+          path: "~/qpixel/tmp/screenshots"
+          when: on_fail
 
   rubocop:
     docker:
-      - image: cimg/ruby:2.7-node
+      - image: cimg/ruby:3.1-node
 
     working_directory: ~/qpixel
 
@@ -79,15 +279,19 @@ jobs:
       - checkout
       - restore_cache:
           keys:
-            - qpixel-{{ checksum "Gemfile" }}
-            - qpixel-
+            - qpixel-ruby31-{{ checksum "Gemfile.lock" }}
+            - qpixel-ruby31-
       - run:
           name: Install Bundler & gems
           command: |
             gem install bundler
             bundle install --path=~/gems
+      - run:
+          name: Clean unnecessary gems
+          command: |
+            bundle clean --force
       - save_cache:
-          key: qpixel-{{ checksum "Gemfile" }}
+          key: qpixel-ruby31-{{ checksum "Gemfile.lock" }}
           paths:
             - ~/gems
       - run:
@@ -97,7 +301,7 @@ jobs:
 
   deploy:
     docker:
-      - image: cimg/ruby:2.7-node
+      - image: cimg/ruby:3.1-node
 
     working_directory: ~/qpixel
 
@@ -115,11 +319,17 @@ jobs:
 workflows:
   test_lint:
     jobs:
-      - test
+      - test-ruby27
+      - test-ruby31
+      - system-test-ruby27
+      - system-test-ruby31
       - rubocop
       - deploy:
           requires:
-            - test
+            - test-ruby27
+            - test-ruby31
+            - system-test-ruby27
+            - system-test-ruby31
             - rubocop
           filters:
             branches:
diff --git a/.github/ISSUE_TEMPLATE/bug-feature-via-meta.md b/.github/ISSUE_TEMPLATE/bug-feature-via-meta.md
new file mode 100644
index 0000000000000000000000000000000000000000..9141bee85b88e0be14ec5274d01f22921d2a0fc7
--- /dev/null
+++ b/.github/ISSUE_TEMPLATE/bug-feature-via-meta.md
@@ -0,0 +1,16 @@
+---
+name: Bug/Feature via Meta
+about: Use when you're copying a bug/feature request here from Meta.
+title: ''
+labels: ''
+assignees: ''
+
+---
+
+<!-- Meta posts are autolinked. Insert the post ID after meta: and a link will be generated for you. -->
+meta:123
+
+<!-- Provide a brief description of what the bug/feature is based on the Meta post. -->
+
+
+<!-- Add any discussion that's been had around the issue, either on Meta, Discord, or elsewhere. Include any suggested resolutions or suggestions for how we could address this. -->
diff --git a/.github/ISSUE_TEMPLATE/bug_report.md b/.github/ISSUE_TEMPLATE/bug_report.md
new file mode 100644
index 0000000000000000000000000000000000000000..3d933dccfb0615ab20589c3359ed5376a964bf8a
--- /dev/null
+++ b/.github/ISSUE_TEMPLATE/bug_report.md
@@ -0,0 +1,27 @@
+---
+name: Bug report
+about: Create a report to help us improve
+title: ''
+labels: ''
+assignees: ''
+
+---
+
+**Describe the bug**
+A clear and concise description of what the bug is.
+
+**To Reproduce**
+Steps to reproduce the behavior:
+1. Go to '...'
+2. Click on '....'
+3. Scroll down to '....'
+4. See error
+
+**Expected behavior**
+A clear and concise description of what you expected to happen.
+
+**Screenshots**
+If applicable, add screenshots to help explain your problem.
+
+**Additional context**
+Add any other context about the problem here. Include device/browser/OS information if it's relevant.
diff --git a/.github/ISSUE_TEMPLATE/feature_request.md b/.github/ISSUE_TEMPLATE/feature_request.md
new file mode 100644
index 0000000000000000000000000000000000000000..bbcbbe7d61558adde3cbfd0c7a63a67c27ed6d30
--- /dev/null
+++ b/.github/ISSUE_TEMPLATE/feature_request.md
@@ -0,0 +1,20 @@
+---
+name: Feature request
+about: Suggest an idea for this project
+title: ''
+labels: ''
+assignees: ''
+
+---
+
+**Is your feature request related to a problem? Please describe.**
+A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
+
+**Describe the solution you'd like**
+A clear and concise description of what you want to happen.
+
+**Describe alternatives you've considered**
+A clear and concise description of any alternative solutions or features you've considered.
+
+**Additional context**
+Add any other context or screenshots about the feature request here.
diff --git a/.gitignore b/.gitignore
index 004c052c28034065a537513964057539b6c3f9ce..d665bfb07a3a5d93d1c43b902d214ba456daefa8 100644
--- a/.gitignore
+++ b/.gitignore
@@ -9,13 +9,16 @@
 
 # Docker environment (production)
 docker/env
+# mount mysql volume so that its easy to interact with the database outside of the container. 
+# This also allows persistent database storage
+docker/mysql
+# allow custom docker-compose files as users might have different needs
+docker-compose*.yml
+!docker-compose.yml
 
 # Don't track changes to the docker-compose .env file only in project root
 /.env
 
-# mount mysql volume so that its easy to interact with the database outside of the container. This also allows persistent database storage
-docker/mysql
-
 # Ignore the default SQLite database.
 /db/*.sqlite3
 /db/*.sqlite3-journal
@@ -29,6 +32,7 @@ coverage/
 coverage/*
 
 .idea
+.vscode
 
 test/reports
 
@@ -51,4 +55,11 @@ qpixel-import.tar.gz
 # Ignore Vim stuff.
 *.swp
 
+# Ignore emacs stuff.
+*~
+
 dump.rdb
+
+# Ignore IRB files
+.irbrc
+.irb_history
diff --git a/.rubocop.yml b/.rubocop.yml
index 5e6705c20c0780b84a9ee94fa59cf0560d0488ac..edf12b0c084a844f74f025109c7a6805469aa882 100644
--- a/.rubocop.yml
+++ b/.rubocop.yml
@@ -4,7 +4,7 @@ require:
   - ./lib/rubocop/path_in_helpers.rb
 
 AllCops:
-  TargetRubyVersion: 2.6
+  TargetRubyVersion: 2.7
   Exclude:
     - 'config/**/*'
     - 'db/**/*'
@@ -43,13 +43,13 @@ Metrics/BlockLength:
 Metrics/BlockNesting:
   Max: 5
 Metrics/ClassLength:
-  Max: 300
+  Max: 350
 Metrics/CyclomaticComplexity:
   Max: 30
 Metrics/MethodLength:
   Max: 60
 Metrics/ModuleLength:
-  Max: 200
+  Max: 250
 Metrics/PerceivedComplexity:
   Enabled: false
 
diff --git a/.sample.irbrc b/.sample.irbrc
new file mode 100644
index 0000000000000000000000000000000000000000..998084f49a298ff40b3935cd8401bc5aa76b4274
--- /dev/null
+++ b/.sample.irbrc
@@ -0,0 +1 @@
+Qpixel.irb! if defined?(Qpixel)
diff --git a/CODE-STANDARDS.md b/CODE-STANDARDS.md
index ad30362a06a80a7ab072867d9db988c2ceca158f..35e2888a922ef5cf27b6683a963dab24870e565e 100644
--- a/CODE-STANDARDS.md
+++ b/CODE-STANDARDS.md
@@ -121,7 +121,7 @@ Pseudo-classes and pseudo-element selectors should appear *after* the main selec
 `@media` and other nested [*at-rules*](https://developer.mozilla.org/en-US/docs/Web/CSS/At-rule) should be added to
 the end of the document, preceded by an extra blank line.
 
-See [*landing-page/primary.css @1ca2f671*](https://github.com/codidact/landing-page/blob/1ca2f671/dist/assets/css/primary.css)
+See [*landing-page/dist/assets/css/primary.css @1ca2f671*](https://github.com/codidact/landing-page/blob/1ca2f671/dist/assets/css/primary.css)
 for an example of all of the above.
 
 ### Spacing
@@ -130,9 +130,9 @@ for an example of all of the above.
 - Do not write more than one statement per line.
 
 ### Line breaks
-Rules should be separated by a blank line, except for the two special cases provided in item
-[**#3**](#Order-of-selectors) - namely, an extra blank line is expected between universal selectors and other
-selectors, as well as before nested _at-rules_. As such, these rule groups should be separated by *two* spaces.
+Rules should be separated by a blank line, except for the two special cases provided in
+[Order of selectors](#order-of-selectors) - namely, an extra blank line is expected between universal selectors and
+other selectors, as well as before nested _at-rules_. As such, these rule groups should be separated by *two* spaces.
 
 All properties are written on their own line and end with a semicolon. The closing bracket must appear in its own
 line.
@@ -259,7 +259,7 @@ When referencing external resources (including those local to the domain), do no
 HTTPS access to resources if possible.
 
 Prefer retrieving resources by canonical URIs when possible, i.e. those that do not redirect upon request. Check
-with a command-line tool or a service such as [apitester.com](https://apitester.com/) to be sure.
+with a command-line tool or a service such as [apitester.org](https://apitester.org/app) to be sure.
 
 ```html
 <script type="application/javascript" src="https://codidact.org/assets/product.js"></script>
@@ -379,20 +379,23 @@ When adding an ID or class to reference an element from JavaScript, prefix the v
 
 #### `<a>`
 - If using `target="_blank"` to open links in a new tab, also include `rel="noopener noreferrer"`.
-- If a JS-enabled link is necessary (it normally shouldn't - see note below), prefer `href="#"` over `href="javascript:void(0)"` (and its equivalent `href="javascript:;"`). Please do combine this with `event.preventDefault()` in order to prevent unwanted scrolling and adding of pointless entries to the user's browsing history.
+- If a JS-enabled link is necessary (it normally shouldn't be - see note below), prefer `href="#"` over
+`href="javascript:void(0)"` (and its equivalent `href="javascript:;"`). Please do combine this with
+`event.preventDefault()` in order to prevent unwanted scrolling and adding of pointless entries to the user's browsing
+history.
 
     **Note:** Since the above directive still requires JavaScript to be enabled, the RECOMMENDED first-line
     approach is to either link to an actual page/resource that performs the same expected action, or use a
     `<button>` element styled as a link instead. The JS-enabled link (`<a>`) strategy MUST be reserved for the
     rare cases, if any, where these are not possible - and ideally, they SHOULD be added ("injected") to the page
-    using JavaScript.<sup>[\[1\]](https://stackoverflow.com/a/134957/3258851) [\[2\]](https://stackoverflow.com/a/20215524/3258851)</sup>
+    using JavaScript[^1][^2]
 
 #### `<img>`
 - Use a compressed image format or small file size where possible.
 - Make use of `<picture>`/`srcset` where possible.
 - Load images asynchronously where possible.
 
-#### <h1-6>`
+#### `<h1-6>`
 - Ensure all pages have a level 1 header (`<h1>`) that is not the website name.
 - Pages MUST NOT have more than one `<h1>` element.
 - Use headings in order; style via CSS rather than using a smaller heading level.
@@ -521,7 +524,7 @@ this.dataset = (Object.keys(data).length > 0) ? data : {};
 Note the use of parentheses around the conditional expression - it makes it more obvious at first glance that this
 is a conditional statement. **This is a requirement.**
 
-For very long or deeply indented expressions that exceed the 120-char line length limit ([item 8](#8-line-length)),
+For very long or deeply indented expressions that exceed the [120-char line length limit](#line-length),
 use the following line-break and indenting style:
 
 ```js
@@ -597,4 +600,7 @@ first and then developing features, it can be helpful to create a commit where t
 **Merges and commits to master** _must_ pass the tests every time. The master branch is considered the stable
 channel - anything on there should be suitable for production deployment. Commits should generally not be made
 directly to master - only organization and repository administrators have the ability to, and should avoid doing
-so if at all possible.
\ No newline at end of file
+so if at all possible.
+
+[^1]: [Which 'href' value should I use for JavaScript links, '#' or 'javascript:void(0)'?](https://stackoverflow.com/a/134957/3258851)
+[^2]: [Prevent href='#' link from changing the URL hash](https://stackoverflow.com/a/20215524/3258851)
diff --git a/Gemfile b/Gemfile
index f5bd290e37d02024b04504d730efff6ced73455b..b09b258fd0be445e8d937448b41d0d1193a11a69 100644
--- a/Gemfile
+++ b/Gemfile
@@ -9,18 +9,19 @@ gem 'image_processing', '~> 1.12'
 gem 'jquery-rails', '~> 4.5.0'
 gem 'mysql2', '~> 0.5.4'
 gem 'puma', '~> 5.6'
-gem 'rails', '~> 7.0.0'
-gem 'rails-html-sanitizer', '~> 1.4'
+gem 'rails', '~> 7.0.8'
+gem 'rails-html-sanitizer', '~> 1.6'
 gem 'redis', '~> 4.8'
 gem 'rotp', '~> 6.2'
 gem 'sass-rails', '~> 6.0'
-gem 'sprockets', '~> 4.1'
+gem 'sprockets', '~> 4.1.0'
 gem 'sprockets-rails', '~> 3.4', require: 'sprockets/railtie'
 gem 'terser', '~> 1.1'
 gem 'tzinfo-data', '~> 1.2022.3'
 
 # Sign in
 gem 'devise', '~> 4.8'
+gem 'devise_saml_authenticatable', '~> 1.9'
 gem 'omniauth', '~> 2.1'
 
 # Markdown support in both directions.
@@ -40,6 +41,7 @@ gem 'will_paginate-bootstrap', '~> 1.0'
 
 # AWS for S3 (image storage) and SES (emails).
 gem 'aws-sdk-s3', '~> 1.61', require: false
+gem 'aws-sdk-sns', '~> 1.72'
 gem 'aws-ses-v4', require: 'aws/ses'
 
 # Task scheduler.
@@ -69,7 +71,7 @@ gem 'net-smtp', '~> 0.3'
 gem 'ruby-progressbar', '~> 1.11'
 
 # Image generation
-gem 'rmagick'
+gem 'rmagick', '~> 5.3'
 
 # Payments. Kinda important, y'know.
 gem 'stripe', '~> 5.55'
@@ -82,6 +84,10 @@ group :test do
   gem 'minitest-ci', '~> 3.4.0'
   gem 'rails-controller-testing', '~> 1.0'
   gem 'term-ansicolor', '~> 1.7'
+
+  gem 'capybara', '~> 3.38'
+  gem 'selenium-webdriver', '~> 4.7'
+  gem 'webdrivers', '~> 5.2'
 end
 
 group :development, :test do
@@ -94,3 +100,5 @@ group :development do
   gem 'spring', '~> 4.0'
   gem 'web-console', '~> 4.2'
 end
+
+gem 'maintenance_tasks', '~> 2.1.1'
diff --git a/Gemfile.lock b/Gemfile.lock
index 4bac246fe5d118dfdf86419a81abbaa128a4c627..c97b4b4aafad38a77a0624ae6388730332979201 100644
--- a/Gemfile.lock
+++ b/Gemfile.lock
@@ -1,81 +1,81 @@
 GEM
   remote: https://rubygems.org/
   specs:
-    actioncable (7.0.4)
-      actionpack (= 7.0.4)
-      activesupport (= 7.0.4)
+    actioncable (7.0.8.7)
+      actionpack (= 7.0.8.7)
+      activesupport (= 7.0.8.7)
       nio4r (~> 2.0)
       websocket-driver (>= 0.6.1)
-    actionmailbox (7.0.4)
-      actionpack (= 7.0.4)
-      activejob (= 7.0.4)
-      activerecord (= 7.0.4)
-      activestorage (= 7.0.4)
-      activesupport (= 7.0.4)
+    actionmailbox (7.0.8.7)
+      actionpack (= 7.0.8.7)
+      activejob (= 7.0.8.7)
+      activerecord (= 7.0.8.7)
+      activestorage (= 7.0.8.7)
+      activesupport (= 7.0.8.7)
       mail (>= 2.7.1)
       net-imap
       net-pop
       net-smtp
-    actionmailer (7.0.4)
-      actionpack (= 7.0.4)
-      actionview (= 7.0.4)
-      activejob (= 7.0.4)
-      activesupport (= 7.0.4)
+    actionmailer (7.0.8.7)
+      actionpack (= 7.0.8.7)
+      actionview (= 7.0.8.7)
+      activejob (= 7.0.8.7)
+      activesupport (= 7.0.8.7)
       mail (~> 2.5, >= 2.5.4)
       net-imap
       net-pop
       net-smtp
       rails-dom-testing (~> 2.0)
-    actionpack (7.0.4)
-      actionview (= 7.0.4)
-      activesupport (= 7.0.4)
-      rack (~> 2.0, >= 2.2.0)
+    actionpack (7.0.8.7)
+      actionview (= 7.0.8.7)
+      activesupport (= 7.0.8.7)
+      rack (~> 2.0, >= 2.2.4)
       rack-test (>= 0.6.3)
       rails-dom-testing (~> 2.0)
       rails-html-sanitizer (~> 1.0, >= 1.2.0)
-    actiontext (7.0.4)
-      actionpack (= 7.0.4)
-      activerecord (= 7.0.4)
-      activestorage (= 7.0.4)
-      activesupport (= 7.0.4)
+    actiontext (7.0.8.7)
+      actionpack (= 7.0.8.7)
+      activerecord (= 7.0.8.7)
+      activestorage (= 7.0.8.7)
+      activesupport (= 7.0.8.7)
       globalid (>= 0.6.0)
       nokogiri (>= 1.8.5)
-    actionview (7.0.4)
-      activesupport (= 7.0.4)
+    actionview (7.0.8.7)
+      activesupport (= 7.0.8.7)
       builder (~> 3.1)
       erubi (~> 1.4)
       rails-dom-testing (~> 2.0)
       rails-html-sanitizer (~> 1.1, >= 1.2.0)
-    activejob (7.0.4)
-      activesupport (= 7.0.4)
+    activejob (7.0.8.7)
+      activesupport (= 7.0.8.7)
       globalid (>= 0.3.6)
-    activemodel (7.0.4)
-      activesupport (= 7.0.4)
-    activerecord (7.0.4)
-      activemodel (= 7.0.4)
-      activesupport (= 7.0.4)
-    activestorage (7.0.4)
-      actionpack (= 7.0.4)
-      activejob (= 7.0.4)
-      activerecord (= 7.0.4)
-      activesupport (= 7.0.4)
+    activemodel (7.0.8.7)
+      activesupport (= 7.0.8.7)
+    activerecord (7.0.8.7)
+      activemodel (= 7.0.8.7)
+      activesupport (= 7.0.8.7)
+    activestorage (7.0.8.7)
+      actionpack (= 7.0.8.7)
+      activejob (= 7.0.8.7)
+      activerecord (= 7.0.8.7)
+      activesupport (= 7.0.8.7)
       marcel (~> 1.0)
       mini_mime (>= 1.1.0)
-    activesupport (7.0.4)
+    activesupport (7.0.8.7)
       concurrent-ruby (~> 1.0, >= 1.0.2)
       i18n (>= 1.6, < 2)
       minitest (>= 5.1)
       tzinfo (~> 2.0)
-    addressable (2.8.1)
+    addressable (2.8.6)
       public_suffix (>= 2.0.2, < 6.0)
     ast (2.4.2)
     awesome_print (1.9.2)
-    aws-eventstream (1.2.0)
-    aws-partitions (1.628.0)
-    aws-sdk-core (3.145.0)
-      aws-eventstream (~> 1, >= 1.0.2)
-      aws-partitions (~> 1, >= 1.525.0)
-      aws-sigv4 (~> 1.1)
+    aws-eventstream (1.3.0)
+    aws-partitions (1.908.0)
+    aws-sdk-core (3.191.6)
+      aws-eventstream (~> 1, >= 1.3.0)
+      aws-partitions (~> 1, >= 1.651.0)
+      aws-sigv4 (~> 1.8)
       jmespath (~> 1, >= 1.6.1)
     aws-sdk-kms (1.58.0)
       aws-sdk-core (~> 3, >= 3.127.0)
@@ -84,17 +84,29 @@ GEM
       aws-sdk-core (~> 3, >= 3.127.0)
       aws-sdk-kms (~> 1)
       aws-sigv4 (~> 1.4)
+    aws-sdk-sns (1.72.0)
+      aws-sdk-core (~> 3, >= 3.191.0)
+      aws-sigv4 (~> 1.1)
     aws-ses-v4 (0.8.1)
       builder
       mail (> 2.2.5)
       mime-types
       xml-simple
-    aws-sigv4 (1.5.1)
+    aws-sigv4 (1.8.0)
       aws-eventstream (~> 1, >= 1.0.2)
     bcrypt (3.1.18)
     bindex (0.8.1)
-    builder (3.2.4)
+    builder (3.3.0)
     byebug (11.1.3)
+    capybara (3.38.0)
+      addressable
+      matrix
+      mini_mime (>= 0.1.3)
+      nokogiri (~> 1.8)
+      rack (>= 1.6.0)
+      rack-test (>= 0.6.3)
+      regexp_parser (>= 1.5, < 3.0)
+      xpath (~> 3.2)
     chartkick (4.2.1)
     chronic (0.10.2)
     chunky_png (1.4.0)
@@ -105,8 +117,8 @@ GEM
       coffee-script-source
       execjs
     coffee-script-source (1.12.2)
-    commonmarker (0.23.5)
-    concurrent-ruby (1.1.10)
+    commonmarker (0.23.10)
+    concurrent-ruby (1.3.4)
     counter_culture (3.2.1)
       activerecord (>= 4.2)
       activesupport (>= 4.2)
@@ -117,30 +129,33 @@ GEM
       thor (>= 0.19.4, < 2.0)
       tins (~> 1.6)
     crass (1.0.6)
-    css_parser (1.11.0)
+    css_parser (1.16.0)
       addressable
+    date (3.4.1)
     devise (4.8.1)
       bcrypt (~> 3.0)
       orm_adapter (~> 0.1)
       railties (>= 4.1.0)
       responders
       warden (~> 1.2.3)
+    devise_saml_authenticatable (1.9.0)
+      devise (> 2.0.0)
+      ruby-saml (~> 1.7)
     diffy (3.4.2)
-    digest (3.1.0)
     docile (1.4.0)
     e2mmap (0.1.0)
-    erubi (1.11.0)
+    erubi (1.13.0)
     execjs (2.8.1)
     fastimage (2.2.6)
     ffi (1.15.5)
     flamegraph (0.9.5)
-    globalid (1.0.0)
-      activesupport (>= 5.0)
+    globalid (1.2.1)
+      activesupport (>= 6.1)
     groupdate (6.1.0)
       activesupport (>= 5.2)
     hashie (5.0.0)
     htmlentities (4.3.4)
-    i18n (1.12.0)
+    i18n (1.14.6)
       concurrent-ruby (~> 1.0)
     image_processing (1.12.2)
       mini_magick (>= 4.9.5, < 5)
@@ -149,6 +164,8 @@ GEM
       actionview (>= 5.0.0)
       activesupport (>= 5.0.0)
     jmespath (1.6.1)
+    job-iteration (1.3.6)
+      activejob (>= 5.2)
     jquery-rails (4.5.0)
       rails-dom-testing (>= 1, < 3)
       railties (>= 4.2.0)
@@ -166,39 +183,44 @@ GEM
     listen (3.7.1)
       rb-fsevent (~> 0.10, >= 0.10.3)
       rb-inotify (~> 0.9, >= 0.9.10)
-    loofah (2.18.0)
+    loofah (2.23.1)
       crass (~> 1.0.2)
-      nokogiri (>= 1.5.9)
-    mail (2.7.1)
+      nokogiri (>= 1.12.0)
+    mail (2.8.1)
       mini_mime (>= 0.1.1)
-    marcel (1.0.2)
+      net-imap
+      net-pop
+      net-smtp
+    maintenance_tasks (2.1.1)
+      actionpack (>= 6.0)
+      activejob (>= 6.0)
+      activerecord (>= 6.0)
+      job-iteration (~> 1.3.6)
+      railties (>= 6.0)
+    marcel (1.0.4)
+    matrix (0.4.2)
     memory_profiler (1.0.0)
-    method_source (1.0.0)
+    method_source (1.1.0)
     mime-types (3.4.1)
       mime-types-data (~> 3.2015)
     mime-types-data (3.2022.0105)
     mini_magick (4.11.0)
-    mini_mime (1.1.2)
+    mini_mime (1.1.5)
     minitest (5.16.3)
     minitest-ci (3.4.0)
       minitest (>= 5.0.6)
     mysql2 (0.5.4)
-    net-imap (0.2.3)
-      digest
+    net-imap (0.4.19)
+      date
       net-protocol
-      strscan
-    net-pop (0.1.1)
-      digest
+    net-pop (0.1.2)
       net-protocol
+    net-protocol (0.2.2)
       timeout
-    net-protocol (0.1.3)
-      timeout
-    net-smtp (0.3.1)
-      digest
+    net-smtp (0.4.0)
       net-protocol
-      timeout
-    nio4r (2.5.8)
-    nokogiri (1.13.8-x86_64-linux)
+    nio4r (2.7.3)
+    nokogiri (1.17.1-x86_64-linux)
       racc (~> 1.4)
     omniauth (2.1.0)
       hashie (>= 3.4.6)
@@ -208,6 +230,7 @@ GEM
     parallel (1.22.1)
     parser (3.1.2.1)
       ast (~> 2.4.1)
+    pkg-config (1.5.5)
     premailer (1.16.0)
       addressable
       css_parser (>= 1.6.0)
@@ -215,49 +238,51 @@ GEM
     premailer-rails (1.11.1)
       actionmailer (>= 3)
       premailer (~> 1.7, >= 1.7.9)
-    public_suffix (5.0.0)
-    puma (5.6.5)
+    public_suffix (5.0.4)
+    puma (5.6.9)
       nio4r (~> 2.0)
-    racc (1.6.0)
-    rack (2.2.4)
+    racc (1.8.1)
+    rack (2.2.10)
     rack-mini-profiler (3.0.0)
       rack (>= 1.2.0)
     rack-protection (2.2.2)
       rack
-    rack-test (2.0.2)
+    rack-test (2.1.0)
       rack (>= 1.3)
-    rails (7.0.4)
-      actioncable (= 7.0.4)
-      actionmailbox (= 7.0.4)
-      actionmailer (= 7.0.4)
-      actionpack (= 7.0.4)
-      actiontext (= 7.0.4)
-      actionview (= 7.0.4)
-      activejob (= 7.0.4)
-      activemodel (= 7.0.4)
-      activerecord (= 7.0.4)
-      activestorage (= 7.0.4)
-      activesupport (= 7.0.4)
+    rails (7.0.8.7)
+      actioncable (= 7.0.8.7)
+      actionmailbox (= 7.0.8.7)
+      actionmailer (= 7.0.8.7)
+      actionpack (= 7.0.8.7)
+      actiontext (= 7.0.8.7)
+      actionview (= 7.0.8.7)
+      activejob (= 7.0.8.7)
+      activemodel (= 7.0.8.7)
+      activerecord (= 7.0.8.7)
+      activestorage (= 7.0.8.7)
+      activesupport (= 7.0.8.7)
       bundler (>= 1.15.0)
-      railties (= 7.0.4)
+      railties (= 7.0.8.7)
     rails-controller-testing (1.0.5)
       actionpack (>= 5.0.1.rc1)
       actionview (>= 5.0.1.rc1)
       activesupport (>= 5.0.1.rc1)
-    rails-dom-testing (2.0.3)
-      activesupport (>= 4.2.0)
+    rails-dom-testing (2.2.0)
+      activesupport (>= 5.0.0)
+      minitest
       nokogiri (>= 1.6)
-    rails-html-sanitizer (1.4.3)
-      loofah (~> 2.3)
-    railties (7.0.4)
-      actionpack (= 7.0.4)
-      activesupport (= 7.0.4)
+    rails-html-sanitizer (1.6.1)
+      loofah (~> 2.21)
+      nokogiri (>= 1.15.7, != 1.16.7, != 1.16.6, != 1.16.5, != 1.16.4, != 1.16.3, != 1.16.2, != 1.16.1, != 1.16.0.rc1, != 1.16.0)
+    railties (7.0.8.7)
+      actionpack (= 7.0.8.7)
+      activesupport (= 7.0.8.7)
       method_source
       rake (>= 12.2)
       thor (~> 1.0)
       zeitwerk (~> 2.5)
     rainbow (3.1.1)
-    rake (13.0.6)
+    rake (13.2.1)
     rb-fsevent (0.11.2)
     rb-inotify (0.10.1)
       ffi (~> 1.0)
@@ -268,8 +293,9 @@ GEM
       railties (>= 5.0)
     reverse_markdown (2.1.1)
       nokogiri
-    rexml (3.2.5)
-    rmagick (4.2.6)
+    rexml (3.3.9)
+    rmagick (5.3.0)
+      pkg-config (~> 1.4)
     rotp (6.2.0)
     rqrcode (2.1.2)
       chunky_png (~> 1.0)
@@ -292,8 +318,12 @@ GEM
       rack (>= 1.1)
       rubocop (>= 1.33.0, < 2.0)
     ruby-progressbar (1.11.0)
+    ruby-saml (1.17.0)
+      nokogiri (>= 1.13.10)
+      rexml
     ruby-vips (2.1.4)
       ffi (~> 1.12)
+    rubyzip (2.3.2)
     sass-rails (6.0.0)
       sassc-rails (~> 2.1, >= 2.1.1)
     sassc (2.4.0)
@@ -304,6 +334,10 @@ GEM
       sprockets (> 3.0)
       sprockets-rails
       tilt
+    selenium-webdriver (4.7.1)
+      rexml (~> 3.2, >= 3.2.5)
+      rubyzip (>= 1.2.2, < 3.0)
+      websocket (~> 1.0)
     simplecov (0.16.1)
       docile (~> 1.1)
       json (>= 1.8, < 3)
@@ -319,20 +353,19 @@ GEM
       sprockets (>= 3.0.0)
     stackprof (0.2.21)
     stripe (5.55.0)
-    strscan (3.0.4)
     sync (0.5.0)
     term-ansicolor (1.7.1)
       tins (~> 1.0)
     terser (1.1.12)
       execjs (>= 0.3.0, < 3)
-    thor (1.2.1)
+    thor (1.3.1)
     thwait (0.2.0)
       e2mmap
     tilt (2.0.11)
-    timeout (0.3.0)
+    timeout (0.4.3)
     tins (1.31.1)
       sync
-    tzinfo (2.0.5)
+    tzinfo (2.0.6)
       concurrent-ruby (~> 1.0)
     tzinfo-data (1.2022.3)
       tzinfo (>= 1.0.0)
@@ -344,7 +377,12 @@ GEM
       activemodel (>= 6.0.0)
       bindex (>= 0.4.0)
       railties (>= 6.0.0)
-    websocket-driver (0.7.5)
+    webdrivers (5.2.0)
+      nokogiri (~> 1.6)
+      rubyzip (>= 1.3.0)
+      selenium-webdriver (~> 4.0)
+    websocket (1.2.9)
+    websocket-driver (0.7.6)
       websocket-extensions (>= 0.1.0)
     websocket-extensions (0.1.5)
     whenever (1.0.0)
@@ -354,7 +392,9 @@ GEM
       will_paginate (>= 3.0.3)
     xml-simple (1.1.9)
       rexml
-    zeitwerk (2.6.0)
+    xpath (3.2.0)
+      nokogiri (~> 1.8)
+    zeitwerk (2.6.17)
 
 PLATFORMS
   x86_64-linux
@@ -362,14 +402,17 @@ PLATFORMS
 DEPENDENCIES
   awesome_print (~> 1.9)
   aws-sdk-s3 (~> 1.61)
+  aws-sdk-sns (~> 1.72)
   aws-ses-v4
   byebug (~> 11.1)
+  capybara (~> 3.38)
   chartkick (~> 4.2)
   coffee-rails (~> 5.0.0)
   commonmarker (~> 0.23)
   counter_culture (~> 3.2)
   coveralls (~> 0.8)
   devise (~> 4.8)
+  devise_saml_authenticatable (~> 1.9)
   diffy (~> 3.4)
   e2mmap (~> 0.1)
   fastimage (~> 2.2)
@@ -380,6 +423,7 @@ DEPENDENCIES
   jquery-rails (~> 4.5.0)
   letter_opener_web (~> 2.0)
   listen (~> 3.7)
+  maintenance_tasks (~> 2.1.1)
   memory_profiler (~> 1.0)
   minitest (~> 5.16.0)
   minitest-ci (~> 3.4.0)
@@ -389,20 +433,21 @@ DEPENDENCIES
   premailer-rails (~> 1.11)
   puma (~> 5.6)
   rack-mini-profiler (~> 3.0)
-  rails (~> 7.0.0)
+  rails (~> 7.0.8)
   rails-controller-testing (~> 1.0)
-  rails-html-sanitizer (~> 1.4)
+  rails-html-sanitizer (~> 1.6)
   redis (~> 4.8)
   reverse_markdown (~> 2.1)
-  rmagick
+  rmagick (~> 5.3)
   rotp (~> 6.2)
   rqrcode (~> 2.1)
   rubocop (~> 1)
   rubocop-rails (~> 2.15)
   ruby-progressbar (~> 1.11)
   sass-rails (~> 6.0)
+  selenium-webdriver (~> 4.7)
   spring (~> 4.0)
-  sprockets (~> 4.1)
+  sprockets (~> 4.1.0)
   sprockets-rails (~> 3.4)
   stackprof (~> 0.2)
   stripe (~> 5.55)
@@ -411,6 +456,7 @@ DEPENDENCIES
   thwait (~> 0.2)
   tzinfo-data (~> 1.2022.3)
   web-console (~> 4.2)
+  webdrivers (~> 5.2)
   whenever (~> 1.0)
   will_paginate (~> 3.3)
   will_paginate-bootstrap (~> 1.0)
@@ -419,4 +465,4 @@ RUBY VERSION
    ruby 2.7.6p219
 
 BUNDLED WITH
-   2.3.21
+   2.4.13
diff --git a/INSTALLATION.md b/INSTALLATION.md
index 93e42a159ffaa3fad53c1fa1e4f649489e6f179a..c65ebd5e2b1e6011b1dfa7d31a21a2f7b58fb23a 100644
--- a/INSTALLATION.md
+++ b/INSTALLATION.md
@@ -4,7 +4,7 @@ These instructions are for setting up a development instance of QPixel. QPixel i
 built with Ruby on Rails.
 
 In that guide it is assumed that you already have a Unix environment available
-with Ruby  and Bundler installed. WSL works as well. Windows (core) has not been tested.
+with Ruby and Bundler installed. WSL works as well. Windows (core) has not been tested.
 
 For an installation with **Docker** see the README.md in the [docker](docker) folder
 for further instructions.
@@ -13,6 +13,8 @@ If you don't already have Ruby installed, use [RVM](https://rvm.io/) or
 [rbenv](https://github.com/rbenv/rbenv#installation) to install it before following
 these instructions.
 
+QPixel is tested with Ruby 3 (and works with Ruby 2.7 as of December 2022).
+
 ## Prerequisites
 
 For Debian-Based Linux:
@@ -43,7 +45,30 @@ brew install mysql bison openssl mysql-client
 bundle config --global build.mysql2 --with-opt-dir="$(brew --prefix openssl)"
 ```
 
-QPixel requires Ruby 2.7+.
+
+## Environment
+
+The following lists environment variables provided for QPixel customization
+(this section is best-effort, please check for `ENV['<variable name>']`) in source code for the full list of available variables (for Docker-specific variables, see [Docker README](/docker/README.md)):
+
+| Name                              | Value                                                  | Required? | Default                    | Description                                                                                                    |
+| --------------------------------- | ------------------------------------------------------ | --------- | -------------------------- | -------------------------------------------------------------------------------------------------------------- |
+| `BACKTRACE`                       | `<1>`                                                  | no        | -                          | Enables backtrace for libraries (see [backtrace_silencers.rb](/config/initializers/backtrace_silencers.rb))    |
+| `BUNDLE_GEMFILE`                  |                                                        | no        |                            |                                                                                                                |
+| `CONFIRMABLE_ALLOWED_ACCESS_DAYS` | `<number>`                                             | no        | `0`                        | Sets for how long (in days) an unconfirmed account can access the instance                                     |
+| `DRIVER`                          | `<headless_chrome\|chrome\|headless_firefox\|firefox>` | no        | `headless_firefox`         | Sets browser to use when running system tests                                                                  |
+| `MAILER_PROTOCOL`                 | `http\|https`                                          | no        | `https`                    | Sets default URL protocol to use with mailes (f.e., confirmation emails)                                       |
+| `PIDFILE`                         | `<string>`                                             | no        | `tmp/pids/server.pid`      | Sets pidfile (a file where the id of a process is written to) for Puma                                         |
+| `PORT`                            | `<number>`                                             | no        | `3000`                     | Sets the port on which the server will listen for incoming requests                                            |
+| `RAILS_ENV`                       | `<development\|production\|test>`                      | no        | `development`              | Sets the environment to use (see [config/environments](/config/environments/))                                 |
+| `RAILS_MAX_THREADS`               | `<number>`                                             | no        | `5`                        | Sets the maximum number of threads from the internal thread pool to use for requests                           |
+| `RAILS_MIN_THREADS`               | `<number>`                                             | no        | `5`                        | Sets the minimum number of threads from the internal pool to use for requests                                  |
+| `RAILS_SERVE_STATIC_FILES`        | `<boolean>`                                            | no        | -                          |                                                                                                                |
+| `REDIS_URL`                       | `<string>`                                             | no        | `redis://localhost:6379/1` |                                                                                                                |
+| `SECRET_KEY_BASE`                 | `<string>`                                             | yes       | -                          | Sets the secret key for signed cookie verification (can be generated with `rake secret`, used in `production`) |
+| `SEEDS`                           | `<seeds source name>`                                  | no        | -                          | Runs only a specified set of seeds from [db/seeds](/db/seeds/)                                                 |
+| `UPDATE_POSTS`                    | `<boolean>`                                            | no        | -                          | If set to `true`, updates seeded posts when running post seeds                                                 |
+| `WEB_CONCURRENCY`                 | `<number>`                                             | no        | `2`                        |                                                                                                                |
 
 ### Install JS runtime
 
@@ -51,11 +76,27 @@ If you already have Node.JS installed, you can skip this step. If not,
 [download and install it](https://nodejs.org/en/download/) or for example
 `sudo apt install nodejs`.
 
+On Mac with homebrew, `brew install node` .
+
 ### Install Redis
 
 If you haven't already got it, [download and install Redis](https://redis.io/download)
 or for example `sudo apt install redis-server`.
 
+For mac with homebrew, `brew install redis` .
+
+### Install Imagemagick
+
+If you haven't already installed Imagemagick, you'll need to
+[install it for your system](https://imagemagick.org/script/download.php).
+
+If you install Imagemagick from APT on a Debian-based system, you may need to
+also install the `libmagickwand-dev` package.
+
+`sudo apt install libmagick++-dev` should also work.
+
+For Mac with homebrew, `brew install imagemagick` .
+
 ### Install Libvips
 
 If you haven't already installed Libvips, you'll need to [install it for
@@ -64,6 +105,8 @@ your system](https://www.libvips.org/).
 To install libvips from APT on a Debian-based system, use
 `sudo apt install libvips`
 
+For Mac with homebrew, `brew install vips` .
+
 ## Install QPixel
 
 Clone the repository and `cd` into the directory:
@@ -78,7 +121,7 @@ After downloading QPixel, you need to install all the dependencies. For that, yo
 If Ruby complains, that the Bundler hasn't been installed yet, use `gem install bundler` and
 then re-run the above command.
 
-### Setting up the Database
+### Set up the Database
 
 If you weren't asked to set the root MySQL user password during `mysql-server` installation,
 the installation is likely to be using Unix authentication instead. You'll need to sign into
@@ -95,10 +138,12 @@ Copy `config/database.sample.yml` to `config/database.yml` and fill in the corre
 username, and password for your environment. If you've followed these instructions (i.e. you
 have installed MySQL locally), the correct host is `localhost` or `127.0.0.1`.
 
-You'll also need to fill in details for the Redis connection. If you've followed these instructions,
+You will need to set the Redis connection details there too. If you've followed these instructions,
 the sample file should already contain the correct values for you, but if you've customised your
 setup you'll need to correct them.
 
+You'll also need to copy the Active Storage configuration from `config/storage.sample.yml` to `config/storage.yml`.
+
 If you are using MariaDB instead of MySQL, you will need to replace all occurrences of
 `utf8mb4_0900_ai_ci` with `utf8mb4_unicode_ci` in `db/schema.rb`.
 
@@ -109,6 +154,10 @@ Set up the database:
     rails r db/scripts/create_tags_path_view.rb
     rails db:migrate
 
+We also recommend you load the QPixel console extensions for easier development:
+
+    cp .sample.irbrc .irbrc
+
 You'll need to create a Community record and purge the Rails cache before you can seed the database.
 In a Rails console (`rails c`), run:
 
@@ -117,7 +166,7 @@ Community.create(name: 'Dev Community', host: 'localhost:3000')
 Rails.cache.clear
 ```
 
-After that you can call `rails db:seed` to fill the database with necessary seed data, such as settings, help posts and default templates.  (If you are preparing a production deployment, you might choose to edit some of the help seeds first.  See "Help Topics" at the end of this guide.)
+After that you can run `rails db:seed` to fill the database with necessary seed data, such as settings, help posts and default templates. (If you are preparing a production deployment, you might choose to edit some of the help seeds first. The "policy" topics are not included in the initial seed. See "Help Topics" at the end of this guide.)
 
     $ rails db:seed
     Category: Created 2, skipped 0
@@ -129,6 +178,9 @@ Now comes the big moment: You can start the QPixel server for the first time. Ru
 
 Open a web browser and visit your server, which should be running under [http://localhost:3000](http://localhost:3000).
 
+![img/qpixel-dev.png](img/qpixel-dev.png)
+
+
 ### Create administrator account
 
 You can create the first user account in the application through the "Sign up" route.
@@ -150,53 +202,52 @@ While being logged into your administrator account, go to [http://localhost:3000
 Review the settings (if you want; you can change them later) and click "Save and continue" to complete
 setting up the dev server.
 
-### Configure Categories
+## Create a Post
 
-Before you try to create a post we need to configure categories!
-Go to `http://localhost:3000/categories/`
+You can now create your first post. There are character requirements for the
+body and title, and you are required to add at least one tag.
 
-![img/categories.png](img/categories.png)
+![img/create-post.png](img/create-post.png)
 
- Click "edit" for each category and scroll down to see the "Tag Set" field. This
- will be empty on first setup.
+When you've met the length requirements and added a tag, the "Save Post in Q&A" button is enabled and you can click it.
 
-![img/tagset.png](img/tagset.png)
+![img/post.png](img/post.png)
 
-You will need to select a tag set for each category! For example, the Meta category can be
-associated with the "Meta" tag set, and the Q&A category can be associated with "Main"
 
-![img/tagset-selected.png](img/tagset-selected.png)
+## Optional: Single Sign On
 
-Make sure to click save for each one.<br>
-<em>Note:</em> You may need to run `rails db:seed` again.
+Please see our wiki for [detailed instructions](https://github.com/codidact/qpixel/wiki/Setting-up-SAML-Single-Sign-On) on setting up SAML Single Sign-On.
 
-## Create a Post
 
-You should then be able to create a post! There are character requirements for the
-body and title, and you are required at least one tag.
+## Optional: Required Tags
 
-![img/create-post.png](img/create-post.png)
+The special Meta tags (discussion, bug, support, feature-request) are not seeded. (We do not assume that all deployments want to manage user feedback the same way.) You can create tags directly on the Meta Tags page:
 
-And then click to "Save Post in Q&A"
+![img/meta-tags.png](img/meta-tags.png)
+
+Next, edit the Meta category settings:
+
+![img/edit-category.png](img/edit-category.png)
+
+Add the tags to the "Required tags" section:
+
+![img/required-tags.png](img/required-tags.png)
 
-![img/post.png](img/post.png)
 
 ## Optional: Help Topics
 
-If you are running a development server, you might not care a lot about what's in the help.  If you are planning to deploy a server for actual use, however, note that the seeds have some placeholder text you'll want to edit.  We have provided starting points (to be edited) for the following topics:
+If you are running a development server, you might not care a lot about what's in the help. If you are planning to deploy a server for actual use, however, note that the seeds have some placeholder text you'll want to edit. We have provided starting points (to be edited) for the following topics:
 
-- Terms of service (TOS)  
-- Code of conduct (COC)  
-- Privacy policy  
-- Spam policy  
+- Terms of service (TOS)
+- Code of conduct (COC)
+- Privacy policy
+- Spam policy
 - Global (network) FAQ
 
-The corresponding posts in db/seeds/posts have some places marked with "$EDIT" where you will probably want to insert URLs, email addresses, and the like.  We recommend reviewing all of the content in these topics.  There are two ways to edit these topics: in the source files before adding to your database, or through the UI in your running instance.
+The corresponding posts in db/seeds/posts have some places marked with "$EDIT" where you will probably want to insert URLs, email addresses, and the like. We recommend reviewing all of the content in these topics. There are two ways to edit these topics: in the source files before adding to your database, or through the UI in your running instance.
 
 If you edit the seed files, use the following command to add them to your database:
 
 `UPDATE_POSTS=true rails db:seed`
 
-You can also edit the topics in the UI.  As an administrator, you'll see an edit button on help topics when you view them, and the editor provides an option to deploy changes across your network of communities.  Administrators can update help topics in this way at any time.
-
-
+You can also edit the topics in the UI. As an administrator, you'll see an edit button on help topics when you view them, and the editor provides an option to deploy changes across your network of communities. Administrators can update help topics in this way at any time.
diff --git a/app/assets/images/favicon.ico b/app/assets/images/favicon.ico
new file mode 100644
index 0000000000000000000000000000000000000000..07c724156d1b16222e5e433721795642616256f1
Binary files /dev/null and b/app/assets/images/favicon.ico differ
diff --git a/app/assets/images/scoring_table.png b/app/assets/images/scoring_table.png
index e3f703b1b9871144fb5f3d55e9f2abfc16052e75..42c001f31b6b0856589a55f2ba6149ea03cc0cce 100644
Binary files a/app/assets/images/scoring_table.png and b/app/assets/images/scoring_table.png differ
diff --git a/app/assets/images/scoring_table.svg b/app/assets/images/scoring_table.svg
new file mode 100644
index 0000000000000000000000000000000000000000..48835ef0fcaa05613f3c52794e52e40281d9ab49
--- /dev/null
+++ b/app/assets/images/scoring_table.svg
@@ -0,0 +1 @@
+<svg width="2128" height="629" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" xml:space="preserve" overflow="hidden"><defs><clipPath id="clip0"><rect x="536" y="586" width="2128" height="629"/></clipPath><image width="2128" height="629" xlink:href="" preserveAspectRatio="none" id="img1"></image></defs><g clip-path="url(#clip0)" transform="translate(-536 -586)"><use width="100%" height="100%" xlink:href="#img1" fill="none" transform="translate(536 586)"></use></g></svg>
\ No newline at end of file
diff --git a/app/assets/javascripts/character_count.js b/app/assets/javascripts/character_count.js
index 13287284b9fc1561a52a30c2e79a7dc0bf4a3ee3..d2f827c1ef63226956f230954fedeb1f2b743fbc 100644
--- a/app/assets/javascripts/character_count.js
+++ b/app/assets/javascripts/character_count.js
@@ -1,60 +1,89 @@
 $(() => {
-  const setIcon = (el, icon) => {
+  /**
+   * @typedef {'fa-ellipsis-h'|'fa-times'|'fa-exclamation-circle'|'fa-check'} CounterIcon
+   * @typedef {'info'|'warning'|'error'|'default'} CounterState
+   * @typedef {'valid'|'invalid'} InputValidationState
+   * @typedef {'disabled'|'enabled'} SubmitButtonDisabledState
+   */
+
+  /**
+   * Sets the icon to show before the counter, if any
+   * @param {CounterIcon} icon name of the icon to show
+   */
+  const setCounterIcon = (el, icon) => {
     const icons = ['fa-ellipsis-h', 'fa-check', 'fa-exclamation-circle', 'fa-times'];
     el.removeClass(icons.join(' ')).addClass(icon);
   };
 
-  $(document).on('keyup change paste', '[data-character-count]', ev => {
+  /**
+   * Sets the counter's state
+   * @param {CounterState} state the state to set
+   */
+  const setCounterState = (el, state) => {
+    if (state === 'info') {
+      el.removeClass('has-color-yellow-700 has-color-red-500').addClass('has-color-primary');
+    }
+    else if (state === 'warning') {
+      el.removeClass('has-color-red-500 has-color-primary').addClass('has-color-yellow-700');
+    }
+    else if (state === 'error') {
+      el.removeClass('has-color-yellow-700 has-color-primary').addClass('has-color-red-500');
+    }
+    else {
+      el.removeClass('has-color-red-500 has-color-yellow-700 has-color-primary');
+    }
+  };
+
+  /**
+   * Sets the input's validation state
+   * @param {InputValidationState} state the state to set
+   */
+  const setInputValidationState = (el, state) => {
+    const isInvalid = state === 'invalid';
+    el.toggleClass('failed-validation', isInvalid);
+  };
+
+  /**
+   * Sets the submit button's disabled state
+   * @param {SubmitButtonDisabledState} state the state to set
+   */
+  const setSubmitButtonDisabledState = (el, state) => {
+    const isDisabled = state === 'disabled';
+    el.attr('disabled', isDisabled).toggleClass('is-muted', isDisabled);
+  };
+
+  $(document).on('keyup change paste', '[data-character-count]', (ev) => {
     const $tgt = $(ev.target);
     const $counter = $($tgt.attr('data-character-count'));
-    const $button = $counter.parents('form').find('input[type="submit"]');
+    const $button = $counter.parents('form').find('input[type="submit"],.js-suggested-edit-approve');
     const $count = $counter.find('.js-character-count__count');
     const $icon = $counter.find('.js-character-count__icon');
 
-    const displayAt = parseFloat($counter.attr('data-display-at'));
+    const count = $tgt.val().length;
     const max = parseInt($counter.attr('data-max'), 10);
     const min = parseInt($counter.attr('data-min'), 10);
-    const count = $tgt.val().length;
-    const text = `${count} / ${max}`;
+    const threshold = parseFloat($counter.attr('data-threshold'));
 
-    if (displayAt) {
-      if (count >= displayAt * max) {
-        $counter.removeClass('hide');
-      }
-      else {
-        $counter.addClass('hide');
-      }
-    }
+    const gtnMax = count > max;
+    const ltnMin = count < min;
+    const gteThreshold = count >= threshold * max;
 
-    if (count > max) {
-      $counter.removeClass('has-color-yellow-700 has-color-primary').addClass('has-color-red-500');
-      setIcon($icon, 'fa-times');
-      if ($button) {
-        $button.attr('disabled', true).addClass('is-muted');
-      }
-    }
-    else if (count > 0.75 * max) {
-      $counter.removeClass('has-color-red-500 has-color-primary').addClass('has-color-yellow-700');
-      setIcon($icon, 'fa-exclamation-circle');
-      if ($button) {
-        $button.attr('disabled', false).removeClass('is-muted');
-      }
-    }
-    else if (min && count < min) {
-      $counter.removeClass('has-color-yellow-700 has-color-red-500').addClass('has-color-primary');
-      setIcon($icon, 'fa-ellipsis-h');
-      if ($button) {
-        $button.attr('disabled', true).addClass('is-muted');
-      }
-      $tgt.addClass('failed-validation');
-    }
-    else {
-      $counter.removeClass('has-color-red-500 has-color-yellow-700 has-color-primary');
-      setIcon($icon, 'fa-check');
-      if ($button) {
-        $button.attr('disabled', false).removeClass('is-muted');
-      }
-      $tgt.removeClass('failed-validation');
+    const text = `${count} / ${ltnMin ? min : max}`;
+    
+    if (gtnMax || ltnMin) {
+      setCounterState($counter, 'error');
+      setCounterIcon($icon, 'fa-times');
+      setSubmitButtonDisabledState($button, 'disabled');
+      setInputValidationState($tgt, 'invalid');
+    } else if (gteThreshold) {
+      setCounterState($counter, 'warning');
+      setCounterIcon($icon, 'fa-exclamation-circle');
+      setSubmitButtonDisabledState($button, 'enabled');
+    } else {
+      setCounterState($counter, 'default');
+      setCounterIcon($icon, 'fa-check');
+      setSubmitButtonDisabledState($button, 'enabled');
+      setInputValidationState($tgt, 'valid');
     }
 
     $count.text(text);
diff --git a/app/assets/javascripts/codeblocks.js b/app/assets/javascripts/codeblocks.js
new file mode 100644
index 0000000000000000000000000000000000000000..a2f0db722a1e214bd74c1b6a869891a0d594e611
--- /dev/null
+++ b/app/assets/javascripts/codeblocks.js
@@ -0,0 +1,16 @@
+$(() => {
+  $(".post--content pre > code")
+    .parent()
+    .each(function() {
+      const content = $(this).text()
+      $(this)
+        .wrap('<div style="position:relative;"></div>')
+        .parent()
+        .prepend($('<button class="copy-button button is-muted is-outlined has-margin-2">Copy</button>')
+          .click(function () {
+            navigator.clipboard.writeText(content);
+            $(this).text('Copied!');
+            setTimeout(() => { $(this).text('Copy'); }, 2000);
+          }))
+  });
+});
\ No newline at end of file
diff --git a/app/assets/javascripts/comments.js b/app/assets/javascripts/comments.js
index 508de72d6ca2e88ec2fa13c0be37c4344e71fa2d..b2af26bf78f71ee12acd2d0eeb5dcd365cd3aefd 100644
--- a/app/assets/javascripts/comments.js
+++ b/app/assets/javascripts/comments.js
@@ -77,7 +77,10 @@ $(() => {
     const $tgt = $(evt.target);
     const $comment = $tgt.parents('.comment');
     const $commentBody = $comment.find('.comment--body');
+    const $thread = $comment.parents('.thread');
     const commentId = $comment.attr('data-id');
+    const postId = $thread.attr('data-post');
+    const threadId = $thread.attr('data-thread');
     const originalComment = $commentBody.clone();
 
     const resp = await fetch(`/comments/${commentId}`, {
@@ -89,7 +92,7 @@ $(() => {
 
     const formTemplate = `<form action="/comments/${commentId}/edit" method="POST" class="comment-edit-form" data-remote="true">
       <label for="comment-content" class="form-element">Comment body:</label>
-      <textarea id="comment-content" rows="6" class="form-element is-small" data-character-count=".js-character-count-comment-body" name="comment[content]">${content}</textarea>
+      <textarea id="comment-content" rows="6" class="form-element is-small" data-thread="${threadId}" data-post="${postId}" data-character-count=".js-character-count-comment-body" name="comment[content]">${content}</textarea>
       <input type="submit" class="button is-muted is-filled" value="Update comment" />
       <input type="button" name="js-discard-edit" data-comment-id="${commentId}" value="Discard Edit" class="button is-danger is-outlined js-discard-edit" />
       <span class="has-float-right has-font-size-caption js-character-count-comment-body"
@@ -101,6 +104,8 @@ $(() => {
 
     $commentBody.html(formTemplate);
 
+    $commentBody.find(`#comment-content`).on('keyup', pingable_popup);
+
     $(`.js-discard-edit[data-comment-id="${commentId}"]`).click(() => {
       $commentBody.html(originalComment.html());
     });
@@ -195,7 +200,9 @@ $(() => {
   });
 
   const pingable = {};
-  $(document).on('keyup', '.js-comment-field', async ev => {
+  $(document).on('keyup', '.js-comment-field', pingable_popup);
+
+  async function pingable_popup(ev) {
     if (QPixel.Popup.isSpecialKey(ev.keyCode)) {
       return;
     }
@@ -242,7 +249,7 @@ $(() => {
     else {
       QPixel.Popup.destroyAll();
     }
-  });
+  }
 
   $('.js-new-thread-link').on('click', async ev => {
     ev.preventDefault();
diff --git a/app/assets/javascripts/embed.js b/app/assets/javascripts/embed.js
index f8a6547e1114bb0e28fbd63a4b9ac9c703d6ecf4..5797df56b5eec26c03946ec4f5f86dd36e1755d8 100644
--- a/app/assets/javascripts/embed.js
+++ b/app/assets/javascripts/embed.js
@@ -10,9 +10,11 @@ $(() => {
     // Only embed raw YT links, i.e. not [text](link), only [link](link)
     if ((href.startsWith('https://youtube.com') || href.startsWith('https://www.youtube.com')) && $tgt.text() === href) {
       const videoId = /v=([^$&]+)/.exec(href);
-      $tgt.after(`<iframe width="100%" height="380" src="https://www.youtube-nocookie.com/embed/${videoId[1]}" frameborder="0" allowfullscreen
-                          allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture"></iframe>`);
-      $tgt.remove();
+      if (videoId) {
+        $tgt.after(`<iframe width="100%" height="380" src="https://www.youtube-nocookie.com/embed/${videoId[1]}" frameborder="0" allowfullscreen
+                            allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture"></iframe>`);
+        $tgt.remove();
+      }
     }
 
     // Likewise, only raw Spotify links
diff --git a/app/assets/javascripts/filters.js b/app/assets/javascripts/filters.js
new file mode 100644
index 0000000000000000000000000000000000000000..b8b8d63e1145a59dd133b7196992738c805ff196
--- /dev/null
+++ b/app/assets/javascripts/filters.js
@@ -0,0 +1,148 @@
+$(() => {
+  $('.js-filter-select').each(async (_, el) => {
+    const $select = $(el);
+    const $form = $select.closest('form');
+    const $formFilters = $form.find('.form--filter');
+    const $saveButton = $form.find('.filter-save');
+    const $isDefaultCheckbox = $form.find('.filter-is-default');
+    const categoryId = $isDefaultCheckbox.val();
+    let defaultFilter = await QPixel.defaultFilter(categoryId);
+    const $deleteButton = $form.find('.filter-delete');
+
+    // Enables/Disables Save & Delete buttons programatically
+    async function computeEnables() {
+      const filters = await QPixel.filters();
+      const filterName = $select.val();
+
+      // Nothing set
+      if (!filterName) {
+        $saveButton.prop('disabled', true);
+        $deleteButton.prop('disabled', true);
+        return;
+      }
+
+      const filter = filters[filterName]
+
+      // New filter
+      if (!filter) {
+        $saveButton.prop('disabled', false);
+        $deleteButton.prop('disabled', true);
+        return;
+      }
+
+      // Not a new filter
+      $deleteButton.prop('disabled', filter.system);
+
+      const hasChanges = [...$formFilters].some(el => {
+        const filterValue = filter[el.dataset.name];
+        let elValue = $(el).val();
+        if (filterValue?.constructor == Array) {
+          elValue = elValue ?? [];
+          return filterValue.length != elValue.length || filterValue.some((v, i) => v[1] != elValue[i]);
+        }
+        else {
+          return filterValue ? filterValue != elValue : elValue;
+        }
+      });
+      const defaultStatusChanged = $isDefaultCheckbox.prop('checked') != (defaultFilter === $select.val());
+      $saveButton.prop('disabled', !defaultStatusChanged && (filter.system || !hasChanges));
+    }
+
+    async function initializeSelect() {
+      defaultFilter = await QPixel.defaultFilter(categoryId);
+      $isDefaultCheckbox.prop('checked', defaultFilter === $select.val());
+      const filters = await QPixel.filters();
+
+      function template(option) {
+        if (option.id == '') { return 'Default'; }
+
+        const filter = filters[option.id];
+        const name = `<span>${option.text}</span>`;
+        const systemIndicator = filter?.system
+          ? ' <span has-font-size-caption">(System)</span>'
+          : '';
+        const newIndicator = !filter
+          ? ' <span has-font-size-caption">(New)</span>'
+          : '';
+        return $(name + systemIndicator + newIndicator);
+      }
+
+      // Clear out any old options
+      $select.children().filter((_, option) => option.value && !filters[option.value]).detach();
+      $select.select2({
+        data: Object.keys(filters),
+        tags: true,
+
+        templateResult: template,
+        templateSelection: template
+      }).on('select2:select', async evt => {
+        const filterName = evt.params.data.id;
+        const preset = filters[filterName];
+
+        $isDefaultCheckbox.prop('checked', defaultFilter === $select.val());
+        computeEnables();
+
+        // Name is not one of the presets, i.e user is creating a new preset
+        if (!preset) {
+          return;
+        }
+
+        for (const [name, value] of Object.entries(preset)) {
+          const $el = $form.find(`.form--filter[data-name=${name}]`);
+          if (value?.constructor == Array) {
+            $el.val(null);
+            for (const val of value) {
+              $el.append(new Option(val[0], val[1], false, true));
+            }
+            $el.trigger('change');
+          } else {
+            $el.val(value).trigger('change');
+          }
+        }
+      });
+      computeEnables();
+    }
+
+    initializeSelect();
+
+    // Enable saving when the filter is changed
+    $formFilters.on('change', computeEnables);
+    $isDefaultCheckbox.on('change', computeEnables);
+
+    async function saveFilter() {
+      if (!$form[0].reportValidity()) { return; }
+
+      const filter = {};
+
+      for (const el of $formFilters) {
+        filter[el.dataset.name] = $(el).val();
+      }
+
+      await QPixel.setFilter($select.val(), filter, categoryId, $isDefaultCheckbox.prop('checked'));
+      defaultFilter = await QPixel.defaultFilter(categoryId);
+
+      // Reinitialize to get new options
+      await initializeSelect();
+    }
+
+    $saveButton.on('click', saveFilter);
+
+    function clear() {
+      $select.val(null).trigger('change');
+      $form.find('.form--filter').val(null).trigger('change');
+      $isDefaultCheckbox.prop('checked', false);
+      computeEnables();
+    }
+
+    $deleteButton?.on('click', async evt => {
+      if (confirm(`Are you sure you want to delete ${$select.val()}?`)) {
+        await QPixel.deleteFilter($select.val());
+        // Reinitialize to get new options
+        await initializeSelect();
+        clear();
+      }
+    });
+
+    $form.find('.filter-clear').on('click', clear);
+  });
+});
\ No newline at end of file
diff --git a/app/assets/javascripts/flags.js b/app/assets/javascripts/flags.js
index 68ff6e14e3c20a378e034790707018bd6b99e2c8..dba93f7ec1d0a977a4bcbdc015174eb287410063 100644
--- a/app/assets/javascripts/flags.js
+++ b/app/assets/javascripts/flags.js
@@ -28,9 +28,9 @@ $(() => {
       'reason': $(`#flag-post-${postId}`).val()
     };
 
-    if (requiresDetails && data['reason'].length < 15) {
+    if (requiresDetails && data['reason'].length < 1) {
       QPixel.createNotification('danger',
-                                'Details are required for this flag type - please enter at least 15 characters.');
+                                'Details are required for this flag type - please enter a message.');
       return;
     }
 
@@ -50,7 +50,7 @@ $(() => {
           const messages = {
             comment: `<strong>Thanks!</strong> Your flag has been added as a comment for the author to review.`
           };
-          const defaultMessage = `<strong>Thanks!</strong> A moderator will review your flag.`;
+          const defaultMessage = `<strong>Thanks!</strong> We will review your flag.`;
           QPixel.createNotification('success', messages[responseType] || defaultMessage);
           $(`#flag-post-${postId}`).val('');
         }
diff --git a/app/assets/javascripts/keyboard_tools.js b/app/assets/javascripts/keyboard_tools.js
index 3ae4034cea8b14c70dd79cb516a15e9cd29751d8..5e7d08e99ac5e6ea5a4ba1d3a617b6ca4c665d32 100644
--- a/app/assets/javascripts/keyboard_tools.js
+++ b/app/assets/javascripts/keyboard_tools.js
@@ -77,8 +77,27 @@ $(() => {
       }
     });
 
+    /**
+     * Checks common modifier states on a given keyboard event
+     * @param {KeyboardEvent} e
+     * @returns {boolean}
+     */
+    const getModifierState = (e) => {
+      return !!e.altKey || !!e.ctrlKey || !!e.metaKey || !!e.shiftKey;
+    };
+
+    /**
+     * Handles the "home" keyboard state
+     * @param {KeyboardEvent} e
+     */
     function homeMenu(e) {
-      if (e.key === "?") {
+      const isHelp = e.key === "?";
+
+      if (!isHelp && getModifierState(e)) {
+        return;
+      }
+
+      if (isHelp) {
         _CodidactKeyboard.dialog(
           'Codidact Keyboard Shortcuts\n' +
           '===========================\n' +
@@ -156,7 +175,15 @@ $(() => {
       }
     }
 
+    /**
+     * Handles "goto" keyboard state
+     * @param {KeyboardEvent} e
+     */
     function gotoMenu(e) {
+      if (getModifierState(e)) {
+        return;
+      }
+
       if (e.key === 'm') {
         window.location.href = '/';
       } else if (e.key === 'u') {
@@ -208,7 +235,15 @@ $(() => {
       }
     }
 
+    /**
+     * Handles the "goto/category" keyboard state
+     * @param {KeyboardEvent} e
+     */
     function categoryMenu(e) {
+      if (getModifierState(e)) {
+        return;
+      }
+
       const number = parseInt(e.key);
       if (!isNaN(number)) {
         const data = _CodidactKeyboard.categories();
@@ -219,7 +254,15 @@ $(() => {
       }
     }
 
+    /**
+     * Handles the "goto/category-tags" keyboard state
+     * @param {KeyboardEvent} e
+     */
     function categoryTagsMenu(e) {
+      if (getModifierState(e)) {
+        return;
+      }
+
       const number = parseInt(e.key);
       if (!isNaN(number)) {
         const data = Object.entries(_CodidactKeyboard.categories());
@@ -229,7 +272,15 @@ $(() => {
       }
     }
 
+    /**
+     * Handles the "goto/category-edits" keyboard state
+     * @param {KeyboardEvent} e
+     */
     function categorySuggestedEditsMenu(e) {
+      if (getModifierState(e)) {
+        return;
+      }
+
       const number = parseInt(e.key);
       if (!isNaN(number)) {
         const data = Object.entries(_CodidactKeyboard.categories());
@@ -239,7 +290,15 @@ $(() => {
       }
     }
 
+    /**
+     * Handles the "tools" keyboard state
+     * @param {KeyboardEvent} e
+     */
     function toolsMenu(e) {
+      if (getModifierState(e)) {
+        return;
+      }
+
       if (e.key === 'e') {
         window.location.href = $(_CodidactKeyboard.selectedItem).find('.tools--item i.fa.fa-pencil-alt').parent().attr("href");
       } else if (e.key === 'h') {
@@ -276,7 +335,15 @@ $(() => {
 
     }
 
+    /**
+     * Handles the "tools/vote" keyboard state
+     * @param {KeyboardEvent} e
+     */
     function voteMenu(e) {
+      if (getModifierState(e)) {
+        return;
+      }
+
       if (e.key === 'u') {
         const cl = $(_CodidactKeyboard.selectedItem).find('.vote-button[data-vote-type="1"]');
         cl.click();
diff --git a/app/assets/javascripts/markdown.js b/app/assets/javascripts/markdown.js
index 1166a69767bd968ea2fd2645136b1a5df2d08615..c3173ec907d22d7e57e937673a5cfdbdb5dc362e 100644
--- a/app/assets/javascripts/markdown.js
+++ b/app/assets/javascripts/markdown.js
@@ -30,7 +30,8 @@ $(() => {
       numbered: ['\n 1. ', null],
       heading: ['\n# ', null],
       hr: ['\n\n-----\n\n', null],
-      table: ['\n\n| Title1 | Title2 |\n|- | - |\n| row1_1 | row1_2 |\n\n', null]
+      table: ['\n\n| Title1 | Title2 |\n|- | - |\n| row1_1 | row1_2 |\n\n', null],
+      mathjax: ['$', '$']
     };
 
     if (Object.keys(actions).indexOf(action) !== -1) {
@@ -117,7 +118,7 @@ $(() => {
 
   QPixel.addPrePostValidation(text => {
     // This regex catches Markdown images with no or default alt text.
-    const altRegex = /!\[(?:Image alt text)?\](?:\(.+(?!\\\))\)|\[.+(?!\\\])\])/gi;
+    const altRegex = /!\[(?:Image_alt_text)?\](?:\(.+(?!\\\))\)|\[.+(?!\\\])\])/gi;
     if (text.match(altRegex)) {
       const message = `It looks like you're posting an image with no alt text. Alt text is important for ` +
                       `accessibility. Consider adding alt text to the images in your post - ` +
diff --git a/app/assets/javascripts/notifications.js b/app/assets/javascripts/notifications.js
index 269a3ff5953268488ccea9c093fc132acca0ce51..db8c6dec2c71de54e5ea8cd77163f37a7c62b681 100644
--- a/app/assets/javascripts/notifications.js
+++ b/app/assets/javascripts/notifications.js
@@ -8,7 +8,7 @@ $(() => {
                 <span data-livestamp="${notification.created_at}">${notification.created_at}</span>
             </div>
             <p><a href="${notification.link}" data-id="${notification.id}"
-                  class="h-fw-bold is-not-underlined ${notification.is_read ? 'read' : ''}">${notification.content}</a></p>
+                  class="h-fw-bold is-not-underlined ${notification.is_read ? 'read' : ''} notification-link">${notification.content}</a></p>
             <p class="has-font-size-caption"><a href="javascript:void(0)" data-notif-id="${notification.id}" class="js-notification-toggle">
                 <i class="fas fa-${notification.is_read ? 'envelope' : 'envelope-open'}"></i>
                 mark ${notification.is_read ? 'unread' : 'read'}
@@ -67,8 +67,6 @@ $(() => {
         const item = $(makeNotification(notification));
         $inboxContainer.append(item);
       });
-  
-      $inboxContainer.append(`<a href="/users/me/notifications" class="button is-muted is-small">See all your notifications &raquo;</a>`);
     }
   });
 
@@ -122,4 +120,8 @@ $(() => {
     const change = data.notification.is_read ? -1 : +1;
     changeInboxCount(change);
   });
+
+  $(document).on('click', '.notification-link', async ev => {
+    $(ev.target).parents('.inbox').removeClass('is-active');
+  });
 });
\ No newline at end of file
diff --git a/app/assets/javascripts/post_histories.js b/app/assets/javascripts/post_histories.js
new file mode 100644
index 0000000000000000000000000000000000000000..4d8c7d9855a7af12b1b8d0d68101f59ae4e73b50
--- /dev/null
+++ b/app/assets/javascripts/post_histories.js
@@ -0,0 +1,9 @@
+$(() => {
+  const openRelevantEditOnly = () => {
+    $("details.history-event").prop('open', false);
+    $(location.hash).prop('open', true);
+  }
+
+  window.addEventListener("hashchange", openRelevantEditOnly);
+  openRelevantEditOnly();
+});
diff --git a/app/assets/javascripts/posts.js b/app/assets/javascripts/posts.js
index 2bdc896ad4b9e20453fdb9f0e6ad62c8a96aa2f1..2b0210eede719be8b0147aee5a43f85ff57e7097 100644
--- a/app/assets/javascripts/posts.js
+++ b/app/assets/javascripts/posts.js
@@ -4,15 +4,30 @@ const ALLOWED_TAGS = ['a', 'p', 'span', 'b', 'i', 'em', 'strong', 'hr', 'h1', 'h
   'summary', 'ins', 'table', 'thead', 'tbody', 'tr', 'th', 'td', 's'];
 const ALLOWED_ATTR = ['id', 'class', 'href', 'title', 'src', 'height', 'width', 'alt', 'rowspan', 'colspan', 'lang',
   'start', 'dir'];
+// this is a list of constructors to ignore even if they are removed by sanitizer (mostly comments & body)
+const IGNORE_UNSUPPORTED = [Comment, HTMLBodyElement];
 
 $(() => {
+  DOMPurify.addHook("uponSanitizeAttribute", (node, event) => {
+    const rowspan = node.getAttribute("rowspan");
+    const colspan = node.getAttribute("colspan");
+
+    if (rowspan && Number.isNaN(+rowspan)) {
+      event.keepAttr = false;
+    }
+
+    if (colspan && Number.isNaN(+colspan)) {
+      event.keepAttr = false;
+    }
+  });
+
   const $uploadForm = $('.js-upload-form');
 
   const stringInsert = (str, idx, insert) => str.slice(0, idx) + insert + str.slice(idx);
 
   const placeholder = "![Uploading, please wait...]()";
 
-  $uploadForm.find('input[type="file"]').on('change', async evt => {
+  $uploadForm.find('input[type="file"]').on('change', async (evt) => {
     const $postField = $('.js-post-field');
     const postText = $postField.val();
     const cursorPos = $postField[0].selectionStart;
@@ -24,7 +39,7 @@ $(() => {
     $form.submit();
   });
 
-  $uploadForm.on('submit', async evt => {
+  $uploadForm.on('submit', async (evt) => {
     evt.preventDefault();
 
     const $tgt = $(evt.target);
@@ -63,7 +78,7 @@ $(() => {
 
     const $postField = $('.js-post-field');
     const postText = $postField.val();
-    $postField.val(postText.replace(placeholder, `![Image alt text](${data.link})`));
+    $postField.val(postText.replace(placeholder, `![Image_alt_text](${data.link})`));
     $tgt.parents('.modal').removeClass('is-active');
   });
 
@@ -80,7 +95,24 @@ $(() => {
     tags: true
   });
 
-  const saveDraft = async (postText, $field, manual = false) => {
+  /**
+   * @typedef {{
+   *  body: string
+   *  comment?: string
+   *  excerpt?: string
+   *  license?: string
+   *  tag_name?: string
+   *  tags?: string[]
+   *  title?: string
+   * }} PostDraft
+   * 
+   * Attempts to save a post draft
+   * @param {PostDraft} draft post draft
+   * @param {JQuery<Element>} $field body input element
+   * @param {boolean} [manual] whether manual draft saving is enabled
+   * @returns {Promise<void>}
+   */
+  const saveDraft = async (draft, $field, manual = false) => {
     const autosavePref = await QPixel.preference('autosave', true);
     if (autosavePref !== 'on' && !manual) {
       return;
@@ -93,23 +125,63 @@ $(() => {
         'X-CSRF-Token': QPixel.csrfToken(),
         'Content-Type': 'application/json'
       },
-      body: JSON.stringify({
-        post: postText,
-        path: location.pathname
-      })
+      body: JSON.stringify({ ...draft, path: location.pathname })
     });
+
     if (resp.status === 200) {
-      const $el = $(`<span>&middot; <span class="has-color-green-600">draft saved</span></span>`);
-      $field.parents('.widget').find('.js-post-field-footer').append($el);
-      $el.fadeOut(1500, function () { $(this).remove() });
+      const $statusEl = $field.parents('.widget').find('.js-post-draft-status');
+
+      $statusEl.removeClass('transparent');
+
+      setTimeout(() => {
+        $statusEl.addClass('transparent');
+      }, 1500);
     }
   };
 
-  $('.js-save-draft').on('click', async ev => {
-    const $tgt = $(ev.target);
-    const $field = $tgt.parents('.widget').find('.js-post-field');
-    const postText = $field.val();
-    await saveDraft(postText, $field, true);
+  /**
+   * Extracts draft info from a given target
+   * @param {EventTarget} target post input field or "save draft" button
+   * @returns {{ draft: PostDraft, field: any }}
+   */
+  const parseDraft = (target) => {
+    const $tgt = $(target);
+    const $form = $tgt.parents('form');
+
+    const $bodyField = $form.find('.js-post-field');
+    const $licenseField = $form.find('.js-license-select');
+    const $excerptField = $form.find('.js-tag-excerpt');
+    
+    const $tagsField = $form.find('#post_tags_cache');
+    const $titleField = $form.find('#post_title');
+    const $commentField = $form.find('#edit_comment');
+    const $tagNameField = $form.find('#tag_name');
+
+    const bodyText = $bodyField.val();
+    const commentText = $commentField.val();
+    const excerptText = $excerptField.val();
+    const license = $licenseField.val();
+    const tags = $tagsField.val();
+    const titleText = $titleField.val();
+    const tagName = $tagNameField.val();
+
+    /** @type {PostDraft} */
+    const draft = {
+      body: bodyText,
+      comment: commentText,
+      excerpt: excerptText,
+      license: license,
+      tags: tags,
+      tag_name: tagName,
+      title: titleText,
+    };
+
+    return { draft, field: $bodyField };
+  };
+
+  $('.js-save-draft').on('click', async (ev) => {
+    const { draft, field } = parseDraft(ev.target);
+    await saveDraft(draft, field, true);
   });
 
   let featureTimeout = null;
@@ -117,7 +189,27 @@ $(() => {
 
   const postFields = $('.post-field');
 
-  postFields.on('paste', async evt => {
+  const draftFieldsSelectors = [
+    '.js-post-field',
+    '.js-license-select',
+    '.js-tag-excerpt',
+    '#edit_comment',
+    '#post_tags_cache',
+    '#post_title',
+    '#tag_parent_id',
+    '#tag_name',
+  ];
+
+  // TODO: consider merging with post fields
+  $(draftFieldsSelectors.join(', ')).on('keyup change', (ev) => {
+    clearTimeout(draftTimeout);
+    draftTimeout = setTimeout(() => {
+      const { draft, field } = parseDraft(ev.target);
+      saveDraft(draft, field);
+    }, 1000);
+  });
+
+  postFields.on('paste', async (evt) => {
     if (evt.originalEvent.clipboardData.files.length > 0) {
       const $fileInput = $uploadForm.find('input[type="file"]');
       $fileInput[0].files = evt.originalEvent.clipboardData.files;
@@ -125,52 +217,71 @@ $(() => {
     }
   });
 
-  postFields.on('focus keyup paste change markdown', evt => {
-    const $tgt = $(evt.target);
-
-    if (!window.converter) {
-      window.converter = window.markdownit({
-        html: true,
-        breaks: false,
-        linkify: true
-      });
-      window.converter.use(window.markdownitFootnote);
-      window.converter.use(window.latexEscape);
-    }
-    window.setTimeout(() => {
-      const converter = window.converter;
+  postFields.on('focus keyup paste change markdown', (() => {
+    let previous = null;
+    return evt => {
+      const $tgt = $(evt.target);
       const text = $(evt.target).val();
-      const unsafe_html = converter.render(text);
-      const html = DOMPurify.sanitize(unsafe_html, {
-        USE_PROFILES: { html: true },
-        ALLOWED_TAGS,
-        ALLOWED_ATTR
-      });
-      $tgt.parents('.form-group').siblings('.post-preview').html(html);
-      $tgt.parents('form').find('.js-post-html[name="__html"]').val(html + '<!-- g: js, mdit -->');
-    }, 0);
-
-    if (featureTimeout) {
-      clearTimeout(featureTimeout);
-    }
-
-    featureTimeout = setTimeout(() => {
-      if (window['MathJax']) {
-        MathJax.typeset();
+      // Don't bother re-rendering if nothing's changed
+      if (text === previous) { return; }
+      previous = text;
+      if (!window.converter) {
+        window.converter = window.markdownit({
+          html: true,
+          breaks: false,
+          linkify: true
+        });
+        window.converter.use(window.markdownitFootnote);
+        window.converter.use(window.latexEscape);
       }
-      if (window['hljs']) {
-        hljs.highlightAll();
+      window.setTimeout(() => {
+        const converter = window.converter;
+        const unsafe_html = converter.render(text);
+        const html = DOMPurify.sanitize(unsafe_html, {
+          ALLOWED_TAGS,
+          ALLOWED_ATTR
+        });
+
+        const removedElements = [...new Set(DOMPurify.removed
+          .filter(entry => entry.element && !IGNORE_UNSUPPORTED.some((ctor) => entry.element instanceof ctor))
+          .map(entry => entry.element.localName))];
+
+        const removedAttributes = [...new Set(DOMPurify.removed
+          .filter(entry => entry.attribute)
+          .map(entry => [
+            entry.attribute.name + (entry.attribute.value ? `='${entry.attribute.value}'` : ''),
+            entry.from.localName
+          ]))]
+
+        $tgt.parents('form')
+          .find('.rejected-elements')
+          .toggleClass('hide', removedElements.length === 0 && removedAttributes.length === 0)
+          .find('ul')
+          .empty()
+          .append(
+            removedElements.map(name => $(`<li><code>&lt;${name}&gt;</code></li>`)),
+            removedAttributes.map(([attr, elName]) => $(`<li><code>${attr}</code> (in <code>&lt;${elName}&gt;</code>)</li>`)));
+
+        $tgt.parents('.form-group').siblings('.post-preview').html(html);
+        $tgt.parents('form').find('.js-post-html[name="__html"]').val(html + '<!-- g: js, mdit -->');
+      }, 0);
+
+      if (featureTimeout) {
+        clearTimeout(featureTimeout);
       }
-    }, 1000);
-  }).on('keyup', ev => {
-    clearTimeout(draftTimeout);
-    const text = $(ev.target).val();
-    draftTimeout = setTimeout(() => {
-      saveDraft(text, $(ev.target));
-    }, 3000);
-  }).trigger('markdown');
 
-  postFields.parents('form').on('submit', async ev => {
+      featureTimeout = setTimeout(() => {
+        if (window['MathJax']) {
+          MathJax.typeset();
+        }
+        if (window['hljs']) {
+          hljs.highlightAll();
+        }
+      }, 1000);
+    };
+  })()).trigger('markdown');
+
+  postFields.parents('form').on('submit', async (ev) => {
     const $tgt = $(ev.target);
     const field = $tgt.find('.post-field');
 
@@ -252,24 +363,64 @@ $(() => {
   $('.js-draft-loaded').each((i, e) => {
     $(e).parents('.widget').after(`<div class="notice is-info has-font-size-caption">
       <i class="fas fa-exclamation-circle"></i> <strong>Draft loaded.</strong>
-      You've edited this post before but didn't save it. We loaded your edits here for you.
+      You had edited this before but haven't saved it. We loaded the edits for you.
     </div>`);
   });
 
-  $('.js-permalink > .js-text').text('Copy Link');
-  $('.js-permalink').on('click', ev => {
-    ev.preventDefault();
+  const setCopyButtonState = ($button, state) => {
+    const isSuccess = state === "success";
+    const buttonClass = isSuccess ? "is-green" : "is-danger";
+    const iconClass = isSuccess ? "fa-check" : "fa-times";
+
+    const $icon = $button.find(".fa");
+
+    $icon.removeClass("fa-copy");
+    $icon.addClass(iconClass);
+    $button.addClass(buttonClass);
 
-    const $tgt = $(ev.target).is('a') ? $(ev.target) : $(ev.target).parents('a');
-    const link = $tgt.attr('href');
-    navigator.clipboard.writeText(link);
-    $tgt.find('.js-text').text('Copied!');
     setTimeout(() => {
-      $tgt.find('.js-text').text('Copy Link');
-    }, 1000);
+      $icon.removeClass(iconClass);
+      $button.removeClass(buttonClass);
+      $icon.addClass("fa-copy");
+    }, 1e3);
+  };
+
+  $(".js-permalink-trigger").removeAttr("hidden");
+
+  $(".js-permalink-copy").on("click", async (ev) => {
+    ev.preventDefault();
+
+    const $tgt = $(ev.target);
+
+    const $button = $tgt.hasClass("js-permalink-copy")
+      ? $tgt
+      : $tgt.parents(".js-permalink-copy");
+
+    const postId = $button.data("post-id");
+    const linkType = $button.data("link-type");
+
+    if (!postId || !linkType) {
+      return;
+    }
+
+    const $input = $(`#permalink-${postId}-${linkType}`);
+
+    const url = $input.val();
+
+    if (!url) {
+      return;
+    }
+
+    try {
+      await navigator.clipboard.writeText(url);
+      setCopyButtonState($button, "success");
+    }
+    catch (_e) {
+      setCopyButtonState($button, "error");
+    }
   });
 
-  $('.js-nominate-promotion').on('click', async ev => {
+  $('.js-nominate-promotion').on('click', async (ev) => {
     ev.preventDefault();
 
     const $tgt = $(ev.target);
@@ -291,7 +442,7 @@ $(() => {
     $('.js-mod-tools').removeClass('is-active');
   });
 
-  $('.js-cancel-edit').on('click', async ev => {
+  $('.js-cancel-edit').on('click', async (ev) => {
     ev.preventDefault();
 
     let $btn = $(ev.target);
diff --git a/app/assets/javascripts/privileges.js b/app/assets/javascripts/privileges.js
index b61d17a0a43add81fed952f7b4f3b21cd9dd5ee3..f63feeb69d9f2b3839a397d4833d1c657bec295c 100644
--- a/app/assets/javascripts/privileges.js
+++ b/app/assets/javascripts/privileges.js
@@ -27,7 +27,11 @@ $(() => {
     const $input = $td.find('.js-privilege-edit');
     const name = $input.data('name');
     const type = $input.data('type');
-    const value = parseFloat($input.val() || '') || null;
+
+    // incorrect input values will cause rawValue to be NaN
+    const rawValue = parseFloat($input.val())
+
+    const value = Number.isNaN(rawValue) ? null : rawValue;
 
     const resp = await fetch(`/admin/privileges/${name}`, {
       method: 'POST',
diff --git a/app/assets/javascripts/qpixel_api.js b/app/assets/javascripts/qpixel_api.js
index df4236002fe159bff01762aa799f8eb793d5944c..d63f6bdf42743026dbc65cf81ba9e48eb190424f 100644
--- a/app/assets/javascripts/qpixel_api.js
+++ b/app/assets/javascripts/qpixel_api.js
@@ -3,6 +3,30 @@ const validators = [];
 /** Counts notifications popped up at any time. */
 let popped_modals_ct = 0;
 
+/**
+ * @typedef {{
+ *  min_score: number | null,
+ *  max_score: number | null,
+ *  min_answers: number | null,
+ *  max_answers: number | null,
+ *  include_tags: [string, number][],
+ *  exclude_tags: [string, number][],
+ *  status: 'any' | 'closed' | 'open',
+ *  system: boolean,
+ * }} Filter
+ *
+ * @typedef {{
+ *  id: number,
+ *  username: string,
+ *  is_moderator: boolean,
+ *  is_admin: boolean,
+ *  is_global_moderator: boolean,
+ *  is_global_admin: boolean,
+ *  trust_level: number,
+ *  se_acct_id: string | null,
+ * }} User
+ */
+
 window.QPixel = {
   /**
    * Get the current CSRF anti-forgery token. Should be passed as the X-CSRF-Token header when
@@ -20,7 +44,7 @@ window.QPixel = {
    * @param type the type to apply to the popup - warning, danger, etc.
    * @param message the message to show
    */
-  createNotification: function(type, message) {
+  createNotification: function (type, message) {
     // Some messages include a date stamp, `append_date` governs that.
     let append_date = false;
     let message_with_date = message;
@@ -42,26 +66,26 @@ window.QPixel = {
     }
     const span = '<span aria-hidden="true">&times;</span>';
     const button = ('<button type="button" class="button is-close-button" data-dismiss="alert" aria-label="Close">' +
-        span + '</button>');
-    $("<div></div>")
-    .addClass("notice has-shadow-3 is-" + type)
-    .html(button + '<p>' + message_with_date + '</p>')
-    .css({
-      'position': 'fixed',
-      'top': '50px',
-      'left': '50%',
-      'transform': 'translateX(-50%)',
-      'width': '100%',
-      'max-width': '800px',
-      'cursor': 'pointer'
-    })
-    .on('click', function(ev) {
-      $(this).fadeOut(200, function() {
-        $(this).remove();
-        popped_modals_ct = popped_modals_ct > 0 ? (popped_modals_ct - 1) : 0;
-      });
-    })
-    .appendTo(document.body);
+      span + '</button>');
+    $('<div></div>')
+      .addClass('notice has-shadow-3 is-' + type)
+      .html(button + '<p>' + message_with_date + '</p>')
+      .css({
+        'position': 'fixed',
+        'top': '50px',
+        'left': '50%',
+        'transform': 'translateX(-50%)',
+        'width': '100%',
+        'max-width': '800px',
+        'cursor': 'pointer'
+      })
+      .on('click', function (ev) {
+        $(this).fadeOut(200, function () {
+          $(this).remove();
+          popped_modals_ct = popped_modals_ct > 0 ? (popped_modals_ct - 1) : 0;
+        });
+      })
+      .appendTo(document.body);
     popped_modals_ct += 1;
   },
 
@@ -70,7 +94,7 @@ window.QPixel = {
    * @param el the element for which to find the offset.
    * @returns {{top: integer, left: integer, bottom: integer, right: integer}}
    */
-  offset: function(el) {
+  offset: function (el) {
     const topLeft = $(el).offset();
     return {
       top: topLeft.top,
@@ -151,21 +175,59 @@ window.QPixel = {
     $field.val(prev.substring(0, $field[0].selectionStart) + text + prev.substring($field[0].selectionEnd));
   },
 
+  /**
+   * Used to prevent launching multiple requests to /users/me
+   * @type {Promise<Response>|null}
+   */
+  _pendingUserResponse: null,
+
+  /**
+   * @type {User|null}
+   */
   _user: null,
 
   /**
-   * Get the user object for the current user.
-   * @returns {Promise<Object>} a JSON object containing user details
+   * FIFO-style fetch wrapper for /users/me requests
+   * @returns {Promise<Response>}
    */
-  user: async () => {
-    if (QPixel._user) return QPixel._user;
-    const resp = await fetch('/users/me', {
+  _fetchUser () {
+    if (QPixel._pendingUserResponse) {
+      return QPixel._pendingUserResponse;
+    }
+
+    const myselfPromise = fetch('/users/me', {
       credentials: 'include',
       headers: {
         'Accept': 'application/json'
       }
     });
-    QPixel._user = await resp.json();
+
+    QPixel._pendingUserResponse = myselfPromise;
+
+    return myselfPromise;
+  },
+
+  /**
+   * Get the user object for the current user.
+   * @returns {Promise<User>} a JSON object containing user details
+   */
+  user: async () => {
+    if (QPixel._user != null || document.body.dataset.userId === 'none') {
+      return QPixel._user;
+    }
+
+    try {
+      const resp = await QPixel._fetchUser();
+
+      if (!resp.bodyUsed) {
+        QPixel._user = await resp.json();
+      }
+    }
+    finally {
+      // ensures pending user is cleared regardless of network errors
+      QPixel._pendingUserResponse = null;
+    }
+
     return QPixel._user;
   },
 
@@ -176,30 +238,24 @@ window.QPixel = {
    * localStorage, or Redis via AJAX.
    * @returns {Promise<Object>} a JSON object containing user preferences
    */
-  preferences: async () => {
-    if (this._preferences == null && !!localStorage['qpixel.user_preferences']) {
-      this._preferences = JSON.parse(localStorage['qpixel.user_preferences']);
-
-      // If we don't have the global key, we're probably using an old preferences schema.
-      if (!this._preferences.global) {
-        delete localStorage['qpixel.user_preferences'];
-        this._preferences = null;
-      }
+  _getPreferences: async () => {
+    // Early return for the most frequent case (local variable already contains the preferences)
+    if (QPixel._preferences != null) {
+      return QPixel._preferences;
     }
-    else if (this._preferences == null) {
-      // If they're still null (or undefined) after loading from localStorage, we're probably on a site we haven't
-      // loaded them for yet. Load from Redis via AJAX.
-      const resp = await fetch('/users/me/preferences', {
-        credentials: 'include',
-        headers: {
-          'Accept': 'application/json'
-        }
-      });
-      const data = await resp.json();
-      localStorage['qpixel.user_preferences'] = JSON.stringify(data);
-      this._preferences = data;
+    // Early return the preferences from localStorage unless null or undefined
+    const key = QPixel._preferencesLocalStorageKey();
+    const localStoragePreferences = (key in localStorage)
+      ? JSON.parse(localStorage[key])
+      : null;
+    if (localStoragePreferences != null) {
+      QPixel._preferences = localStoragePreferences;
+      return QPixel._preferences;
     }
-    return this._preferences;
+    // Preferences are still null (or undefined) after loading from localStorage, so we're probably on a site we
+    // haven't loaded them for yet. Load from Redis via AJAX.
+    await QPixel._cachedFetchPreferences();
+    return QPixel._preferences;
   },
 
   /**
@@ -209,29 +265,25 @@ window.QPixel = {
    * @returns {Promise<*>} the value of the requested preference
    */
   preference: async (name, community = false) => {
-    let prefs = await QPixel.preferences();
-    let value = community ? prefs.community[name] : prefs.global[name];
-
-    // Deliberate === here: null is a valid value for a preference, but undefined means we haven't fetched it.
-    // If we haven't fetched a preference, that probably means it's new - run a full re-fetch.
-    if (value === undefined) {
-      const resp = await fetch('/users/me/preferences', {
-        credentials: 'include',
-        headers: {
-          'Accept': 'application/json'
-        }
-      });
-      const data = await resp.json();
-      localStorage['qpixel.user_preferences'] = JSON.stringify(data);
-      this._preferences = data;
+    const user = await QPixel.user();
 
-      prefs = await QPixel.preferences();
-      value = community ? prefs.community[name] : prefs.global[name];
-      return value;
+    if (!user) {
+      return null;
     }
-    else {
+
+    let prefs = await QPixel._getPreferences();
+    let value = community ? prefs.community[name] : prefs.global[name];
+
+    // Note that null is a valid value for a preference, but undefined means we haven't fetched it.
+    if (typeof (value) !== 'undefined') {
       return value;
     }
+    // If we haven't fetched a preference, that probably means it's new - run a full re-fetch.
+    await QPixel._cachedFetchPreferences();
+
+    prefs = await QPixel._getPreferences();
+    value = community ? prefs.community[name] : prefs.global[name];
+    return value;
   },
 
   /**
@@ -250,7 +302,7 @@ window.QPixel = {
         'Accept': 'application/json',
         'Content-Type': 'application/json'
       },
-      body: JSON.stringify({ name, value, community })
+      body: JSON.stringify({name, value, community})
     });
     const data = await resp.json();
     if (data.status !== 'success') {
@@ -258,11 +310,163 @@ window.QPixel = {
       console.error(resp);
     }
     else {
-      this._preferences = data.preferences;
-      localStorage['qpixel.user_preferences'] = JSON.stringify(this._preferences);
+      QPixel._updatePreferencesLocally(data.preferences);
+    }
+  },
+
+  /**
+   * @returns {Promise<Record<string, Filter>>}
+   */
+  filters: async () => {
+    if (this._filters == null) {
+      // If they're still null (or undefined) after loading from localStorage, we're probably on a site we haven't
+      // loaded them for yet. Load via AJAX.
+      const resp = await fetch('/users/me/filters', {
+        credentials: 'include',
+        headers: {
+          'Accept': 'application/json'
+        }
+      });
+      const data = await resp.json();
+      localStorage['qpixel.user_filters'] = JSON.stringify(data);
+      this._filters = data;
+    }
+
+    return this._filters;
+  },
+
+  /**
+   * Fetches default user filter for a given category
+   * @param categoryId id of the category to fetch
+   * @returns {Promise<string>}
+   */
+  defaultFilter: async (categoryId) => {
+    const user = await QPixel.user();
+
+    if (!user) {
+      return '';
+    }
+
+    const resp = await fetch(`/users/me/filters/default?category=${categoryId}`, {
+      credentials: 'include',
+      headers: {
+        'Accept': 'application/json'
+      }
+    });
+
+    const data = await resp.json();
+    return data.name;
+  },
+
+  setFilterAsDefault: async (categoryId, name) => {
+    const resp = await fetch(`/categories/${categoryId}/filters/default`, {
+      method: 'POST',
+      credentials: 'include',
+      headers: {
+        'X-CSRF-Token': QPixel.csrfToken(),
+        'Accept': 'application/json',
+        'Content-Type': 'application/json'
+      },
+      body: JSON.stringify({name})
+    });
+  },
+
+  setFilter: async (name, filter, category, isDefault) => {
+    const resp = await fetch('/users/me/filters', {
+      method: 'POST',
+      credentials: 'include',
+      headers: {
+        'X-CSRF-Token': QPixel.csrfToken(),
+        'Accept': 'application/json',
+        'Content-Type': 'application/json'
+      },
+      body: JSON.stringify(Object.assign(filter, {name, category, is_default: isDefault}))
+    });
+    const data = await resp.json();
+    if (data.status !== 'success') {
+      console.error(`Filter persist failed (${name})`);
+      console.error(resp);
+    }
+    else {
+      this._filters = data.filters;
+      localStorage['qpixel.user_filters'] = JSON.stringify(this._filters);
+    }
+  },
+
+  deleteFilter: async (name, system = false) => {
+    const resp = await fetch('/users/me/filters', {
+      method: 'DELETE',
+      credentials: 'include',
+      headers: {
+        'X-CSRF-Token': QPixel.csrfToken(),
+        'Accept': 'application/json',
+        'Content-Type': 'application/json'
+      },
+      body: JSON.stringify({name, system})
+    });
+    const data = await resp.json();
+    if (data.status !== 'success') {
+      console.error(`Filter deletion failed (${name})`);
+      console.error(resp);
+    }
+    else {
+      this._filters = data.filters;
+      localStorage['qpixel.user_filters'] = JSON.stringify(this._filters);
     }
   },
 
+  /**
+   * Get the key to use for storing user preferences in localStorage, to avoid conflating users
+   * @returns string the localStorage key
+   */
+  _preferencesLocalStorageKey: () => {
+    const id = document.body.dataset.userId;
+    const key = `qpixel.user_${id}_preferences`;
+    QPixel._preferencesLocalStorageKey = () => key;
+    return key;
+  },
+
+  /**
+   * Call _fetchPreferences but only the first time to prevent redundant HTTP requests
+   * @returns {Promise<void>}
+   */
+  _cachedFetchPreferences: async () => {
+    // No 'await' because we want the promise not its value
+    const cachedPromise = QPixel._fetchPreferences();
+    // Redefine this function to await this same initial promise on every subsequent call
+    // This prevents multiple calls from triggering multiple redundant '_fetchPreferences' calls
+    QPixel._cachedFetchPreferences = async () => {
+      await cachedPromise;
+    };
+    // Remember to await the promise so the very first call does not return before '_fetchPreferences' returns
+    await cachedPromise;
+  },
+
+  /**
+   * Update local variable _preferences and localStorage with an AJAX call for the user preferences
+   * @returns {Promise<void>}
+   */
+  _fetchPreferences: async () => {
+    const resp = await fetch('/users/me/preferences', {
+      credentials: 'include',
+      headers: {
+        'Accept': 'application/json'
+      }
+    });
+    const data = await resp.json();
+    QPixel._updatePreferencesLocally(data);
+  },
+
+  /**
+   * Set local variable _preferences and localStorage to new preferences data
+   * @param data an object, containing the new preferences data
+   */
+  _updatePreferencesLocally: data => {
+    QPixel._preferences = data;
+    const key = QPixel._preferencesLocalStorageKey();
+    localStorage[key] = JSON.stringify(QPixel._preferences);
+  },
+
   /**
    * Get the word in a string that the given position is in, and the position within that word.
    * @param splat an array, containing the string already split by however you define a "word"
diff --git a/app/assets/javascripts/suggested_edit.js b/app/assets/javascripts/suggested_edit.js
index b4aae89a4d701562397a7bdd89050860af8c8257..4763022ebdebf7e85a4940291d1cbfb549ee64fb 100644
--- a/app/assets/javascripts/suggested_edit.js
+++ b/app/assets/javascripts/suggested_edit.js
@@ -3,6 +3,7 @@ $(document).on('ready', function () {
     ev.preventDefault();
     const self = $(ev.target);
     const editId = self.attr('data-suggested-edit-approve');
+    const comment = $('#summary').val();
 
     const resp = await fetch(`/posts/suggested-edit/${editId}/approve`, {
       method: 'POST',
@@ -10,7 +11,8 @@ $(document).on('ready', function () {
       headers: {
         'Content-Type': 'application/json',
         'X-CSRF-Token': QPixel.csrfToken()
-      }
+      },
+      body: JSON.stringify({ comment })
     });
     const data = await resp.json();
 
diff --git a/app/assets/javascripts/tags.js b/app/assets/javascripts/tags.js
index a2d71685fd74bcd033f2cd12161048f675451f79..460c215b5a77d9ca033dd18d21fd22711daea645 100644
--- a/app/assets/javascripts/tags.js
+++ b/app/assets/javascripts/tags.js
@@ -14,7 +14,8 @@ $(() => {
   };
 
   const template = (tag) => {
-    const tagSpan = `<span>${tag.text}</span>`;
+    const tagSynonyms = !!tag.synonyms ? ` <i>(${tag.synonyms})</i>` : '';
+    const tagSpan = `<span>${tag.text}${tagSynonyms}</span>`;
     let desc = !!tag.desc ? splitWordsMaxLength(tag.desc, 120) : '';
     const descSpan = !!tag.desc ?
       `<br/><span class="has-color-tertiary-900 has-font-size-caption">${desc[0]}${desc.length > 1 ? '...' : ''}</span>` :
@@ -22,7 +23,7 @@ $(() => {
     return $(tagSpan + descSpan);
   }
 
-  $('.js-tag-select').each((i, el) => {
+  $('.js-tag-select').each((_i, el) => {
     const $tgt = $(el);
     let $this;
     const useIds = $tgt.attr('data-use-ids') === 'true';
@@ -38,16 +39,16 @@ $(() => {
         data: function (params) {
           $this = $(this);
           // (for the tour)
-          if ($this.data('tag-set') === '-1') {
-            return Object.assign(params, { tag_set: "1" });
+          if (Number($this.data('tag-set')) === -1) {
+            return Object.assign(params, { tag_set: '1' });
           }
           return Object.assign(params, { tag_set: $this.data('tag-set') });
         },
         headers: { 'Accept': 'application/json' },
         delay: 100,
-        processResults: data => {
+        processResults: (data) => {
           // (for the tour)
-          if ($this.data('tag-set') === '-1') {
+          if (Number($this.data('tag-set')) === -1) {
             return {
               results: [
                 { id: 1, text: 'hot-red-firebreather', desc: 'Very cute dragon' },
@@ -61,17 +62,66 @@ $(() => {
             results: data.map(t => ({
               id: useIds ? t.id : t.name,
               text: t.name.replace(/</g, '&#x3C;').replace(/>/g, '&#x3E;'),
+              synonyms: processSynonyms($this, t.tag_synonyms),
               desc: t.excerpt
             }))
           };
         },
       },
+      placeholder: '',
       templateResult: template,
       allowClear: true
     });
   });
 
-  $('.js-add-required-tag').on('click', ev => {
+  function processSynonyms($search, synonyms) {
+    if (!synonyms) return synonyms;
+
+    let displayedSynonyms;
+    if (synonyms.length > 3) {
+      const searchValue = $search.data('select2').selection.$search.val().toLowerCase();
+      displayedSynonyms = synonyms.filter(ts => ts.name.includes(searchValue)).slice(0, 3);
+    } else {
+      displayedSynonyms = synonyms;
+    }
+    let synonymsString = displayedSynonyms.map((ts) => `${ts.name.replace(/</g, '&#x3C;').replace(/>/g, '&#x3E;')}`).join(', ');
+    if (synonyms.length > displayedSynonyms.length) {
+      synonymsString += `, ${synonyms.length - displayedSynonyms.length} more synonyms`;
+    }
+    return synonymsString;
+  }
+
+  $('#add-tag-synonym').on('click', (ev) => {
+    const $wrapper = $('#tag-synonyms-wrapper');
+    const lastId = $wrapper.children('.tag-synonym').last().attr('data-id');
+    const newId = parseInt(lastId, 10) + 1;
+
+    //Duplicate the first element at the end of the wrapper
+    const newField = $wrapper.find('.tag-synonym[data-id="0"]')[0]
+                             .outerHTML
+                             .replace(/data-id="0"/g, 'data-id="' + newId + '"')
+                             .replace(/(?<connector>attributes(\]\[)|(_))0/g, '$<connector>' + newId)
+    $wrapper.append(newField);
+
+    //Alter the newly added tag synonym
+    const $newTagSynonym = $wrapper.children().last();
+    $newTagSynonym.find('.tag-synonym-name').removeAttr('value').removeAttr('readonly').removeAttr('disabled');
+    $newTagSynonym.find('.destroy-tag-synonym').attr('value', 'false');
+    $newTagSynonym.show();
+
+    //Add handler for removing an element
+    $newTagSynonym.find(`.remove-tag-synonym`).click(removeTagSynonym);
+  });
+
+  $('.remove-tag-synonym').click(removeTagSynonym);
+
+  function removeTagSynonym() {
+    const synonym = $(this).closest('.tag-synonym');
+    synonym.find('.destroy-tag-synonym').attr('value', 'true');
+    synonym.hide();
+  }
+
+  $('.js-add-required-tag').on('click', (ev) => {
     const $tgt = $(ev.target);
     const useIds = $tgt.attr('data-use-ids') === 'true';
     const tagId = $tgt.attr('data-tag-id');
@@ -87,7 +137,7 @@ $(() => {
     }
   });
 
-  $('.js-rename-tag').on('click', async ev => {
+  $('.js-rename-tag').on('click', async (ev) => {
     const $tgt = $(ev.target).is('a') ? $(ev.target) : $(ev.target).parents('a');
     const categoryId = $tgt.attr('data-category');
     const tagId = $tgt.attr('data-tag');
diff --git a/app/assets/javascripts/two_factor.js b/app/assets/javascripts/two_factor.js
new file mode 100644
index 0000000000000000000000000000000000000000..406dadc054d6945726c7f497a6ec3af3dad70cd7
--- /dev/null
+++ b/app/assets/javascripts/two_factor.js
@@ -0,0 +1,33 @@
+$(() => {
+   $('.js-backup-code-form').on('submit', async ev => {
+      ev.preventDefault();
+      const $tgt = $(ev.target);
+      const $input = $tgt.find('input[name="code"]');
+      const code = $input.val();
+      const req = await fetch('/users/two-factor/backup', {
+          method: 'POST',
+          headers: {
+              'X-CSRF-Token': QPixel.csrfToken(),
+              'Content-Type': 'application/json'
+          },
+          body: JSON.stringify({ code })
+      });
+      const res = await req.json();
+
+      if (res.status === 'error') {
+          const $label = $tgt.find('label[for="code"]');
+          $label.text(res.message);
+          $input.addClass('is-danger');
+          $tgt.find('input[type="submit"]').removeAttr('disabled');
+      }
+      else if (res.status === 'success') {
+        const codeForm = $(`<details>
+                              <summary>Show code</summary>
+                              <label for="backup-code" class="form-element">2FA backup code</label>
+                              <input class="form-element" type="text" readonly name="backup-code" id="backup-code" value="${res.code}" />
+                            </details>`);
+        $tgt.after(codeForm);
+        $tgt.remove();
+      }
+   });
+});
diff --git a/app/assets/javascripts/votes.js b/app/assets/javascripts/votes.js
index 611a78ffeaa5f666dad9887d5da8297aed7e2f9d..6dae33b0a1437d9b4667eb529f1a8f3354e3aec3 100644
--- a/app/assets/javascripts/votes.js
+++ b/app/assets/javascripts/votes.js
@@ -2,22 +2,29 @@ $(() => {
   $(document).on('click', '.vote-button', async evt => {
     const $tgt = $(evt.target).is('button') ? $(evt.target) : $(evt.target).parents('button');
     const $post = $tgt.parents('.post');
-    const $up = $post.find('.post--votes').find('.js-upvote-count');
-    const $down = $post.find('.post--votes').find('.js-downvote-count');
+
+    const $container = $post.find(".post--votes");
+
+    const $up = $container.find('.js-upvote-count');
+    const $down = $container.find('.js-downvote-count');
     const voteType = $tgt.data('vote-type');
     const voted = $tgt.hasClass('is-active');
 
     if (voted) {
       const voteId = $tgt.attr('data-vote-id');
+
       const resp = await fetch(`/votes/${voteId}`, {
         method: 'DELETE',
         credentials: 'include',
         headers: { 'X-CSRF-Token': QPixel.csrfToken() }
       });
+
       const data = await resp.json();
+
       if (data.status === 'OK') {
         $up.text(`+${data.upvotes}`);
         $down.html(`&minus;${data.downvotes}`);
+        $container.attr("title", `Score: ${data.score}`);
         $tgt.removeClass('is-active')
             .removeAttr('data-vote-id');
       }
@@ -34,10 +41,13 @@ $(() => {
         headers: { 'Content-Type': 'application/json', 'X-CSRF-Token': QPixel.csrfToken() },
         body: JSON.stringify({post_id: $post.data('post-id'), vote_type: voteType})
       });
+
       const data = await resp.json();
+
       if (data.status === 'modified' || data.status === 'OK') {
         $up.text(`+${data.upvotes}`);
         $down.html(`&minus;${data.downvotes}`);
+        $container.attr("title", `Score: ${data.score}`);
         $tgt.addClass('is-active')
             .attr('data-vote-id', data.vote_id);
 
diff --git a/app/assets/stylesheets/application.scss b/app/assets/stylesheets/application.scss
index 2b105d33f3ee4bc059d779243cccf17c66ca5755..44a2e1d213f1f74e57eee82df2ad9015f735d629 100644
--- a/app/assets/stylesheets/application.scss
+++ b/app/assets/stylesheets/application.scss
@@ -145,22 +145,6 @@ hr {
   }
 }
 
-.footnote-ref a::before {
-  content: '[';
-}
-
-div.post-preview .footnote-ref a::before {
-  content: '';
-}
-
-.footnote-ref a::after {
-  content: ']';
-}
-
-div.post-preview .footnote-ref a::after {
-  content: '';
-}
-
 .footnotes-sep + .footnotes {
   border-top: 0;
 }
@@ -209,6 +193,10 @@ img {
   border-top: 1px solid #9daeb7;
 }
 
+.widget .widget--body .widget--body-extra {
+  margin-left: 0.75em;
+}
+
 pre {
   background: #f0f0f0;
   border: 0;
diff --git a/app/assets/stylesheets/categories.scss b/app/assets/stylesheets/categories.scss
index 93b1c4f891ddc669cd5577ff796d9087c6958a8f..6989a8781d8bf9060cb67221b9a9de02360dd514 100644
--- a/app/assets/stylesheets/categories.scss
+++ b/app/assets/stylesheets/categories.scss
@@ -16,6 +16,12 @@
     align-items: center;
     justify-content: space-between;
   }
+
+  & .category-meta--start {
+    display: flex;
+    align-items: center;
+    gap: 0.5em;
+  }
 }
 
 .category-header--nav {
diff --git a/app/assets/stylesheets/comments.scss b/app/assets/stylesheets/comments.scss
index df46861660f06483aaa7014b5a1ccf0e97385c6c..f803527585bd71a40c0765fb5f7a38b5851a1472 100644
--- a/app/assets/stylesheets/comments.scss
+++ b/app/assets/stylesheets/comments.scss
@@ -65,6 +65,17 @@
   font-style: italic;
 }
 
+.post--comments-header {
+  align-items: center;
+  display: flex;
+  justify-content: space-between;
+  margin-bottom: 0.75rem;
+}
+
+.post--comments-container {
+  margin-bottom: 1rem;
+}
+
 .post--comments-thread.is-inline {
   padding: 0.5rem 0.25rem;
   display: flex;
@@ -139,7 +150,7 @@
 .new-thread-modal {
   box-shadow: 0 3px 5px -2px #eee;
   border: 1px solid #d0d9dd;
-  margin-top: 10px;
+  margin-top: 1rem;
   padding: 0.7em;
   display: none;
 }
diff --git a/app/assets/stylesheets/forms.scss b/app/assets/stylesheets/forms.scss
index baa824c4de7ce8d52f0ee9667a71a090d69d01e1..284c3418e6818240188af6360742eec146c745f1 100644
--- a/app/assets/stylesheets/forms.scss
+++ b/app/assets/stylesheets/forms.scss
@@ -17,14 +17,6 @@ select.form-element {
 
   @media screen and (min-width: $screen-md) {
     flex-direction: row;
-
-    & > :first-child {
-      margin: 0 0.5em 0 0 !important;
-    }
-
-    & > :last-child {
-      margin: 0 0 0 0.5em !important;
-    }
   }
 
   & > .form-group {
diff --git a/app/assets/stylesheets/post_history.scss b/app/assets/stylesheets/post_history.scss
index 55d9411b29aa85b39dd9e144c7be36063e1fccb2..12cfdcaf9f01d8a8b88e5de1a1c999d342e26a37 100644
--- a/app/assets/stylesheets/post_history.scss
+++ b/app/assets/stylesheets/post_history.scss
@@ -13,6 +13,10 @@
     display: unset;
     font-weight: unset;
     color: unset;
+
+    & .droppanel {
+      position: fixed;
+    }
   }
 
   &:last-of-type {
diff --git a/app/assets/stylesheets/posts.scss b/app/assets/stylesheets/posts.scss
index a1a548af67f875059905cafb91cdd8a75e29d319..67e911056d76113f137d14d367b3b662a96d7ae4 100644
--- a/app/assets/stylesheets/posts.scss
+++ b/app/assets/stylesheets/posts.scss
@@ -89,7 +89,25 @@ h1 .badge.is-tag.is-master-tag {
   width: calc(100% + 2px);
 
   + .widget--footer {
-    margin-bottom: 0;
+    border-top: none;
+    align-items: center;
+    margin: 0;
+
+    &.mdhint {
+      display: flex;
+      flex-wrap: wrap;
+      gap: 1em 0;
+      justify-content: space-between;
+
+      & > * {
+        padding: 0;
+      }
+    }
+
+    & > .draft-status {
+      text-align: center;
+      transition: opacity 0.5s ease-in-out;
+    }
   }
 }
 
diff --git a/app/assets/stylesheets/site_settings.scss b/app/assets/stylesheets/site_settings.scss
index f9a52d4d044a9dbb788ad8fdb9500bba644d1acd..9cdec80291ef7baa04ea9c2d8b1795fa3b02a44c 100644
--- a/app/assets/stylesheets/site_settings.scss
+++ b/app/assets/stylesheets/site_settings.scss
@@ -1,4 +1,5 @@
 .site-setting--value {
   min-height: 1em;
   min-width: 2em;
+  overflow-wrap: anywhere;
 }
\ No newline at end of file
diff --git a/app/assets/stylesheets/users.scss b/app/assets/stylesheets/users.scss
index a79c0662adb0074366eeffaf7f508102a2ef6bf8..867e962e3b2f68747466fefea9a9e8a4758a1a2e 100644
--- a/app/assets/stylesheets/users.scss
+++ b/app/assets/stylesheets/users.scss
@@ -43,6 +43,7 @@
   }
 }
 
+
 .profile-text {
   padding: 0.125em;
   line-height: 1.5;
@@ -50,17 +51,35 @@
   overflow: auto;
 }
 
-.user-profile-heading {
-  padding: 0.5em;
+.user-profile-heading-container {
+  align-items: center;
   border-bottom: 1px solid #ddd;
-  margin-bottom: 0;
-}
+  display: flex;
+  flex-wrap: wrap;
+  justify-content: space-between;
+  gap: 0.5em;
+
+  & > .user-profile-heading {
+    flex-grow: 1;
+    margin-bottom: 0;
+    margin-top: 0;
+    padding: 0.5em;
 
+    &:not(:last-child) {
+      padding-right: 0;
+    }
+  }
+
+  & > .button:last-child {
+    margin-right: 0;
+  }
+}
 .user-profile--image {
   text-align: center;
 
   img {
     width: 100%;
+    object-fit: contain;
   }
 }
 
diff --git a/app/assets/stylesheets/utilities.scss b/app/assets/stylesheets/utilities.scss
index be684f5f986693de8f37296b656ef8f6e5bb7638..2ed9f314baf533c846415dffe0e67a69cd5574ab 100644
--- a/app/assets/stylesheets/utilities.scss
+++ b/app/assets/stylesheets/utilities.scss
@@ -95,6 +95,16 @@ pre.pre-wrap {
   white-space: pre-wrap !important;
 }
 
+.copy-button {
+  display: none;
+  position: absolute;
+  right: 0;
+}
+
+div:hover > .copy-button {
+  display: block;
+}
+
 .stat-panel {
   flex: 1;
   border: 1px solid $muted-graphic;
@@ -271,3 +281,11 @@ span.spoiler {
     color: $key;
   }
 }
+
+.clearfix {
+  overflow: hidden;
+}
+
+.transparent {
+  opacity: 0;
+}
diff --git a/app/controllers/active_storage/base_controller.rb b/app/controllers/active_storage/base_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..63cf547088a94e25669b86410d72203a2bb31416
--- /dev/null
+++ b/app/controllers/active_storage/base_controller.rb
@@ -0,0 +1,17 @@
+class ActiveStorage::BaseController < ActionController::Base
+  before_action :enforce_signed_in
+  include ActiveStorage::SetCurrent
+  protect_from_forgery with: :exception
+
+  self.etag_with_template_digest = false
+
+  protected
+
+  def enforce_signed_in
+    # If not restricted, the user is signed in or the environment is test, allow all content.
+    return true if !SiteSetting['RestrictedAccess'] || user_signed_in? || Rails.env.test?
+
+    redirect_to '/', status: :forbidden
+    false
+  end
+end
diff --git a/app/controllers/admin_controller.rb b/app/controllers/admin_controller.rb
index b7453e9ced6ede292529bfbb136b34e3020de811..5d13f6d3b8fa99e65a503dd86fec4d4d7af763fe 100644
--- a/app/controllers/admin_controller.rb
+++ b/app/controllers/admin_controller.rb
@@ -3,7 +3,7 @@ class AdminController < ApplicationController
   before_action :verify_admin, except: [:change_back, :verify_elevation]
   before_action :verify_global_admin, only: [:admin_email, :send_admin_email, :new_site, :create_site, :setup,
                                              :setup_save, :hellban]
-  before_action :verify_developer, only: [:change_users, :impersonate]
+  before_action :verify_developer, only: [:change_users, :impersonate, :all_email, :send_all_email]
 
   def index; end
 
@@ -52,6 +52,18 @@ class AdminController < ApplicationController
     redirect_to admin_path
   end
 
+  def all_email; end
+
+  def send_all_email
+    Thread.new do
+      AdminMailer.with(body_markdown: params[:body_markdown], subject: params[:subject]).to_all_users.deliver_now
+    end
+    AuditLog.admin_audit(event_type: 'send_all_email', user: current_user,
+                         comment: "Subject: #{params[:subject]}")
+    flash[:success] = t 'admin.email_being_sent'
+    redirect_to admin_path
+  end
+
   def audit_log
     @logs = if current_user.is_global_admin
               AuditLog.unscoped.where.not(log_type: ['user_annotation', 'user_history'])
@@ -170,7 +182,12 @@ class AdminController < ApplicationController
     return not_found unless session[:impersonator_id].present?
 
     @impersonator = User.find session[:impersonator_id]
-    if @impersonator&.valid_password? params[:password]
+    if @impersonator&.sso_profile.present?
+      session.delete :impersonator_id
+      AuditLog.admin_audit(event_type: 'impersonation_end', related: current_user, user: @impersonator)
+      sign_out @impersonator
+      redirect_to new_saml_user_session_path
+    elsif @impersonator&.valid_password? params[:password]
       session.delete :impersonator_id
       AuditLog.admin_audit(event_type: 'impersonation_end', related: current_user, user: @impersonator)
       sign_in @impersonator
diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb
index ce8c81040301ea46cc73e07d5a4da8535819618b..efa5963cad640b000329fd45a6249e7fe40cb9ba 100644
--- a/app/controllers/application_controller.rb
+++ b/app/controllers/application_controller.rb
@@ -7,9 +7,15 @@ class ApplicationController < ActionController::Base
   protect_from_forgery with: :exception
   before_action :configure_permitted_parameters, if: :devise_controller?
   before_action :set_globals
+  before_action :enforce_signed_in, unless: :devise_controller?
   before_action :check_if_warning_or_suspension_pending
   before_action :distinguish_fake_community
   before_action :stop_the_awful_troll
+
+  # Before checking 2fa enforcing or access, store the location that the user is trying to access.
+  # In case re-authentication is necessary / the user signs in, we can direct back to this location directly.
+  before_action :store_user_location!, if: :storable_location?
+
   before_action :enforce_2fa
   before_action :block_write_request, if: :read_only_mode?
 
@@ -17,15 +23,19 @@ class ApplicationController < ActionController::Base
 
   def upload
     if ActiveStorage::Blob.service.class.name.end_with?('S3Service')
-      redirect_to helpers.upload_remote_url(params[:key]), status: 301
+      redirect_to helpers.upload_remote_url(params[:key]), status: 301, allow_other_host: true
     else
       blob = params[:key]
-      redirect_to url_for(ActiveStorage::Blob.find_by(key: blob.is_a?(String) ? blob : blob.key))
+      redirect_to url_for(ActiveStorage::Blob.find_by(key: blob.is_a?(String) ? blob : blob.key)),
+                  allow_other_host: true
     end
   end
 
   def dashboard
     @communities = Community.all
+    @edits = Post.unscoped do
+      SuggestedEdit.unscoped.joins(:post).where(active: true).group(Arel.sql('posts.category_id')).count
+    end
     render layout: 'without_sidebar'
   end
 
@@ -309,7 +319,8 @@ class ApplicationController < ActionController::Base
        # Enable users to log out even if 2fa is enforced
        !request.fullpath.end_with?('/users/sign_out') &&
        (current_user.is_global_admin ||
-         current_user.is_global_moderator)
+         current_user.is_global_moderator) &&
+       (current_user.sso_profile.blank? || SiteSetting['Enable2FAForSsoUsers'])
       redirect_path = '/users/two-factor'
       unless request.fullpath.end_with?(redirect_path)
         flash[:notice] = 'All global admins and global moderators must enable two-factor authentication to continue' \
@@ -319,6 +330,39 @@ class ApplicationController < ActionController::Base
     end
   end
 
+  # Ensure that the user is signed in before showing any content. If the user is not signed in, display the main page.
+  #
+  # Exceptions:
+  # - 4** and 500 error pages
+  # - stylesheets and javascript
+  # - assets
+  # - /help, /policy, /help/* and /policy/*
+  def enforce_signed_in
+    # If not restricted, the user is signed in or the environment is test, allow all content.
+    return true if !SiteSetting['RestrictedAccess'] || user_signed_in? || Rails.env.test?
+
+    # Allow error pages and assets
+    path = request.fullpath
+    return true if path.start_with?('/4') || path == '/500' ||
+                   path.start_with?('/assets/') ||
+                   path.end_with?('.css') || path.end_with?('.js')
+
+    # Make available to controller that the we should not leak posts in the sidebar
+    @prevent_sidebar = true
+
+    # Allow /help (help center), /help/* and /policy/* depending on settings
+    help = SiteSetting['RestrictedAccessHelpPagesPublic']
+    policy = SiteSetting['RestrictedAccessPolicyPagesPublic']
+    return true if (help && path.start_with?('/help/')) ||
+                   (policy && path.start_with?('/policy/')) ||
+                   (path == '/help' && (help || policy))
+
+    store_location_for(:user, request.fullpath) if storable_location?
+
+    render 'errors/restricted_content', layout: 'without_sidebar', status: :forbidden
+    false
+  end
+
   def block_write_request(**add)
     respond_to do |format|
       format.html do
@@ -342,12 +386,24 @@ class ApplicationController < ActionController::Base
     helpers.user_signed_in?
   end
 
+  def sso_sign_in_enabled?
+    helpers.sso_sign_in_enabled?
+  end
+
+  def devise_sign_in_enabled?
+    helpers.devise_sign_in_enabled?
+  end
+
   def authenticate_user!(_fav = nil, **_opts)
     unless user_signed_in?
       respond_to do |format|
         format.html do
           flash[:error] = 'You need to sign in or sign up to continue.'
-          redirect_to new_user_session_path
+          if devise_sign_in_enabled?
+            redirect_to new_user_session_path
+          else
+            redirect_to new_saml_user_session_path
+          end
         end
         format.json do
           render json: { error: 'You need to sign in or sign up to continue.' }, status: 401
@@ -355,4 +411,27 @@ class ApplicationController < ActionController::Base
       end
     end
   end
+
+  # Checks if the requested location should be stored.
+  #
+  # Its important that the location is NOT stored if:
+  # - The request method is not GET (non idempotent)
+  # - The request is handled by a Devise controller such as Devise::SessionsController as that could cause an
+  #    infinite redirect loop.
+  # - The request is an Ajax request as this can lead to very unexpected behaviour.
+  # - The request is to a location we dont want to store, such as:
+  #   - Anything trying to fetch for the current user (filters, preferences, etc) as it is not the actual page
+  #   - The mobile login, as it would redirect to the code url after the sign in
+  #   - Uploaded files, as these appear in posts and are not the main route we would want to store
+  def storable_location?
+    request.get? && is_navigational_format? && !devise_controller? && !request.xhr? &&
+      !request.path.start_with?('/users/me') &&
+      !request.path.start_with?('/users/mobile-login') &&
+      !request.path.start_with?('/uploads/')
+  end
+
+  # Stores the location in the system for the current session, such that after login we send them back to the same page.
+  def store_user_location!
+    store_location_for(:user, request.fullpath)
+  end
 end
diff --git a/app/controllers/categories_controller.rb b/app/controllers/categories_controller.rb
index b8076e3663ad653d57acd3f3b090a6e6d0980618..c319160f35ceb2e4f9125b1d4280b8cc2453f9a6 100644
--- a/app/controllers/categories_controller.rb
+++ b/app/controllers/categories_controller.rb
@@ -144,6 +144,7 @@ class CategoriesController < ApplicationController
                                      :color_code, :min_view_trust_level, :license_id, :sequence,
                                      :asking_guidance_override, :answering_guidance_override,
                                      :use_for_hot_posts, :use_for_advertisement,
+                                     :min_title_length, :min_body_length, :default_filter_id,
                                      display_post_types: [], post_type_ids: [], required_tag_ids: [],
                                      topic_tag_ids: [], moderator_tag_ids: [])
   end
@@ -161,8 +162,41 @@ class CategoriesController < ApplicationController
                     native: Arel.sql('att_source IS NULL DESC, last_activity DESC') }
     sort_param = sort_params[params[:sort]&.to_sym] || { last_activity: :desc }
     @posts = @category.posts.undeleted.where(post_type_id: @category.display_post_types)
-                      .includes(:post_type, :tags).list_includes.paginate(page: params[:page], per_page: 50)
-                      .order(sort_param)
+                      .includes(:post_type, :tags).list_includes
+    filter_qualifiers = helpers.params_to_qualifiers
+    @active_filter = helpers.active_filter
+
+    if filter_qualifiers.blank? && @active_filter[:name].blank?
+      if user_signed_in?
+        default_filter_id = helpers.default_filter(current_user.id, @category.id)
+        default_filter = Filter.find_by(id: default_filter_id)
+        default = :user if default_filter.present?
+      end
+
+      if default_filter.nil?
+        default_filter = @category.default_filter
+        default = :category if default_filter.present?
+      end
+
+      unless default_filter.nil?
+        filter_qualifiers = helpers.filter_to_qualifiers default_filter
+        @active_filter = {
+          default: default,
+          name: default_filter.name,
+          min_score: default_filter.min_score,
+          max_score: default_filter.max_score,
+          min_answers: default_filter.min_answers,
+          max_answers: default_filter.max_answers,
+          include_tags: default_filter.include_tags,
+          exclude_tags: default_filter.exclude_tags,
+          status: default_filter.status
+        }
+      end
+    end
+
+    @posts = helpers.qualifiers_to_sql(filter_qualifiers, @posts)
+    @filtered = filter_qualifiers.any?
+    @posts = @posts.paginate(page: params[:page], per_page: 50).order(sort_param)
   end
 
   def update_last_visit(category)
diff --git a/app/controllers/comments_controller.rb b/app/controllers/comments_controller.rb
index 57894f2975b0c1d682fcaecaecfee7398ab11188..607d391a8983497cf492202cc3c268f40e4c285e 100644
--- a/app/controllers/comments_controller.rb
+++ b/app/controllers/comments_controller.rb
@@ -101,12 +101,19 @@ class CommentsController < ApplicationController
   end
 
   def update
+    @post = @comment.post
+    @comment_thread = @comment.comment_thread
     before = @comment.content
+    before_pings = check_for_pings @comment_thread, before
     if @comment.update comment_params
       unless current_user.id == @comment.user_id
         AuditLog.moderator_audit(event_type: 'comment_update', related: @comment, user: current_user,
                                  comment: "from <<#{before}>>\nto <<#{@comment.content}>>")
       end
+
+      after_pings = check_for_pings @comment_thread, @comment.content
+      apply_pings(after_pings - before_pings - @comment_thread.thread_follower.to_a)
+
       render json: { status: 'success',
                      comment: render_to_string(partial: 'comments/comment', locals: { comment: @comment }) }
     else
diff --git a/app/controllers/email_logs_controller.rb b/app/controllers/email_logs_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..40cded41f6fcc5804d5ee0d463c079a40c532802
--- /dev/null
+++ b/app/controllers/email_logs_controller.rb
@@ -0,0 +1,28 @@
+class EmailLogsController < ApplicationController
+  skip_forgery_protection only: [:log]
+  skip_before_action :set_globals, only: [:log]
+  skip_before_action :distinguish_fake_community, only: [:log]
+  skip_before_action :enforce_signed_in, only: [:log]
+
+  def log
+    message_type = request.headers['X-Amz-SNS-Message-Type']
+    if ['SubscriptionConfirmation', 'Notification'].include? message_type
+      verifier = Aws::SNS::MessageVerifier.new
+      body = request.body.read
+      if verifier.authentic? body
+        aws_data = JSON.parse body
+        if message_type == 'SubscriptionConfirmation'
+          EmailLog.create(log_type: 'SubscriptionConfirmation', data: aws_data)
+        else
+          message_data = JSON.parse aws_data['Message']
+          log_type = message_data['notificationType']
+          destination = message_data['mail']['destination'].join(', ')
+          EmailLog.create(log_type: log_type, destination: destination, data: aws_data['Message'])
+        end
+        render plain: 'OK'
+      else
+        render plain: "You're not AWS. Go away.", status: 401
+      end
+    end
+  end
+end
diff --git a/app/controllers/flags_controller.rb b/app/controllers/flags_controller.rb
index a83947d37731ceae94f15c78d3efc61b425ac80c..f505f08fc4e50f4adde8e7bc5bdabc3d96503904 100644
--- a/app/controllers/flags_controller.rb
+++ b/app/controllers/flags_controller.rb
@@ -41,7 +41,7 @@ class FlagsController < ApplicationController
   end
 
   def history
-    @user = User.find(params[:id])
+    @user = helpers.user_with_me params[:id]
     unless @user == current_user || (current_user.is_admin || current_user.is_moderator)
       not_found
       return
diff --git a/app/controllers/moderator_controller.rb b/app/controllers/moderator_controller.rb
index ae482cd4f0e9e33197c17e0e2e7365b76327e1af..04d2a94b5419ffe482b60bc7007d47429b389f4c 100644
--- a/app/controllers/moderator_controller.rb
+++ b/app/controllers/moderator_controller.rb
@@ -48,8 +48,8 @@ class ModeratorController < ApplicationController
     render json: { status: 'success', success: true }
   end
 
-  VoteData = Struct.new(:cast, :received)
-  VoteSummary = Struct.new(:breakdown, :types, :total)
+  VoteData = Struct.new(:cast, :received, keyword_init: true)
+  VoteSummary = Struct.new(:breakdown, :types, :total, keyword_init: true)
 
   def user_vote_summary
     @user = User.find params[:id]
diff --git a/app/controllers/notifications_controller.rb b/app/controllers/notifications_controller.rb
index 02477f56a6d9e46a32a5d6f8d9677349a3d03ec0..b47d7fcfcebc26fab7bacc5a3da0ea0793de982a 100644
--- a/app/controllers/notifications_controller.rb
+++ b/app/controllers/notifications_controller.rb
@@ -6,7 +6,7 @@ class NotificationsController < ApplicationController
     @notifications = Notification.unscoped.where(user: current_user).paginate(page: params[:page], per_page: 100)
                                  .order(Arel.sql('is_read ASC, created_at DESC'))
     respond_to do |format|
-      format.html { render :index }
+      format.html { render :index, layout: 'without_sidebar' }
       format.json { render json: @notifications, methods: :community_name }
     end
   end
diff --git a/app/controllers/post_history_controller.rb b/app/controllers/post_history_controller.rb
index 050c4ba38a3c3f210a8af9c5f060f28f57bfc4d7..543771e027a6caabf50e3c2a42f477b2fed988c9 100644
--- a/app/controllers/post_history_controller.rb
+++ b/app/controllers/post_history_controller.rb
@@ -6,8 +6,30 @@ class PostHistoryController < ApplicationController
       return not_found
     end
 
-    @history = PostHistory.where(post_id: params[:id]).includes(:post_history_type, :user, post_history_tags: [:tag])
-                          .order(created_at: :desc).paginate(per_page: 20, page: params[:page])
-    render layout: 'without_sidebar'
+    @history = PostHistory.where(post_id: params[:id])
+                          .includes(:post_history_type, :user, post_history_tags: [:tag])
+                          .order(created_at: :desc, id: :desc)
+                          .paginate(per_page: 20, page: params[:page])
+
+    if @post&.help_category.nil?
+      render layout: 'without_sidebar'
+    else
+      render 'post_history/post', layout: 'without_sidebar', locals: { show_content: false }
+    end
+  end
+
+  def slug_post
+    @post = Post.by_slug(params[:slug], current_user)
+
+    if @post.nil?
+      return not_found
+    end
+
+    @history = PostHistory.where(post_id: @post.id)
+                          .includes(:post_history_type, :user)
+                          .order(created_at: :desc, id: :desc)
+                          .paginate(per_page: 20, page: params[:page])
+
+    render 'post_history/post', layout: 'without_sidebar', locals: { show_content: false }
   end
 end
diff --git a/app/controllers/posts_controller.rb b/app/controllers/posts_controller.rb
index d3295ea7729f0c1aec73d4ad2b55e99031e2be46..5e8b293a8811b13fe2e7ec074c4b24d2e2b2d5a3 100644
--- a/app/controllers/posts_controller.rb
+++ b/app/controllers/posts_controller.rb
@@ -26,7 +26,7 @@ class PostsController < ApplicationController
       return
     end
 
-    if ['HelpDoc', 'PolicyDoc'].include?(@post_type.name)
+    if @post_type.system?
       check_permissions
       # return # uncomment if you add more code after this
     end
@@ -73,7 +73,7 @@ class PostsController < ApplicationController
       return
     end
 
-    if ['HelpDoc', 'PolicyDoc'].include?(@post_type.name) && !check_permissions
+    if @post_type.system? && !check_permissions
       return
     end
 
@@ -144,12 +144,13 @@ class PostsController < ApplicationController
       return not_found
     end
 
+    # @post = @post.includes(:flags, flags: :post_flag_type)
     @children = if current_user&.privilege?('flag_curate')
                   Post.where(parent_id: @post.id)
                 else
                   Post.where(parent_id: @post.id).undeleted
                       .or(Post.where(parent_id: @post.id, user_id: current_user&.id).where.not(user_id: nil))
-                end.includes(:votes, :user, :comments, :license, :post_type)
+                end.includes(:votes, :user, :comments, :license, :post_type, :flags, flags: :post_flag_type)
                 .order(Post.arel_table[:id].not_eq(params[:answer]))
                 .user_sort({ term: params[:sort], default: Arel.sql('deleted ASC, score DESC, RAND()') },
                            score: Arel.sql('deleted ASC, score DESC, RAND()'), active: :last_activity,
@@ -159,6 +160,49 @@ class PostsController < ApplicationController
 
   def edit; end
 
+  # Attempts to update a given post
+  # @param post [Post] post the user is attempting to update
+  # @param user [User] user attempting to update the post
+  # @param body_rendered [String] new post body
+  # @param edit_post_params [ActionController::Parameters] edit parameters
+  # @return [Boolean] status of the operation
+  def do_update(post, user, body_rendered, **edit_post_params)
+    post.update(edit_post_params.merge(body: body_rendered,
+                                       last_edited_at: DateTime.now,
+                                       last_edited_by: user,
+                                       last_activity: DateTime.now,
+                                       last_activity_by: user))
+  end
+
+  # Attempts to update a given post network-wide. The update is manual to avoid
+  # skipping validations and fail early if at least one validation fails.
+  # @param post [Post] post from which the network push is initiated
+  # @param posts [ActiveRecord::Result] network posts to be updated
+  # @param user [User] user attempting to push updates to network
+  # @param body_rendered [String] new post body
+  # @param edit_post_params [ActionController::Parameters] edit parameters
+  # @return [Boolean] status of the operation
+  def do_update_network(post, posts, user, body_rendered, **edit_post_params)
+    update_status = true
+
+    posts.each do |network_post|
+      network_post.update(edit_post_params.merge(body: body_rendered,
+                                                 last_edited_at: DateTime.now,
+                                                 last_edited_by_id: user.id,
+                                                 last_activity: DateTime.now,
+                                                 last_activity_by_id: user.id))
+
+      if network_post.errors.any?
+        post.errors.merge!(network_post.errors)
+        update_status = false
+      end
+
+      next if update_status == true
+    end
+
+    update_status
+  end
+
   def update
     before = { body: @post.body_markdown, title: @post.title, tags: @post.tags.to_a }
     body_rendered = helpers.post_markdown(:post, :body_markdown)
@@ -169,39 +213,67 @@ class PostsController < ApplicationController
       return redirect_to post_path(@post)
     end
 
-    if current_user.privilege?('edit_posts') || current_user.is_moderator || current_user == @post.user || \
-       (@post_type.is_freely_editable && current_user.privilege?('unrestricted'))
-      if ['HelpDoc', 'PolicyDoc'].include?(@post_type.name) && (current_user.is_global_moderator || \
-         current_user.is_global_admin) && params[:network_push] == 'true'
-        posts = Post.unscoped.where(post_type_id: [PolicyDoc.post_type_id, HelpDoc.post_type_id],
-                                    doc_slug: @post.doc_slug, body: @post.body)
-        update_params = edit_post_params.to_h.merge(body: body_rendered, last_edited_at: DateTime.now,
-                                                    last_edited_by_id: current_user.id, last_activity: DateTime.now,
-                                                    last_activity_by_id: current_user.id)
-        posts.update_all(**update_params.symbolize_keys)
-        posts.each do |post|
-          PostHistory.post_edited(post, current_user, before: before[:body],
-                                  after: @post.body_markdown, comment: params[:edit_comment],
-                                  before_title: before[:title], after_title: @post.title,
-                                  before_tags: before[:tags], after_tags: @post.tags)
+    if current_user.can_update(@post, @post_type)
+      if current_user.can_push_to_network(@post_type) && params[:network_push] == 'true'
+        # post network push & post histories creation must be atomic to prevent sync issues on error
+        @post.transaction do
+          posts = Post.unscoped.where(post_type_id: [PolicyDoc.post_type_id, HelpDoc.post_type_id],
+                                      doc_slug: @post.doc_slug,
+                                      body: @post.body)
+
+          update_status = do_update_network(@post, posts, current_user, body_rendered, **edit_post_params)
+
+          if update_status
+            posts.each do |post|
+              history_entry = PostHistory.post_edited(post, current_user, before: before[:body],
+                                      after: @post.body_markdown, comment: params[:edit_comment],
+                                      before_title: before[:title], after_title: @post.title,
+                                      before_tags: before[:tags], after_tags: @post.tags)
+
+              if history_entry&.errors&.any?
+                @post.errors.merge!(history_entry.errors)
+                raise ActiveRecord::Rollback
+              end
+            end
+
+            flash[:success] = "#{helpers.pluralize(posts.to_a.size, 'post')} updated."
+            redirect_to help_path(slug: @post.doc_slug)
+          end
+
+          next
         end
-        flash[:success] = "#{helpers.pluralize(posts.to_a.size, 'post')} updated."
-        redirect_to help_path(slug: @post.doc_slug)
       else
-        if @post.update(edit_post_params.merge(body: body_rendered,
-                                               last_edited_at: DateTime.now, last_edited_by: current_user,
-                                               last_activity: DateTime.now, last_activity_by: current_user))
-          PostHistory.post_edited(@post, current_user, before: before[:body],
-                                  after: @post.body_markdown, comment: params[:edit_comment],
-                                  before_title: before[:title], after_title: @post.title,
-                                  before_tags: before[:tags], after_tags: @post.tags)
-          Rails.cache.delete "community_user/#{current_user.community_user.id}/metric/E"
-          do_draft_delete(URI(request.referer || '').path)
-          redirect_to post_path(@post)
-        else
-          render :edit, status: :bad_request
+        # post update & post history creation must be atomic to prevent sync issues on error
+        @post.transaction do
+          update_status = do_update(@post, current_user, body_rendered, **edit_post_params)
+
+          if update_status
+            history_entry = PostHistory.post_edited(@post, current_user, before: before[:body],
+                                    after: @post.body_markdown, comment: params[:edit_comment],
+                                    before_title: before[:title], after_title: @post.title,
+                                    before_tags: before[:tags], after_tags: @post.tags)
+
+            if history_entry&.errors&.any?
+              @post.errors.merge!(history_entry.errors)
+              raise ActiveRecord::Rollback
+            end
+
+            if params[:redact]
+              PostHistory.redact(@post, current_user)
+            end
+            Rails.cache.delete "community_user/#{current_user.community_user.id}/metric/E"
+            do_draft_delete(URI(request.referer || '').path)
+            redirect_to post_path(@post)
+          end
+
+          next
         end
       end
+
+      # this is only reached if we rollback the transaction or fail validations
+      if @post.errors.any?
+        render :edit, status: :bad_request
+      end
     else
       new_user = !current_user.privilege?('unrestricted')
       rate_limit = SiteSetting["RL_#{new_user ? 'NewUser' : ''}SuggestedEdits"]
@@ -232,7 +304,7 @@ class PostsController < ApplicationController
           do_draft_delete(URI(request.referer || '').path)
           redirect_to post_path(@post)
         else
-          @post.errors = edit.errors
+          @post.errors.copy!(edit.errors)
           render :edit, status: :bad_request
         end
       end
@@ -315,6 +387,12 @@ class PostsController < ApplicationController
       return
     end
 
+    if @post.post_type.is_freely_editable && !current_user&.is_moderator
+      flash[:danger] = helpers.i18ns('posts.cant_delete_community')
+      redirect_to post_path(@post)
+      return
+    end
+
     if @post.children.any? { |a| !a.deleted? && a.score >= 0.5 } && !current_user&.is_moderator
       flash[:danger] = helpers.i18ns('posts.cant_delete_responded')
       redirect_to post_path(@post)
@@ -386,15 +464,14 @@ class PostsController < ApplicationController
   end
 
   def document
-    @post = Post.unscoped.where(doc_slug: params[:slug], community_id: [RequestContext.community_id, nil]).first
-    not_found && return if @post.nil?
+    @post = Post.by_slug(params[:slug], current_user)
 
-    if @post&.help_category == '$Disabled'
-      not_found
-    end
-    if @post&.help_category == '$Moderator' && !current_user&.is_moderator
+    if @post.nil?
       not_found
     end
+
+    # Make sure we don't leak featured posts in the sidebar
+    render layout: 'without_sidebar' if @prevent_sidebar
   end
 
   def upload
@@ -417,6 +494,9 @@ class PostsController < ApplicationController
                  .order(:help_ordering, :title)
                  .group_by(&:post_type_id)
                  .transform_values { |posts| posts.group_by { |p| p.help_category.presence } }
+
+    # Make sure we don't leak featured posts in the sidebar
+    render layout: 'without_sidebar' if @prevent_sidebar
   end
 
   def change_category
@@ -506,14 +586,37 @@ class PostsController < ApplicationController
     render json: { status: 'success', success: true }
   end
 
+  # saving by-field is kept for backwards compatibility with old drafts
   def save_draft
-    key = "saved_post.#{current_user.id}.#{params[:path]}"
-    saved_at = "saved_post_at.#{current_user.id}.#{params[:path]}"
-    RequestContext.redis.set key, params[:post]
-    RequestContext.redis.set saved_at, DateTime.now.iso8601
-    RequestContext.redis.expire key, 86_400 * 7
-    RequestContext.redis.expire saved_at, 86_400 * 7
-    render json: { status: 'success', success: true, key: key }
+    expiration_time = 86_400 * 7
+
+    base_key = "saved_post.#{current_user.id}.#{params[:path]}"
+
+    [:body, :comment, :excerpt, :license, :tag_name, :tags, :title].each do |key|
+      next unless params.key?(key)
+
+      key_name = [:body, :saved_at].include?(key) ? base_key : "#{base_key}.#{key}"
+
+      if key == :tags
+        valid_tags = params[key]&.select(&:present?)
+
+        RequestContext.redis.del(key_name)
+
+        if valid_tags.present?
+          RequestContext.redis.sadd(key_name, valid_tags)
+        end
+      else
+        RequestContext.redis.set(key_name, params[key])
+      end
+
+      RequestContext.redis.expire(key_name, expiration_time)
+    end
+
+    saved_at_key = "saved_post_at.#{current_user.id}.#{params[:path]}"
+    RequestContext.redis.set(saved_at_key, DateTime.now.iso8601)
+    RequestContext.redis.expire(saved_at_key, expiration_time)
+
+    render json: { status: 'success', success: true, key: base_key }
   end
 
   def delete_draft
@@ -580,9 +683,13 @@ class PostsController < ApplicationController
   end
 
   def do_draft_delete(path)
-    key = "saved_post.#{current_user.id}.#{path}"
-    saved_at = "saved_post_at.#{current_user.id}.#{path}"
-    RequestContext.redis.del key, saved_at
+    keys = [:body, :comment, :excerpt, :license, :saved_at, :tags, :tag_name, :title].map do |key|
+      pfx = key == :saved_at ? 'saved_post_at' : 'saved_post'
+      base = "#{pfx}.#{current_user.id}.#{path}"
+      [:body, :saved_at].include?(key) ? base : "#{base}.#{key}"
+    end
+
+    RequestContext.redis.del(*keys)
   end
 end
 # rubocop:enable Metrics/MethodLength
diff --git a/app/controllers/reactions_controller.rb b/app/controllers/reactions_controller.rb
index 8899d8fa44979fc6d0eb82843b8c799774d6cd0d..23acd23110de3349c5c7868962ed3d50285d2c97 100644
--- a/app/controllers/reactions_controller.rb
+++ b/app/controllers/reactions_controller.rb
@@ -33,13 +33,20 @@ class ReactionsController < ApplicationController
 
     reaction = Reaction.new(user: current_user, post: @post, reaction_type: reaction_type, comment: comment)
 
-    ActiveRecord::Base.transaction do
-      thread&.save!
-      comment&.save!
-      reaction.save!
-    end
+    begin
+      ActiveRecord::Base.transaction do
+        thread&.save!
+        comment&.save!
+        reaction.save!
+      end
 
-    render json: { status: 'success' }
+      render json: { status: 'success' }
+    rescue
+      render json: { status: 'failed',
+                     message: "Could not create comment thread: #{(thread&.errors&.full_messages.to_a \
+                                                        + comment&.errors&.full_messages.to_a \
+                                                        + reaction&.errors&.full_messages.to_a).join(', ')}" }
+    end
   end
 
   def retract
diff --git a/app/controllers/search_controller.rb b/app/controllers/search_controller.rb
index f2e6673efbd5c18db8fe41d4424d267e1a915483..202a3636bd45cb9dfa4267d5c99636b318553a1c 100644
--- a/app/controllers/search_controller.rb
+++ b/app/controllers/search_controller.rb
@@ -1,20 +1,11 @@
 class SearchController < ApplicationController
   def search
-    @posts = if params[:search].present?
-               search_data = helpers.parse_search(params[:search])
-               posts = (current_user&.is_moderator || current_user&.is_admin ? Post : Post.undeleted)
-                       .qa_only.list_includes
-               posts = helpers.qualifiers_to_sql(search_data[:qualifiers], posts)
-               posts = posts.paginate(page: params[:page], per_page: 25)
+    @posts, @qualifiers = helpers.search_posts
+
+    @signed_out_me = @qualifiers.any? { |q| q[:param] == :user && q[:user_id].nil? }
+
+    @active_filter = helpers.active_filter
 
-               if search_data[:search].present?
-                 posts.search(search_data[:search]).user_sort({ term: params[:sort], default: :search_score },
-                                                              relevance: :search_score, score: :score, age: :created_at)
-               else
-                 posts.user_sort({ term: params[:sort], default: :score },
-                                 score: :score, age: :created_at)
-               end
-             end
     @count = begin
       @posts&.count
     rescue
diff --git a/app/controllers/site_settings_controller.rb b/app/controllers/site_settings_controller.rb
index 9134f8da7cfd2171dfce1143231b5505c5164a6d..256b54ecc0d17f803291dffc451b7a167a2c9b12 100644
--- a/app/controllers/site_settings_controller.rb
+++ b/app/controllers/site_settings_controller.rb
@@ -4,6 +4,14 @@ class SiteSettingsController < ApplicationController
   before_action :verify_admin
   before_action :verify_global_admin, only: [:global]
 
+  # Checks if a given user has access to site settings on a given community
+  # @param [User] user user to check access for
+  # @param [String, nil] community_id id of the community to check access on
+  # @return [Boolean]
+  def access?(user, community_id)
+    community_id.present? || user.is_global_admin
+  end
+
   def index
     # The weird argument to sort_by here sorts without throwing errors on nil values -
     # see https://stackoverflow.com/a/35539062/3160466. 0:1,c sorts nil last, to switch
@@ -26,8 +34,41 @@ class SiteSettingsController < ApplicationController
     render json: @setting&.as_json&.merge(typed: @setting.typed)
   end
 
+  # Adds an audit log for a given site setting update event
+  # @param [User] user initiating user
+  # @param [SiteSetting] before current site setting
+  # @param [SiteSetting] after updated site setting
+  # @return [void]
+  def audit_update(user, before, after)
+    AuditLog.admin_audit(event_type: 'setting_update',
+                         related: after,
+                         user: user,
+                         comment: "from <<SiteSetting #{before}>>\nto <<SiteSetting #{after.attributes_print}>>")
+  end
+
+  # Deletes cache for a given site setting for a given community
+  # @param [SiteSetting] setting site setting to clear cache for
+  # @param [String, nil] community_id community id to clear cache for
+  # @return [Boolean]
+  def clear_cache(setting, community_id)
+    Rails.cache.delete("SiteSettings/#{community_id}/#{setting.name}", include_community: false)
+  end
+
+  # Actually creates a given site setting
+  # @param [SiteSetting] setting site setting to create
+  # @param [String, nil] community_id community id to create a setting for
+  # @return [SiteSetting]
+  def do_create(setting, community_id)
+    SiteSetting.create(name: setting.name,
+                       community_id: community_id,
+                       value: '',
+                       value_type: setting.value_type,
+                       category: setting.category,
+                       description: setting.description)
+  end
+
   def update
-    if params[:community_id].blank? && !current_user.is_global_admin
+    unless access?(current_user, params[:community_id])
       not_found
       return
     end
@@ -36,20 +77,28 @@ class SiteSettingsController < ApplicationController
                  matches = SiteSetting.unscoped.where(community_id: RequestContext.community_id, name: params[:name])
                  if matches.count.zero?
                    global = SiteSetting.unscoped.where(community_id: nil, name: params[:name]).first
-                   SiteSetting.create(name: global.name, community_id: RequestContext.community_id, value: '',
-                                      value_type: global.value_type, category: global.category,
-                                      description: global.description)
+                   do_create(global, RequestContext.community_id)
                  else
                    matches.first
                  end
                else
                  SiteSetting.unscoped.where(community_id: nil, name: params[:name]).first
                end
+
     before = @setting.attributes_print
+
     @setting.update(setting_params)
-    AuditLog.admin_audit(event_type: 'setting_update', related: @setting, user: current_user,
-                         comment: "from <<SiteSetting #{before}>>\nto <<SiteSetting #{@setting.attributes_print}>>")
-    Rails.cache.delete "SiteSettings/#{RequestContext.community_id}/#{@setting.name}"
+
+    audit_update(current_user, before, @setting)
+
+    if @setting.global?
+      Community.all.each do |c|
+        clear_cache(@setting, c.id)
+      end
+    else
+      clear_cache(@setting, RequestContext.community_id)
+    end
+
     render json: { status: 'OK', setting: @setting&.as_json&.merge(typed: @setting.typed) }
   end
 
diff --git a/app/controllers/suggested_edit_controller.rb b/app/controllers/suggested_edit_controller.rb
index 34e0bbc7bcbb10875d90623dab04c7d0e58cf6f8..c724df4a5af3e997dc4c9eaeaeafeb2ea1a548b9 100644
--- a/app/controllers/suggested_edit_controller.rb
+++ b/app/controllers/suggested_edit_controller.rb
@@ -30,21 +30,56 @@ class SuggestedEditController < ApplicationController
       return
     end
 
-    opts = { before: @post.body_markdown, after: @edit.body_markdown, comment: @edit.comment,
-             before_title: @post.title, after_title: @edit.title, before_tags: @post.tags, after_tags: @edit.tags }
-
-    before = { before_body: @post.body, before_body_markdown: @post.body_markdown, before_tags_cache: @post.tags_cache,
-               before_tags: @post.tags.to_a, before_title: @post.title }
+    comment = params[:comment].present? && !params[:comment].empty? ? params[:comment] : @edit.comment
+
+    # The to_a / dup methods called on the tags for `opts` and `before` are necessary.
+    # We need to work on a copy of them, because we update the post before the edit, which will change their values.
+    # (We would otherwise be pointing to the same instance, and only see the updated version).
+    opts = { before: @post.body_markdown,
+             after: @edit.body_markdown,
+             comment: comment,
+             before_title: @post.title,
+             after_title: @edit.title,
+             before_tags: @post.tags.to_a,
+             after_tags: @edit.tags }
+
+    before = { before_body: @post.body,
+               before_body_markdown: @post.body_markdown,
+               before_tags_cache: @post.tags_cache.dup,
+               before_tags: @post.tags.to_a,
+               before_title: @post.title }
+
+    @post.transaction do
+      post_update_status = @post.update(applied_details)
+
+      if post_update_status
+        edit_update_status = @edit.update(before.merge(active: false,
+                                                       accepted: true,
+                                                       comment: comment,
+                                                       rejected_comment: '',
+                                                       decided_at: DateTime.now,
+                                                       decided_by: current_user,
+                                                       updated_at: DateTime.now))
+
+        if @edit.errors.any?
+          @post.errors.merge!(@edit.errors)
+          raise ActiveRecord::Rollback
+        end
+
+        if edit_update_status
+          PostHistory.post_edited(@post, @edit.user, **opts)
+          AbilityQueue.add(@edit.user, "Suggested Edit Approved ##{@edit.id}")
+        end
+      end
+
+      next
+    end
 
-    if @post.update(applied_details)
-      @edit.update(before.merge(active: false, accepted: true, rejected_comment: '', decided_at: DateTime.now,
-                                decided_by: current_user, updated_at: DateTime.now))
-      PostHistory.post_edited(@post, @edit.user, **opts)
+    if @post.errors.any?
+      render json: { status: 'error', message: @post.errors.full_messages.join(', ') }, status: :bad_request
+    else
       flash[:success] = 'Edit approved successfully.'
-      AbilityQueue.add(@edit.user, "Suggested Edit Approved ##{@edit.id}")
       render json: { status: 'success', redirect_url: post_path(@post) }
-    else
-      render json: { status: 'error', message: @post.errors.full_messages.join(', ') }, status: :bad_request
     end
   end
 
diff --git a/app/controllers/tags_controller.rb b/app/controllers/tags_controller.rb
index f1e6dac20a06345642f6d5b5e0b0f7e923afe973..98f1333294cebb405d223c71a3b2f4cc996559ac 100644
--- a/app/controllers/tags_controller.rb
+++ b/app/controllers/tags_controller.rb
@@ -2,7 +2,8 @@ class TagsController < ApplicationController
   before_action :authenticate_user!, only: [:new, :create, :edit, :update, :rename, :merge, :select_merge]
   before_action :set_category, except: [:index]
   before_action :set_tag, only: [:show, :edit, :update, :children, :rename, :merge, :select_merge, :nuke, :nuke_warning]
-  before_action :verify_moderator, only: [:new, :create, :rename, :merge, :select_merge]
+  before_action :verify_tag_editor, only: [:new, :create]
+  before_action :verify_moderator, only: [:rename, :merge, :select_merge]
   before_action :verify_admin, only: [:nuke, :nuke_warning]
 
   def index
@@ -13,10 +14,10 @@ class TagsController < ApplicationController
               (@tag_set&.tags || Tag).search(params[:term])
             else
               (@tag_set&.tags || Tag.all).order(:name)
-            end.paginate(page: params[:page], per_page: 50)
+            end.includes(:tag_synonyms).paginate(page: params[:page], per_page: 50)
     respond_to do |format|
       format.json do
-        render json: @tags
+        render json: @tags.to_json(include: { tag_synonyms: { only: :name } })
       end
     end
   end
@@ -26,18 +27,27 @@ class TagsController < ApplicationController
     @tags = if params[:q].present?
               @tag_set.tags.search(params[:q])
             elsif params[:hierarchical].present?
-              @tag_set.tags_with_paths.order(:path)
+              @tag_set.with_paths(params[:no_excerpt])
             elsif params[:no_excerpt].present?
-              @tag_set.tags.where(excerpt: '').or(@tag_set.tags.where(excerpt: nil))
-                      .order(Arel.sql('COUNT(posts.id) DESC'))
+              @tag_set.tags.where(excerpt: ['', nil])
             else
-              @tag_set.tags.order(Arel.sql('COUNT(posts.id) DESC'))
+              @tag_set&.tags
             end
-    @count = @tags.count
+
     table = params[:hierarchical].present? ? 'tags_paths' : 'tags'
-    @tags = @tags.left_joins(:posts).group(Arel.sql("#{table}.id"))
-                 .select(Arel.sql("#{table}.*, COUNT(posts.id) AS post_count"))
-                 .paginate(per_page: 96, page: params[:page])
+
+    @tags = @tags&.left_joins(:posts)
+                 &.group(Arel.sql("#{table}.id"))
+                 &.select(Arel.sql("#{table}.*, COUNT(DISTINCT IF(posts.deleted = 0, posts.id, NULL)) AS post_count"))
+                 &.paginate(per_page: 96, page: params[:page])
+
+    @tags = if params[:hierarchical].present?
+              @tags&.order(:path)
+            else
+              @tags&.order(Arel.sql('COUNT(posts.id) DESC'))
+            end
+
+    @count = @tags&.length || 0
   end
 
   def show
@@ -49,8 +59,9 @@ class TagsController < ApplicationController
               else
                 @tag.all_children + [@tag.id]
               end
-    post_ids = helpers.post_ids_for_tags(tag_ids)
-    @posts = Post.where(id: post_ids).undeleted.where(post_type_id: @category.display_post_types)
+    displayed_post_types = @tag.tag_set.categories.map(&:display_post_types).flatten
+    @posts = Post.joins(:tags).where(id: PostsTag.select(:post_id).distinct.where(tag_id: tag_ids))
+                 .undeleted.where(post_type_id: displayed_post_types)
                  .includes(:post_type, :tags).list_includes.paginate(page: params[:page], per_page: 50)
                  .order(sort_param)
     respond_to do |format|
@@ -61,6 +72,7 @@ class TagsController < ApplicationController
 
   def new
     @tag = Tag.new
+    @tag.tag_synonyms.build
   end
 
   def create
@@ -76,6 +88,7 @@ class TagsController < ApplicationController
 
   def edit
     check_your_privilege('edit_tags', nil, true)
+    @tag.tag_synonyms.build
   end
 
   def update
@@ -123,57 +136,68 @@ class TagsController < ApplicationController
 
     @subordinate = Tag.find params[:merge_with_id]
 
-    AuditLog.moderator_audit event_type: 'tag_merge', related: @primary, user: current_user,
-                             comment: "#{@subordinate.name} (#{@subordinate.id}) into #{@primary.name} (#{@primary.id})"
-
-    # Take the tag off posts
-    posts_sql = 'UPDATE posts INNER JOIN posts_tags ON posts.id = posts_tags.post_id ' \
-                'SET posts.tags_cache = REPLACE(posts.tags_cache, ?, ?) ' \
-                'WHERE posts_tags.tag_id = ?'
-    exec([posts_sql, "\n- #{@subordinate.name}", "\n- #{@primary.name}", @subordinate.id])
-
-    # Break hierarchies
-    tags_sql = 'UPDATE tags SET parent_id = NULL WHERE parent_id = ?'
-    exec([tags_sql, @subordinate.id])
-
-    # Remove references to the tag
-    sql = 'UPDATE IGNORE $TABLENAME SET tag_id = ? WHERE tag_id = ?'
-    exec([sql.gsub('$TABLENAME', 'posts_tags'), @primary.id, @subordinate.id])
-    exec([sql.gsub('$TABLENAME', 'categories_moderator_tags'), @primary.id, @subordinate.id])
-    exec([sql.gsub('$TABLENAME', 'categories_required_tags'), @primary.id, @subordinate.id])
-    exec([sql.gsub('$TABLENAME', 'categories_topic_tags'), @primary.id, @subordinate.id])
-    exec([sql.gsub('$TABLENAME', 'post_history_tags'), @primary.id, @subordinate.id])
-    exec([sql.gsub('$TABLENAME', 'suggested_edits_tags'), @primary.id, @subordinate.id])
-    exec([sql.gsub('$TABLENAME', 'suggested_edits_before_tags'), @primary.id, @subordinate.id])
-
-    # Nuke it from orbit
-    @subordinate.destroy
+    Post.transaction do
+      AuditLog.moderator_audit event_type: 'tag_merge', related: @primary, user: current_user, comment:
+        "#{@subordinate.name} (#{@subordinate.id}) into #{@primary.name} (#{@primary.id})"
+
+      # Replace subordinate with primary, except when a post already has primary (to avoid giving them a duplicate tag)
+      posts_sql = 'UPDATE posts INNER JOIN posts_tags ON posts.id = posts_tags.post_id ' \
+                  'SET posts.tags_cache = REPLACE(posts.tags_cache, ?, ?) ' \
+                  'WHERE posts_tags.tag_id = ? ' \
+                  'AND posts_tags.post_id NOT IN (SELECT post_id FROM posts_tags WHERE tag_id = ?)'
+      exec_sql([posts_sql, "\n- #{@subordinate.name}\n", "\n- #{@primary.name}\n", @subordinate.id, @primary.id])
+
+      # Remove the subordinate tag from posts that still have it (the ones that were excluded from our previous query)
+      posts2_sql = 'UPDATE posts INNER JOIN posts_tags ON posts.id = posts_tags.post_id ' \
+                   'SET posts.tags_cache = REPLACE(posts.tags_cache, ?, ?) ' \
+                   'WHERE posts_tags.tag_id = ?'
+      exec_sql([posts2_sql, "\n- #{@subordinate.name}\n", "\n", @subordinate.id])
+
+      # Break hierarchies
+      tags_sql = 'UPDATE tags SET parent_id = NULL WHERE parent_id = ?'
+      exec_sql([tags_sql, @subordinate.id])
+
+      # Remove references to the tag
+      sql = 'UPDATE IGNORE $TABLENAME SET tag_id = ? WHERE tag_id = ?'
+      exec_sql([sql.gsub('$TABLENAME', 'posts_tags'), @primary.id, @subordinate.id])
+      exec_sql([sql.gsub('$TABLENAME', 'categories_moderator_tags'), @primary.id, @subordinate.id])
+      exec_sql([sql.gsub('$TABLENAME', 'categories_required_tags'), @primary.id, @subordinate.id])
+      exec_sql([sql.gsub('$TABLENAME', 'categories_topic_tags'), @primary.id, @subordinate.id])
+      exec_sql([sql.gsub('$TABLENAME', 'post_history_tags'), @primary.id, @subordinate.id])
+      exec_sql([sql.gsub('$TABLENAME', 'suggested_edits_tags'), @primary.id, @subordinate.id])
+      exec_sql([sql.gsub('$TABLENAME', 'suggested_edits_before_tags'), @primary.id, @subordinate.id])
+
+      # Nuke it from orbit
+      @subordinate.destroy
+    end
 
     flash[:success] = "Merged #{@subordinate.name} into #{@primary.name}."
     redirect_to tag_path(id: @category.id, tag_id: @primary.id)
   end
 
   def nuke
-    AuditLog.admin_audit event_type: 'tag_nuke', related: @tag, user: current_user,
-                         comment: "#{@tag.name} (#{@tag.id})"
-
-    tables = ['posts_tags', 'categories_moderator_tags', 'categories_required_tags', 'categories_topic_tags',
-              'post_history_tags', 'suggested_edits_tags', 'suggested_edits_before_tags']
-
-    # Remove tag from caches
-    caches_sql = 'UPDATE posts INNER JOIN posts_tags ON posts.id = posts_tags.post_id ' \
-                 'SET posts.tags_cache = REPLACE(posts.tags_cache, ?, ?) ' \
-                 'WHERE posts_tags.tag_id = ?'
-    exec([caches_sql, "\n- #{@tag.name}", '', @tag.id])
-
-    # Delete all references to the tag
-    tables.each do |tbl|
-      sql = "DELETE FROM #{tbl} WHERE tag_id = ?"
-      exec([sql, @tag.id])
-    end
+    Post.transaction do
+      AuditLog.admin_audit event_type: 'tag_nuke', related: @tag, user: current_user,
+                           comment: "#{@tag.name} (#{@tag.id})"
+
+      tables = ['posts_tags', 'categories_moderator_tags', 'categories_required_tags', 'categories_topic_tags',
+                'post_history_tags', 'suggested_edits_tags', 'suggested_edits_before_tags']
+
+      # Remove tag from caches
+      caches_sql = 'UPDATE posts INNER JOIN posts_tags ON posts.id = posts_tags.post_id ' \
+                   'SET posts.tags_cache = REPLACE(posts.tags_cache, ?, ?) ' \
+                   'WHERE posts_tags.tag_id = ?'
+      exec_sql([caches_sql, "\n- #{@tag.name}\n", "\n", @tag.id])
+
+      # Delete all references to the tag
+      tables.each do |tbl|
+        sql = "DELETE FROM #{tbl} WHERE tag_id = ?"
+        exec_sql([sql, @tag.id])
+      end
 
-    # Nuke it
-    @tag.destroy
+      # Nuke it
+      @tag.destroy
+    end
 
     flash[:success] = "Deleted #{@tag.name}"
     redirect_to category_tags_path(@category)
@@ -192,10 +216,29 @@ class TagsController < ApplicationController
   end
 
   def tag_params
-    params.require(:tag).permit(:excerpt, :wiki_markdown, :parent_id, :name)
+    params.require(:tag).permit(:excerpt, :wiki_markdown, :parent_id, :name,
+                                tag_synonyms_attributes: [:id, :name, :_destroy])
   end
 
-  def exec(sql_array)
+  def exec_sql(sql_array)
     ApplicationRecord.connection.execute(ActiveRecord::Base.sanitize_sql_array(sql_array))
   end
+
+  def verify_tag_editor
+    unless user_signed_in? && (current_user.privilege?(:edit_tags) ||
+      current_user.is_moderator ||
+      current_user.is_admin)
+      respond_to do |format|
+        format.html do
+          render 'errors/not_found', layout: 'without_sidebar', status: :not_found
+        end
+        format.json do
+          render json: { status: 'failed', success: false, errors: ['not_found'] }, status: :not_found
+        end
+      end
+
+      return false
+    end
+    true
+  end
 end
diff --git a/app/controllers/tour_controller.rb b/app/controllers/tour_controller.rb
index b7101473dfa66aade9a2fe6ed23568341b9aa56b..13ea7f27efe8c9135645c43e07a964d860e20a98 100644
--- a/app/controllers/tour_controller.rb
+++ b/app/controllers/tour_controller.rb
@@ -5,7 +5,9 @@ class TourController < ApplicationController
 
   def question1; end
 
-  def question2; end
+  def question2
+    @tagset_id = TagSet.find_by(name: 'Tour')&.id || -1
+  end
 
   def question3; end
 
diff --git a/app/controllers/two_factor_controller.rb b/app/controllers/two_factor_controller.rb
index 4d85ecba2d78761f3a73c238f5fbbc1bcba3c423..fea0dc9c66cf0308372000b42efaf43855849f0a 100644
--- a/app/controllers/two_factor_controller.rb
+++ b/app/controllers/two_factor_controller.rb
@@ -5,11 +5,18 @@ class TwoFactorController < ApplicationController
   def tf_status; end
 
   def enable_2fa
+    if current_user.sso_profile.present? && !SiteSetting['Enable2FAForSsoUsers']
+      flash[:danger] = 'You cannot enable 2FA because you sign in through SSO.'
+      redirect_to two_factor_status_path
+      return
+    end
+
     case params[:method]
     when 'app'
-      secret = ROTP::Base32.random
-      current_user.update(two_factor_token: secret, two_factor_method: 'app')
-      totp = ROTP::TOTP.new(secret, issuer: 'codidact.com')
+      @secret = ROTP::Base32.random
+      current_user.update(two_factor_token: @secret, two_factor_method: 'app',
+                          backup_2fa_code: SecureRandom.alphanumeric(24))
+      totp = ROTP::TOTP.new(@secret, issuer: 'codidact.com')
       uri = totp.provisioning_uri("#{current_user.id}@users-2fa.codidact.com")
       qr_svg = RQRCode::QRCode.new(uri).as_svg
       @qr_uri = "data:image/svg+xml;base64,#{Base64.encode64(qr_svg)}"
@@ -52,7 +59,7 @@ class TwoFactorController < ApplicationController
 
     totp = ROTP::TOTP.new(current_user.two_factor_token)
     if totp.verify(params[:code], drift_behind: 15, drift_ahead: 15)
-      current_user.update(two_factor_token: nil, enabled_2fa: false)
+      current_user.update(two_factor_token: nil, enabled_2fa: false, backup_2fa_code: nil)
       AuditLog.user_history(event_type: 'two_factor_disabled', related: current_user)
       flash[:success] = 'Success! 2FA has been disabled on your account.'
       redirect_to two_factor_status_path
@@ -81,4 +88,13 @@ class TwoFactorController < ApplicationController
     flash[:success] = 'Success! 2FA has been disabled on your account.'
     redirect_to two_factor_status_path
   end
+
+  def show_backup_code
+    totp = ROTP::TOTP.new(current_user.two_factor_token)
+    if totp.verify(params[:code], drift_behind: 15, drift_ahead: 15)
+      render json: { status: 'success', code: current_user.backup_2fa_code }
+    else
+      render json: { status: 'error', message: 'Wrong code - please try again.' }, status: 401
+    end
+  end
 end
diff --git a/app/controllers/users/registrations_controller.rb b/app/controllers/users/registrations_controller.rb
index 72e61c7b50c798735a9c9d6ba66da90e912f46b6..00f0722b8d9fec5a4402c1986a1b8a6be58cd075 100644
--- a/app/controllers/users/registrations_controller.rb
+++ b/app/controllers/users/registrations_controller.rb
@@ -1,7 +1,18 @@
 class Users::RegistrationsController < Devise::RegistrationsController
   protected
 
+  layout 'without_sidebar', only: :edit
+
+  before_action :check_sso, only: :update
+
   def after_update_path_for(resource)
     edit_user_registration_path(resource)
   end
+
+  def check_sso
+    if current_user && current_user.sso_profile.present?
+      flash['danger'] = 'You sign in with SSO, so updating your email/password is not possible.'
+      redirect_to edit_user_registration_path
+    end
+  end
 end
diff --git a/app/controllers/users/saml_sessions_controller.rb b/app/controllers/users/saml_sessions_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..ff0ca3e58f7452f814ea666f11cd8c68a3e12fa4
--- /dev/null
+++ b/app/controllers/users/saml_sessions_controller.rb
@@ -0,0 +1,200 @@
+class Users::SamlSessionsController < Devise::SamlSessionsController
+  # Called when someone is redirected to sign into the application using SSO/SAML.
+  def new
+    # If this is not the base community, then redirect them there for the sign in
+    base = base_community
+    if base.id != RequestContext.community_id
+      redirect_to "//#{base.host}#{sign_in_request_from_other_path(RequestContext.community_id)}",
+                  allow_other_host: true
+      return
+    end
+
+    # If we are the base community, use normal behavior
+    super
+  end
+
+  # This method is almost the same code as the Users::SessionsController#create, and any changes
+  # made here should probably also be applied over there.
+  def create
+    super do |user|
+      return unless post_sign_in(user, false)
+
+      # SSO Only - Redirect to filler endpoint to actually get the clients cookie values (not sent to us here).
+      # We need to check cookies because we may be signing in for another community.
+      redirect_to after_sign_in_check_path
+      return
+    end
+  end
+
+  # On the initial return from the SSO the client does not send along its cookies (CORS/CSRF/XSS protections).
+  # Instead, we redirect the user after the sign-in to this endpoint, such that we get their cookies.
+  # Then we can check whether we were supposed to sign them in for a different community.
+  def after_sign_in_check
+    if cookies.encrypted[:signing_in_for].present? &&
+       cookies.encrypted[:signing_in_for] != RequestContext.community_id
+      handle_sign_in_for_other_community(current_user)
+      return
+    end
+
+    return unless post_sign_in(current_user, true)
+
+    redirect_to after_sign_in_path_for(current_user)
+  end
+
+  # Another community requests to sign in via this community.
+  def sign_in_request_from_other
+    # Check whether the requested community actually exists
+    unless Community.exists?(params[:id])
+      raise ArgumentError, 'User is trying to sign in to non-existing community'
+    end
+
+    # Store in a cookie which community we are signing in for such that we can redirect back after the sign in.
+    cookies.encrypted[:signing_in_for] = {
+      value: params[:id],
+      httponly: true,
+      expires: 15.minutes.from_now
+    }
+
+    # If already signed in, sign them in in the other community as well. Otherwise redirect to SAML sign in.
+    if user_signed_in?
+      handle_sign_in_for_other_community(current_user)
+    else
+      redirect_to new_saml_user_session_path
+    end
+  end
+
+  # User was signed in at the base community, now sign in here.
+  def sign_in_return_from_base
+    # Figure out which user was signed in.
+    # If we get a blank result then the message is either too old or the user messed with it.
+    user_info = decrypt_user_info(params[:message])
+    if user_info.blank?
+      flash[:notice] = nil
+      flash[:danger] = 'Something went wrong signing in, please try again.'
+      redirect_to root_path
+    end
+
+    # Determine the user we are trying to sign in as and report error if we can't
+    user = User.find(user_info)
+    if user.nil?
+      flash[:notice] = nil
+      flash[:danger] = 'Something went wrong signing in, please contact support.'
+      redirect_to root_path
+    end
+
+    # Actually sign in the user and handle the post-sign-in behavior
+    sign_in(user)
+    return unless post_sign_in(user, true)
+
+    # Finish with default devise behavior for sign ins
+    redirect_to after_sign_in_path_for(user)
+  end
+
+  private
+
+  # After a sign in, this method is called to check whether special conditions apply to the user.
+  # The user may be signed out by this method.
+  #
+  # In general, this method should have similar behavior to the Users::SessionsController#post_sign_in method.
+  # If you make changes here, you may also have to update that method.
+  #
+  # @param user [User]
+  # @param final_destination [Boolean] whether the current community is the one the user is trying to sign into
+  # @return [Boolean] false if the user was redirected by this
+  def post_sign_in(user, final_destination = false)
+    # If the user was banished, let them know non-specifically.
+    if user.deleted? || user.community_user&.deleted?
+      # The IDP already confirmed the sign in, so we can't fool the user any more that their credentials were incorrect.
+      sign_out user
+      flash[:notice] = nil
+      flash[:danger] = 'We could not sign you in because of an issue with your account.'
+      redirect_to root_path
+      return false
+    end
+
+    # Enforce 2fa if enabled for SSO users
+    if SiteSetting['Enable2FAForSsoUsers'] && user.enabled_2fa
+      if final_destination
+        handle_2fa_login(user)
+        return false
+      else
+        # User needs to do 2FA, but we are (potentially) signing in for a different community.
+        # Sign them out and continue the sign in process, when they reach the final destination we will do 2FA there.
+        sign_out user
+        return true
+      end
+    end
+
+    true
+  end
+
+  def handle_2fa_login(user, host = nil)
+    host ||= request.hostname
+    sign_out user
+    case user.two_factor_method
+    when 'app'
+      id = user.id
+      Users::SessionsController.first_factor << id
+      redirect_to login_verify_2fa_path(uid: id)
+    when 'email'
+      TwoFactorMailer.with(user: user, host: host).login_email.deliver_now
+      flash[:notice] = nil
+      flash[:info] = 'Please check your email inbox for a link to sign in.'
+      redirect_to after_sign_in_path_for(user)
+    end
+  end
+
+  # Handles a successful sign in at the base community when it was requested to do from another community.
+  # @param user [User]
+  def handle_sign_in_for_other_community(user)
+    # Determine which community we are signing in for, log out if not found (user messed with encrypted cookie/expired)
+    community = Community.find(cookies.encrypted[:signing_in_for])
+    if community.nil?
+      sign_out(user)
+      flash[:notice] = nil
+      flash[:danger] = 'Something went wrong trying to sign you in.'
+      redirect_to root_path
+      return
+    end
+
+    # Clear the cookie to prevent future issues
+    cookies.delete(:signing_in_for)
+
+    # We signed in for a different community, we need to send them back to the original community with encrypted
+    # info about who signed in.
+    encrypted_user_info = encrypt_user_info(user)
+    redirect_to "//#{community.host}#{sign_in_return_from_base_path}?message=#{CGI.escape(encrypted_user_info)}",
+                allow_other_host: true
+  end
+
+  # Encrypts user information for sending them to a different community.
+  # @param user [User]
+  def encrypt_user_info(user)
+    len = ActiveSupport::MessageEncryptor.key_len - 1
+    key = Rails.application.secrets.secret_key_base || Rails.application.credentials.secret_key_base
+    crypt = ActiveSupport::MessageEncryptor.new(key[0..len])
+    crypt.encrypt_and_sign(user.id, expires_in: 1.minute)
+  end
+
+  # Decrypts the user information when received at a different community.
+  # @param data
+  def decrypt_user_info(data)
+    len = ActiveSupport::MessageEncryptor.key_len - 1
+    key = Rails.application.secrets.secret_key_base || Rails.application.credentials.secret_key_base
+    crypt = ActiveSupport::MessageEncryptor.new(key[0..len])
+    crypt.decrypt_and_verify(data)
+  end
+
+  # @return [Community] the community to which the SSO is connected, and which must be used to sign in via.
+  def base_community
+    uri = URI.parse(Devise.saml_config.assertion_consumer_service_url)
+    host = if uri.port != 80 && uri.port != 443 && !uri.port.nil?
+             "#{uri.hostname}:#{uri.port}"
+           else
+             uri.hostname
+           end
+    Community.find_by(host: host) || Community.first
+  rescue
+    Community.first
+  end
+end
diff --git a/app/controllers/users/sessions_controller.rb b/app/controllers/users/sessions_controller.rb
index da34a1d181ed2726c122e22b898b219794613cb6..9719fd7452d9af744fcf45953b56494857a84682 100644
--- a/app/controllers/users/sessions_controller.rb
+++ b/app/controllers/users/sessions_controller.rb
@@ -1,42 +1,12 @@
 class Users::SessionsController < Devise::SessionsController
   protect_from_forgery except: [:create]
 
-  @@first_factor = []
+  mattr_accessor :first_factor, default: [], instance_writer: false, instance_reader: false
 
+  # Any changes made here may also require changes to Users::SamlSessionsController#create.
   def create
     super do |user|
-      if user.deleted?
-        sign_out user
-        flash[:notice] = nil
-        flash[:danger] = 'Invalid Email or password.'
-        render :new
-        return
-      end
-
-      if user.community_user&.deleted?
-        sign_out user
-        flash[:notice] = nil
-        flash[:danger] = 'Your profile on this community has been deleted.'
-        render :new
-        return
-      end
-
-      if user.present? && user.enabled_2fa
-        sign_out user
-        case user.two_factor_method
-        when 'app'
-          id = user.id
-          @@first_factor << id
-          redirect_to login_verify_2fa_path(uid: id)
-          return
-        when 'email'
-          TwoFactorMailer.with(user: user, host: request.hostname).login_email.deliver_now
-          flash[:notice] = nil
-          flash[:info] = 'Please check your email inbox for a link to sign in.'
-          redirect_to root_path
-          return
-        end
-      end
+      return unless post_sign_in(user)
     end
   end
 
@@ -51,16 +21,26 @@ class Users::SessionsController < Devise::SessionsController
     end
 
     totp = ROTP::TOTP.new(target_user.two_factor_token)
-    if totp.verify(params[:code], drift_ahead: 15, drift_behind: 15)
+    if totp.verify(params[:code], drift_ahead: 15, drift_behind: 15) || params[:code] == target_user.backup_2fa_code
       if @@first_factor.include? params[:uid].to_i
+        if params[:code] == target_user.backup_2fa_code
+          target_user.update(enabled_2fa: false, two_factor_token: nil, backup_2fa_code: nil)
+          flash[:warning] = 'Two-factor authentication has been disabled for your account because you signed in with ' \
+                            'a backup code. Please re-configure two-factor authentication via your profile.'
+        end
+
         AuditLog.user_history(event_type: 'two_factor_success', related: target_user)
         @@first_factor.delete params[:uid].to_i
         flash[:info] = 'Signed in successfully.'
-        sign_in_and_redirect User.find(params[:uid])
+        sign_in_and_redirect target_user
       else
         AuditLog.user_history(event_type: 'two_factor_fail', related: target_user, comment: 'first factor not present')
         flash[:danger] = "You haven't entered your password yet."
-        redirect_to new_session_path(target_user)
+        if devise_sign_in_enabled?
+          redirect_to new_session_path(target_user)
+        else
+          redirect_to new_saml_user_session_path(target_user)
+        end
       end
     else
       AuditLog.user_history(event_type: 'two_factor_fail', related: target_user, comment: 'wrong code')
@@ -68,4 +48,65 @@ class Users::SessionsController < Devise::SessionsController
       redirect_to login_verify_2fa_path(uid: params[:uid])
     end
   end
+
+  private
+
+  # After a sign in, this method is called to check whether special conditions apply to the user.
+  # The user may be signed out by this method.
+  #
+  # In general, this method should have similar behavior to the Users::SamlSessionsController#post_sign_in method.
+  # If you make changes here, you may also have to update that method.
+  # @param user [User]
+  # @return [Boolean] false if the handling by the calling method should be stopped
+  def post_sign_in(user)
+    # For a deleted user (banished), tell them non-specifically that there was a mistake with their credentials.
+    if user.deleted?
+      sign_out user
+      flash[:notice] = nil
+      flash[:danger] = 'Invalid Email or password.'
+      render :new
+      return false
+    end
+
+    # If profile is deleted, the user was banished. Inform them and send them back to the sign in page.
+    if user.community_user&.deleted?
+      sign_out user
+      flash[:notice] = nil
+      flash[:danger] = 'Your profile on this community has been deleted.'
+      render :new
+      return false
+    end
+
+    # For users who are linked to an SSO Profile, disallow normal login and let them sign in through SSO instead.
+    if user.sso_profile.present?
+      sign_out user
+      flash[:notice] = nil
+      flash[:danger] = 'Please sign in using the Single Sign-On service of your institution.'
+      redirect_to new_saml_user_session_path
+      return false
+    end
+
+    # Enforce 2FA
+    if user.enabled_2fa
+      handle_2fa_login(user)
+      return false
+    end
+
+    true
+  end
+
+  def handle_2fa_login(user)
+    sign_out user
+    case user.two_factor_method
+    when 'app'
+      id = user.id
+      @@first_factor << id
+      redirect_to login_verify_2fa_path(uid: id)
+    when 'email'
+      TwoFactorMailer.with(user: user, host: request.hostname).login_email.deliver_now
+      flash[:notice] = nil
+      flash[:info] = 'Please check your email inbox for a link to sign in.'
+      redirect_to after_sign_in_path_for(user)
+    end
+  end
 end
diff --git a/app/controllers/users_controller.rb b/app/controllers/users_controller.rb
index 11e95df0dec4419ffe76b4588bc8aef48caa959e..1b4aec6f0c2d168789e5f5b87e0e5ac2529c60b7 100644
--- a/app/controllers/users_controller.rb
+++ b/app/controllers/users_controller.rb
@@ -5,12 +5,13 @@ class UsersController < ApplicationController
   include Devise::Controllers::Rememberable
 
   before_action :authenticate_user!, only: [:edit_profile, :update_profile, :stack_redirect, :transfer_se_content,
-                                            :qr_login_code, :me, :preferences, :set_preference, :my_vote_summary]
+                                            :qr_login_code, :me, :preferences, :set_preference, :my_vote_summary,
+                                            :disconnect_sso, :confirm_disconnect_sso, :filters]
   before_action :verify_moderator, only: [:mod, :destroy, :soft_delete, :role_toggle, :full_log,
                                           :annotate, :annotations, :mod_privileges, :mod_privilege_action]
   before_action :set_user, only: [:show, :mod, :destroy, :soft_delete, :posts, :role_toggle, :full_log, :activity,
                                   :annotate, :annotations, :mod_privileges, :mod_privilege_action,
-                                  :vote_summary, :avatar]
+                                  :vote_summary, :network, :avatar]
   before_action :check_deleted, only: [:show, :posts, :activity]
 
   def index
@@ -26,13 +27,20 @@ class UsersController < ApplicationController
 
   def show
     @abilities = Ability.on_user(@user)
-    @posts = if current_user&.privilege?('flag_curate')
-               @user.posts
-             else
-               @user.posts.undeleted
-             end.list_includes.joins(:category)
-             .where('IFNULL(categories.min_view_trust_level, 0) <= ?', current_user&.trust_level || 0)
-             .order(score: :desc).first(15)
+
+    all_posts = if current_user&.privilege?('flag_curate') || @user == current_user
+                  @user.posts
+                else
+                  @user.posts.undeleted
+                end
+                .list_includes
+                .joins(:category)
+                .where('IFNULL(categories.min_view_trust_level, 0) <= ?', current_user&.trust_level || 0)
+                .user_sort({ term: params[:sort], default: :score },
+                           age: :created_at, score: :score)
+
+    @posts = all_posts.first(15)
+    @total_post_count = all_posts.count
     render layout: 'without_sidebar'
   end
 
@@ -57,6 +65,7 @@ class UsersController < ApplicationController
         prefs = current_user.preferences
         @preferences = prefs[:global]
         @community_prefs = prefs[:community]
+        render layout: 'without_sidebar'
       end
       format.json do
         render json: current_user.preferences
@@ -64,6 +73,99 @@ class UsersController < ApplicationController
     end
   end
 
+  # Helper method to convert it to the form expected by the client
+  def filter_json(filter)
+    {
+      min_score: filter.min_score,
+      max_score: filter.max_score,
+      min_answers: filter.min_answers,
+      max_answers: filter.max_answers,
+      include_tags: Tag.where(id: filter.include_tags).map { |tag| [tag.name, tag.id] },
+      exclude_tags: Tag.where(id: filter.exclude_tags).map { |tag| [tag.name, tag.id] },
+      status: filter.status,
+      system: filter.user_id == -1
+    }
+  end
+
+  def filters_json
+    system_filters = Rails.cache.fetch 'default_system_filters', expires_in: 1.day do
+      User.find(-1).filters.to_h { |filter| [filter.name, filter_json(filter)] }
+    end
+
+    if user_signed_in?
+      current_user.filters.to_h { |filter| [filter.name, filter_json(filter)] }
+                  .merge(system_filters)
+    else
+      system_filters
+    end
+  end
+
+  def filters
+    respond_to do |format|
+      format.html do
+        render layout: 'without_sidebar'
+      end
+      format.json do
+        render json: filters_json
+      end
+    end
+  end
+
+  def set_filter
+    if user_signed_in? && params[:name]
+      filter = Filter.find_or_create_by(user: current_user, name: params[:name])
+
+      filter.update(filter_params)
+
+      unless params[:category].nil? || params[:is_default].nil?
+        helpers.set_filter_default(current_user.id, filter.id, params[:category].to_i, params[:is_default])
+      end
+
+      render json: { status: 'success', success: true, filters: filters_json },
+             status: 200
+    else
+      render json: { status: 'failed', success: false, errors: ['Filter name is required'] },
+             status: 400
+    end
+  end
+
+  def delete_filter
+    unless params[:name]
+      return render json: { status: 'failed', success: false, errors: ['Filter name is required'] },
+                    status: 400
+    end
+
+    as_user = current_user
+
+    if params[:system] == true
+      if current_user&.is_global_admin
+        as_user = User.find(-1)
+      else
+        return render json: { status: 'failed', success: false, errors: ['You do not have permission to delete'] },
+                      status: 400
+      end
+    end
+
+    filter = Filter.find_by(user: as_user, name: params[:name])
+    if filter.destroy
+      render json: { status: 'success', success: true, filters: filters_json }
+    else
+      render json: { status: 'failed', success: false, errors: ['Failed to delete'] },
+             status: 400
+    end
+  end
+
+  def default_filter
+    if user_signed_in? && params[:category]
+      default_filter = helpers.default_filter(current_user.id, params[:category].to_i)
+      render json: { status: 'success', success: true, name: default_filter&.name },
+             status: 200
+    else
+      render json: { status: 'failed', success: false },
+             status: 400
+    end
+  end
+
   def set_preference
     if !params[:name].nil? && !params[:value].nil?
       global_key = "prefs.#{current_user.id}"
@@ -80,7 +182,7 @@ class UsersController < ApplicationController
   end
 
   def posts
-    @posts = if current_user&.privilege?('flag_curate')
+    @posts = if current_user&.privilege?('flag_curate') || @user == current_user
                Post.all
              else
                Post.undeleted
@@ -99,12 +201,22 @@ class UsersController < ApplicationController
     end
   end
 
+  def my_network
+    redirect_to network_path(current_user)
+  end
+
+  def network
+    @communities = Community.all
+    render layout: 'without_sidebar'
+  end
+
   def activity
     @posts = Post.undeleted.where(user: @user).count
     @comments = Comment.joins(:comment_thread, :post).undeleted.where(user: @user, comment_threads: { deleted: false },
                                                                       posts: { deleted: false }).count
     @suggested_edits = SuggestedEdit.where(user: @user).count
-    @edits = PostHistory.joins(:post).where(user: @user, posts: { deleted: false }).count
+    @edits = PostHistory.joins(:post, :post_history_type).where(user: @user, posts: { deleted: false },
+                                                                post_history_types: { name: 'post_edited' }).count
 
     @all_edits = @suggested_edits + @edits
 
@@ -116,16 +228,17 @@ class UsersController < ApplicationController
                                                                     posts: { deleted: false })
             when 'edits'
               SuggestedEdit.where(user: @user) + \
-              PostHistory.joins(:post).where(user: @user, posts: { deleted: false })
+              PostHistory.joins(:post, :post_history_type).where(user: @user, posts: { deleted: false },
+                                                                 post_history_types: { name: 'post_edited' })
             else
               Post.undeleted.where(user: @user) + \
               Comment.joins(:comment_thread, :post).undeleted.where(user: @user, comment_threads: { deleted: false },
                                                                     posts: { deleted: false }) + \
               SuggestedEdit.where(user: @user).all + \
-              PostHistory.joins(:post).where(user: @user, posts: { deleted: false })
+              PostHistory.joins(:post).where(user: @user, posts: { deleted: false }).all
             end
 
-    @items = items.sort_by(&:created_at).reverse
+    @items = items.sort_by(&:created_at).reverse.paginate(page: params[:page], per_page: 50)
     render layout: 'without_sidebar'
   end
 
@@ -168,7 +281,7 @@ class UsersController < ApplicationController
                 Post.where(user: @user).all + Comment.where(user: @user).all + Flag.where(user: @user).all + \
                   SuggestedEdit.where(user: @user).all + PostHistory.where(user: @user).all + \
                   ModWarning.where(community_user: @user.community_user).all
-              end).sort_by(&:created_at).reverse
+              end).sort_by(&:created_at).reverse.paginate(page: params[:page], per_page: 50)
 
     render layout: 'without_sidebar'
   end
@@ -229,13 +342,7 @@ class UsersController < ApplicationController
         return
       end
 
-      AuditLog.moderator_audit(event_type: 'user_delete', related: @user, user: current_user,
-                               comment: @user.attributes_print(join: "\n"))
-      @user.assign_attributes(deleted: true, deleted_by_id: current_user.id, deleted_at: DateTime.now,
-                              username: "user#{@user.id}", email: "#{@user.id}@deleted.localhost",
-                              password: SecureRandom.hex(32))
-      @user.skip_reconfirmation!
-      @user.save
+      @user.do_soft_delete(current_user)
     else
       render json: { status: 'failed', message: 'Unrecognised deletion type.' }, status: 400
       return
@@ -248,14 +355,24 @@ class UsersController < ApplicationController
     render layout: 'without_sidebar'
   end
 
-  def update_profile
-    profile_params = params.require(:user).permit(:username, :profile_markdown, :website, :twitter, :discord)
-    profile_params[:twitter] = profile_params[:twitter].delete('@')
+  def cleaned_profile_websites(profile_params)
+    sites = profile_params[:user_websites_attributes]
 
-    if profile_params[:website].present? && URI.parse(profile_params[:website]).instance_of?(URI::Generic)
-      # URI::Generic indicates the user didn't include a protocol, so we'll add one now so that it can be
-      # parsed correctly in the view later on.
-      profile_params[:website] = "https://#{profile_params[:website]}"
+    sites.transform_values do |w|
+      w.merge({ label: w[:label].presence, url: w[:url].presence })
+    end
+  end
+
+  def update_profile
+    profile_params = params.require(:user).permit(:username,
+                                                  :profile_markdown,
+                                                  :website,
+                                                  :discord,
+                                                  :twitter,
+                                                  user_websites_attributes: [:id, :label, :url])
+
+    if profile_params[:user_websites_attributes].present?
+      profile_params[:user_websites_attributes] = cleaned_profile_websites(profile_params)
     end
 
     @user = current_user
@@ -271,8 +388,14 @@ class UsersController < ApplicationController
       end
     end
 
-    profile_rendered = helpers.post_markdown(:user, :profile_markdown)
-    if @user.update(profile_params.merge(profile: profile_rendered))
+    if params[:user][:profile_markdown].present?
+      profile_rendered = helpers.post_markdown(:user, :profile_markdown)
+      profile_params = profile_params.merge(profile: profile_rendered)
+    end
+
+    status = @user.update(profile_params)
+
+    if status
       flash[:success] = 'Your profile details were updated.'
       redirect_to user_path(current_user)
     else
@@ -301,7 +424,7 @@ class UsersController < ApplicationController
 
       # Set/update ability
       if new_value
-        @user.community_user.grant_privilege 'mod'
+        @user.community_user.grant_privilege! 'mod'
       else
         @user.community_user.privilege('mod').destroy
       end
@@ -330,7 +453,7 @@ class UsersController < ApplicationController
     case params[:do]
     when 'grant'
       if ua.nil?
-        @user.community_user.grant_privilege(ability.internal_id)
+        @user.community_user.grant_privilege!(ability.internal_id)
         AuditLog.admin_audit(event_type: 'ability_grant', related: @user, user: current_user,
                              comment: ability.internal_id.to_s)
       elsif ua.is_suspended
@@ -426,7 +549,7 @@ class UsersController < ApplicationController
       sign_in user
       remember_me user
       AuditLog.user_history(event_type: 'mobile_login', related: user)
-      redirect_to root_path
+      redirect_to after_sign_in_path_for(user)
     else
       flash[:danger] = "That login link isn't valid. Codes expire after 5 minutes - if it's been longer than that, " \
                        'get a new code and try again.'
@@ -456,21 +579,27 @@ class UsersController < ApplicationController
   end
 
   def vote_summary
-    @votes = Vote.where(recv_user: @user) \
-                 .includes(:post).group(:date_of, :post_id, :vote_type)
-    @votes = @votes.select(:post_id, :vote_type) \
-                   .select('count(*) as vote_count') \
-                   .select('date(created_at) as date_of')
+    @votes = Vote.where(recv_user: @user)
+                 .includes(:post)
+                 .group(:date_of, :post_id, :vote_type)
+
+    @votes = @votes.select(:post_id, :vote_type)
+                   .select('count(*) as vote_count')
+                   .select('date(votes.created_at) as date_of')
+
     @votes = @votes.order(date_of: :desc, post_id: :desc).all \
-                   .group_by(&:date_of).map { |k, vl| [k, vl.group_by(&:post) ] } \
+                   .group_by(&:date_of).map do |k, vl|
+                     [k, vl.group_by(&:post), vl.sum { |v| v.vote_type * v.vote_count }]
+                   end \
                    .paginate(page: params[:page], per_page: 15)
-    @votes
+
+    render layout: 'without_sidebar'
   end
 
   def avatar
     respond_to do |format|
       format.png do
-        size = params[:size]&.to_i&.positive? ? params[:size]&.to_i : 64
+        size = params[:size]&.to_i&.positive? ? [params[:size]&.to_i, 256].min : 64
         send_data helpers.user_auto_avatar(size, user: @user).to_blob, type: 'image/png', disposition: 'inline'
       end
     end
@@ -486,10 +615,41 @@ class UsersController < ApplicationController
     end
   end
 
+  def disconnect_sso
+    render layout: 'without_sidebar'
+  end
+
+  def confirm_disconnect_sso
+    if current_user.sso_profile.blank? || !helpers.devise_sign_in_enabled? || !SiteSetting['AllowSsoDisconnect']
+      flash[:danger] = 'You cannot disable Single Sign-On.'
+      redirect_to edit_user_registration_path
+      return
+    end
+
+    if current_user.sso_profile.destroy
+      current_user.send_reset_password_instructions
+      flash[:success] = 'Successfully disconnected from Single Sign-On. Please see your email to set your password.'
+      redirect_to edit_user_registration_path
+    else
+      flash[:danger] = 'Failed to disconnect from Single Sign-On.'
+      redirect_to user_disconnect_sso_path
+    end
+  end
+
   private
 
+  def filter_params
+    params.permit(:min_score, :max_score, :min_answers, :max_answers, :status, :include_tags, :exclude_tags,
+                  include_tags: [], exclude_tags: [])
+  end
+
   def set_user
-    @user = user_scope.find_by(id: params[:id])
+    user_id = if params[:id] == 'me' && user_signed_in?
+                current_user.id
+              else
+                params[:id]
+              end
+    @user = user_scope.find_by(id: user_id)
     not_found if @user.nil?
   end
 
diff --git a/app/controllers/votes_controller.rb b/app/controllers/votes_controller.rb
index 84b57ce79d869f1e86640a06e47312d021e47043..f39a98b4413cf0bccf8ae0637d3efcff90f2d8b0 100644
--- a/app/controllers/votes_controller.rb
+++ b/app/controllers/votes_controller.rb
@@ -42,8 +42,11 @@ class VotesController < ApplicationController
     AbilityQueue.add(post.user, "Vote Change on ##{post.id}")
 
     modified = !destroyed.empty?
-    state = { status: (modified ? 'modified' : 'OK'), vote_id: vote.id, upvotes: post.upvote_count,
-              downvotes: post.downvote_count }
+    state = { status: (modified ? 'modified' : 'OK'),
+              vote_id: vote.id,
+              upvotes: post.upvote_count,
+              downvotes: post.downvote_count,
+              score: post.score }
 
     render json: state
   end
@@ -59,7 +62,10 @@ class VotesController < ApplicationController
 
     if vote.destroy
       AbilityQueue.add(post.user, "Vote Change on ##{post.id}")
-      render json: { status: 'OK', upvotes: post.upvote_count, downvotes: post.downvote_count }
+      render json: { status: 'OK',
+                     upvotes: post.upvote_count,
+                     downvotes: post.downvote_count,
+                     score: post.score }
     else
       render json: { status: 'failed', message: vote.errors.full_messages.join('. ') }, status: :forbidden
     end
diff --git a/app/helpers/advertisements/article_helper.rb b/app/helpers/advertisements/article_helper.rb
index 80a425c75c0fe1e83e0b0726834b8bda2205fb58..fc1dc9d619ea914e2e60affab3f6d539a410f90a 100644
--- a/app/helpers/advertisements/article_helper.rb
+++ b/app/helpers/advertisements/article_helper.rb
@@ -21,8 +21,8 @@ module Advertisements::ArticleHelper
       answer.font = './app/assets/imgfonts/Roboto-Bold.ttf'
       answer.pointsize = 40
       answer.gravity = CenterGravity
-      answer.annotate ad, 600, 120, 0, 10, 'Check out this article' do
-        self.fill = 'white'
+      answer.annotate ad, 600, 120, 0, 10, 'Check out this article' do |s|
+        s.fill = 'white'
       end
 
       icon_path = SiteSetting.find_by(name: 'SiteLogoPath', community: article.community).typed
@@ -37,8 +37,8 @@ module Advertisements::ArticleHelper
         community_name.font = './app/assets/imgfonts/Roboto-Bold.ttf'
         community_name.pointsize = 25
         community_name.gravity = SouthWestGravity
-        community_name.annotate ad, 0, 0, 20, 20, article.community.name do
-          self.fill = '#4B68FF'
+        community_name.annotate ad, 0, 0, 20, 20, article.community.name do |s|
+          s.fill = '#4B68FF'
         end
       end
 
@@ -48,8 +48,8 @@ module Advertisements::ArticleHelper
       community_url.font = './app/assets/imgfonts/Roboto-Bold.ttf'
       community_url.pointsize = 20
       community_url.gravity = SouthEastGravity
-      community_url.annotate ad, 0, 0, 20, 20, article.community.host do
-        self.fill = '#666666'
+      community_url.annotate ad, 0, 0, 20, 20, article.community.host do |s|
+        s.fill = '#666666'
       end
 
       title = Draw.new
@@ -62,15 +62,15 @@ module Advertisements::ArticleHelper
       if article.title.length > 60
         title.pointsize = 35
         wrap_text(do_rtl_witchcraft(article.title), 500, 35).split("\n").each do |line|
-          title.annotate ad, 500, 100, 50, 135 + (position * 55), line do
-            self.fill = '#333333'
+          title.annotate ad, 500, 100, 50, 135 + (position * 55), line do |s|
+            s.fill = '#333333'
           end
           position += 1
         end
       else
         wrap_text(do_rtl_witchcraft(article.title), 500, 55).split("\n").each do |line|
-          title.annotate ad, 500, 100, 50, 160 + (position * 70), line do
-            self.fill = '#333333'
+          title.annotate ad, 500, 100, 50, 160 + (position * 70), line do |s|
+            s.fill = '#333333'
           end
           position += 1
         end
diff --git a/app/helpers/application_helper.rb b/app/helpers/application_helper.rb
index 175469b04e4b9625388ac2e90fb7463f91109fe6..ac3009673883e3bee76d968a9bd9461031c1bc07 100644
--- a/app/helpers/application_helper.rb
+++ b/app/helpers/application_helper.rb
@@ -34,6 +34,10 @@ module ApplicationHelper
     uri.to_s
   end
 
+  def sign_in_link(title)
+    link_to title, new_user_session_url
+  end
+
   def license_link
     link_to SiteSetting['ContentLicenseName'], SiteSetting['ContentLicenseLink']
   end
@@ -91,6 +95,19 @@ module ApplicationHelper
     end
   end
 
+  def generic_share_link_md(post)
+    "[#{post.title}](#{generic_share_link(post)})"
+  end
+
+  def post_history_share_link(post, history, index)
+    post_history_url(post, anchor: history.size - index)
+  end
+
+  def post_history_share_link_md(post, history, index)
+    rev_num = history.size - index
+    "[Revision #{rev_num} — #{post.title}](#{post_history_share_link(post, history, index)})"
+  end
+
   def generic_edit_link(post)
     edit_post_url(post)
   end
@@ -177,4 +194,19 @@ module ApplicationHelper
       false
     end
   end
+
+  def current_commit
+    commit_info = Rails.cache.persistent('current_commit')
+    shasum, timestamp = commit_info
+
+    begin
+      date = DateTime.iso8601(timestamp)
+    rescue
+      date = DateTime.parse(timestamp)
+    end
+
+    [shasum, date]
+  rescue
+    [nil, nil]
+  end
 end
diff --git a/app/helpers/categories_helper.rb b/app/helpers/categories_helper.rb
index 68c77d120ccdf6e1727a082688b7706d9544babf..7efbc16177af38fe867cc23d921e407d40e13f3f 100644
--- a/app/helpers/categories_helper.rb
+++ b/app/helpers/categories_helper.rb
@@ -23,4 +23,10 @@ module CategoriesHelper
                             @article.category
                           end
   end
+
+  def pending_suggestions?
+    Rails.cache.fetch "pending_suggestions/#{current_category.id}" do
+      SuggestedEdit.where(post: Post.undeleted.where(category: current_category), active: true).any?
+    end
+  end
 end
diff --git a/app/helpers/comments_helper.rb b/app/helpers/comments_helper.rb
index 636d0a95da83d71b995fae6886611a473b5596c2..548a269c958099487fa6609a140fddefa90238b8 100644
--- a/app/helpers/comments_helper.rb
+++ b/app/helpers/comments_helper.rb
@@ -81,8 +81,8 @@ end
 class CommentScrubber < Rails::Html::PermitScrubber
   def initialize
     super
-    self.tags = %w[a b i em strong s strike del pre code p blockquote span sup sub br]
-    self.attributes = %w[href title lang dir id class]
+    self.tags = %w[a b i em strong s strike del pre code p blockquote span sup sub br ul ol li]
+    self.attributes = %w[href title lang dir id class start]
   end
 
   def skip_node?(node)
diff --git a/app/helpers/edit_helper.rb b/app/helpers/edit_helper.rb
new file mode 100644
index 0000000000000000000000000000000000000000..db4b05ad4a53b2c185c6fd076f82ac51d8ffcda7
--- /dev/null
+++ b/app/helpers/edit_helper.rb
@@ -0,0 +1,5 @@
+module EditHelper
+  def max_edit_comment_length
+    [SiteSetting['MaxEditCommentLength'] || 255, 255].min
+  end
+end
diff --git a/app/helpers/email_logs_helper.rb b/app/helpers/email_logs_helper.rb
new file mode 100644
index 0000000000000000000000000000000000000000..9b08eeb7ab077f6893d11dcab32b86e2d5241380
--- /dev/null
+++ b/app/helpers/email_logs_helper.rb
@@ -0,0 +1,2 @@
+module EmailLogsHelper
+end
diff --git a/app/helpers/markdown_tools_helper.rb b/app/helpers/markdown_tools_helper.rb
index 0c4f44c75a4379faff4cefa51b659da86fbc8c8d..9dbcaa85d210a920faeadf155ebd08575cdadca7 100644
--- a/app/helpers/markdown_tools_helper.rb
+++ b/app/helpers/markdown_tools_helper.rb
@@ -4,7 +4,8 @@ module MarkdownToolsHelper
                    class: "#{attribs[:class] || ''} button is-muted is-outlined js-markdown-tool",
                    data_action: action,
                    aria_label: label,
-                   title: label
+                   title: label,
+                   role: 'button'
     attribs.transform_keys! { |k| k.to_s.tr('_', '-') }.symbolize_keys!
     if name.nil? && block_given?
       tag.a(**attribs, &block)
@@ -18,7 +19,8 @@ module MarkdownToolsHelper
                    class: "#{attribs[:class] || ''}js-markdown-tool",
                    data_action: action,
                    aria_label: label,
-                   title: label
+                   title: label,
+                   role: 'button'
     attribs.transform_keys! { |k| k.to_s.tr('_', '-') }.symbolize_keys!
     if name.nil? && block_given?
       tag.a(**attribs, &block)
diff --git a/app/helpers/posts_helper.rb b/app/helpers/posts_helper.rb
index 2c9e19ab61dd5932eb06a33f755709bc1798b72f..9fb810fd52b14927be5ba3cca663cd7ee1a479b3 100644
--- a/app/helpers/posts_helper.rb
+++ b/app/helpers/posts_helper.rb
@@ -15,6 +15,35 @@ module PostsHelper
     end
   end
 
+  # @param category [Category, Nil]
+  # @return [Integer] the minimum length for post bodies
+  def min_body_length(category)
+    category&.min_body_length || 30
+  end
+
+  # @param _category [Category, Nil]
+  # @return [Integer] the maximum length for post bodies
+  def max_body_length(_category)
+    30_000
+  end
+
+  # @param category [Category, Nil] post category, if any
+  # @param post_type [PostType] type of the post (system limits are relaxed)
+  # @return [Integer] the minimum length for post titles
+  def min_title_length(category, post_type)
+    if post_type.system?
+      1
+    else
+      category&.min_title_length || 15
+    end
+  end
+
+  # @param _category [Category, Nil]
+  # @return [Integer] the maximum length for post titles
+  def max_title_length(_category)
+    [SiteSetting['MaxTitleLength'] || 255, 255].min
+  end
+
   class PostScrubber < Rails::Html::PermitScrubber
     def initialize
       super
diff --git a/app/helpers/search_helper.rb b/app/helpers/search_helper.rb
index f5786839acd0e703e059e6668ebcb5f73769d051..fe21b7d3533db78d788ba611a86323b7596e2a8a 100644
--- a/app/helpers/search_helper.rb
+++ b/app/helpers/search_helper.rb
@@ -1,4 +1,104 @@
 module SearchHelper
+  def check_posts_permissions
+    (current_user&.is_moderator || current_user&.is_admin ? Post : Post.undeleted)
+      .qa_only.list_includes
+  end
+
+  def search_posts
+    posts = check_posts_permissions
+
+    qualifiers = params_to_qualifiers
+    search_string = params[:search]
+
+    # Filter based on search string qualifiers
+    if search_string.present?
+      search_data = parse_search(search_string)
+      qualifiers += parse_qualifier_strings search_data[:qualifiers]
+      search_string = search_data[:search]
+    end
+
+    posts = qualifiers_to_sql(qualifiers, posts)
+    posts = posts.paginate(page: params[:page], per_page: 25)
+
+    posts = if search_string.present?
+              posts.search(search_data[:search]).user_sort({ term: params[:sort], default: :search_score },
+                                                           relevance: :search_score, score: :score, age: :created_at)
+            else
+              posts.user_sort({ term: params[:sort], default: :score },
+                              score: :score, age: :created_at)
+            end
+
+    [posts, qualifiers]
+  end
+
+  # Convert a Filter record into a form parseable by the search function
+  def filter_to_qualifiers(filter)
+    qualifiers = []
+    qualifiers.append({ param: :score, operator: '>=', value: filter.min_score }) unless filter.min_score.nil?
+    qualifiers.append({ param: :score, operator: '<=', value: filter.max_score }) unless filter.max_score.nil?
+    qualifiers.append({ param: :answers, operator: '>=', value: filter.min_answers }) unless filter.min_answers.nil?
+    qualifiers.append({ param: :answers, operator: '<=', value: filter.max_answers }) unless filter.max_answers.nil?
+    qualifiers.append({ param: :include_tags, tag_ids: filter.include_tags }) unless filter.include_tags.nil?
+    qualifiers.append({ param: :exclude_tags, tag_ids: filter.exclude_tags }) unless filter.exclude_tags.nil?
+    qualifiers.append({ param: :status, value: filter.status }) unless filter.status.nil?
+    qualifiers
+  end
+
+  def active_filter
+    {
+      default: nil,
+      name: params[:predefined_filter],
+      min_score: params[:min_score],
+      max_score: params[:max_score],
+      min_answers: params[:min_answers],
+      max_answers: params[:max_answers],
+      include_tags: params[:include_tags],
+      exclude_tags: params[:exclude_tags],
+      status: params[:status]
+    }
+  end
+
+  def params_to_qualifiers
+    valid_value = {
+      date: /^[\d.]+(?:s|m|h|d|w|mo|y)?$/,
+      status: /any|open|closed/,
+      numeric: /^[\d.]+$/,
+      integer: /^\d+$/
+    }
+
+    filter_qualifiers = []
+
+    if params[:min_score]&.match?(valid_value[:numeric])
+      filter_qualifiers.append({ param: :score, operator: '>=', value: params[:min_score].to_f })
+    end
+
+    if params[:max_score]&.match?(valid_value[:numeric])
+      filter_qualifiers.append({ param: :score, operator: '<=', value: params[:max_score].to_f })
+    end
+
+    if params[:min_answers]&.match?(valid_value[:numeric])
+      filter_qualifiers.append({ param: :answers, operator: '>=', value: params[:min_answers].to_i })
+    end
+
+    if params[:max_answers]&.match?(valid_value[:numeric])
+      filter_qualifiers.append({ param: :answers, operator: '<=', value: params[:max_answers].to_i })
+    end
+
+    if params[:status]&.match?(valid_value[:status])
+      filter_qualifiers.append({ param: :status, value: params[:status] })
+    end
+
+    if params[:include_tags]&.all? { |id| id.match? valid_value[:integer] }
+      filter_qualifiers.append({ param: :include_tags, tag_ids: params[:include_tags] })
+    end
+
+    if params[:exclude_tags]&.all? { |id| id.match? valid_value[:integer] }
+      filter_qualifiers.append({ param: :exclude_tags, tag_ids: params[:exclude_tags] })
+    end
+
+    filter_qualifiers
+  end
+
   def parse_search(raw_search)
     qualifiers_regex = /([\w\-_]+(?<!\\):[^ ]+)/
     qualifiers = raw_search.scan(qualifiers_regex).flatten
@@ -11,13 +111,14 @@ module SearchHelper
   end
 
   # rubocop:disable Metrics/CyclomaticComplexity
-  def qualifiers_to_sql(qualifiers, query)
+  def parse_qualifier_strings(qualifiers)
     valid_value = {
       date: /^[<>=]{0,2}[\d.]+(?:s|m|h|d|w|mo|y)?$/,
+      status: /any|open|closed/,
       numeric: /^[<>=]{0,2}[\d.]+$/
     }
 
-    qualifiers.each do |qualifier| # rubocop:disable Metrics/BlockLength
+    qualifiers.map do |qualifier| # rubocop:disable Metrics/BlockLength
       splat = qualifier.split ':'
       parameter = splat[0]
       value = splat[1]
@@ -27,59 +128,111 @@ module SearchHelper
         next unless value.match?(valid_value[:numeric])
 
         operator, val = numeric_value_sql value
-        query = query.where("score #{operator.presence || '='} ?", val.to_f)
+        { param: :score, operator: operator.presence || '=', value: val.to_f }
       when 'created'
         next unless value.match?(valid_value[:date])
 
         operator, val, timeframe = date_value_sql value
-        query = query.where("created_at #{operator.presence || '='} DATE_SUB(CURRENT_TIMESTAMP, " \
-                            "INTERVAL ? #{timeframe})",
-                            val.to_i)
+        { param: :created, operator: operator.presence || '=', timeframe: timeframe, value: val.to_i }
       when 'user'
-        next unless value.match?(valid_value[:numeric])
+        operator, val = if value.match?(valid_value[:numeric])
+                          numeric_value_sql value
+                        elsif value == 'me'
+                          ['=', current_user&.id&.to_i]
+                        else
+                          next
+                        end
 
-        operator, val = numeric_value_sql value
-        query = query.where("user_id #{operator.presence || '='} ?", val.to_i)
+        { param: :user, operator: operator.presence || '=', user_id: val }
       when 'upvotes'
         next unless value.match?(valid_value[:numeric])
 
         operator, val = numeric_value_sql value
-        query = query.where("upvotes #{operator.presence || '='} ?", val.to_i)
+        { param: :upvotes, operator: operator.presence || '=', value: val.to_i }
       when 'downvotes'
         next unless value.match?(valid_value[:numeric])
 
         operator, val = numeric_value_sql value
-        query = query.where("downvotes #{operator.presence || '='} ?", val.to_i)
+        { param: :downvotes, operator: operator.presence || '=', value: val.to_i }
       when 'votes'
         next unless value.match?(valid_value[:numeric])
 
         operator, val = numeric_value_sql value
-        query = query.where("(upvotes - downvotes) #{operator.presence || '='}", val.to_i)
+        { param: :net_votes, operator: operator.presence || '=', value: val.to_i }
       when 'tag'
-        query = query.where(posts: { id: PostsTag.where(tag_id: Tag.where(name: value).select(:id)).select(:post_id) })
+        { param: :include_tag, tag_id: Tag.where(name: value).select(:id) }
       when '-tag'
-        query = query.where.not(posts: { id: PostsTag.where(tag_id: Tag.where(name: value).select(:id))
-                                                     .select(:post_id) })
+        { param: :exclude_tag, tag_id: Tag.where(name: value).select(:id) }
       when 'category'
         next unless value.match?(valid_value[:numeric])
 
         operator, val = numeric_value_sql value
-        trust_level = current_user&.trust_level || 0
-        allowed_categories = Category.where('IFNULL(min_view_trust_level, -1) <= ?', trust_level)
-        query = query.where("category_id #{operator.presence || '='} ?", val.to_i)
-                     .where(category_id: allowed_categories)
+        { param: :category, operator: operator.presence || '=', category_id: val.to_i }
       when 'post_type'
         next unless value.match?(valid_value[:numeric])
 
         operator, val = numeric_value_sql value
-        query = query.where("post_type_id #{operator.presence || '='} ?", val.to_i)
+        { param: :post_type, operator: operator.presence || '=', post_type_id: val.to_i }
       when 'answers'
         next unless value.match?(valid_value[:numeric])
 
         operator, val = numeric_value_sql value
+        { param: :answers, operator: operator.presence || '=', value: val.to_i }
+      when 'status'
+        next unless value.match?(valid_value[:status])
+
+        { param: :status, value: value }
+      end
+    end.compact
+    # Consider partitioning and telling the user which filters were invalid
+  end
+
+  def qualifiers_to_sql(qualifiers, query)
+    trust_level = current_user&.trust_level || 0
+    allowed_categories = Category.where('IFNULL(min_view_trust_level, -1) <= ?', trust_level)
+    query = query.where(category_id: allowed_categories)
+
+    qualifiers.each do |qualifier| # rubocop:disable Metrics/BlockLength
+      case qualifier[:param]
+      when :score
+        query = query.where("score #{qualifier[:operator]} ?", qualifier[:value])
+      when :created
+        query = query.where("created_at #{qualifier[:operator]} DATE_SUB(CURRENT_TIMESTAMP, " \
+                            "INTERVAL ? #{qualifier[:timeframe]})",
+                            qualifier[:value])
+      when :user
+        query = query.where("user_id #{qualifier[:operator]} ?", qualifier[:user_id])
+      when :upvotes
+        query = query.where("upvote_count #{qualifier[:operator]} ?", qualifier[:value])
+      when :downvotes
+        query = query.where("downvote_count #{qualifier[:operator]} ?", qualifier[:value])
+      when :net_votes
+        query = query.where("(upvote_count - downvote_count) #{qualifier[:operator]} ?", qualifier[:value])
+      when :include_tag
+        query = query.where(posts: { id: PostsTag.where(tag_id: qualifier[:tag_id]).select(:post_id) })
+      when :include_tags
+        qualifier[:tag_ids].each do |id|
+          query = query.where(id: PostsTag.where(tag_id: id).select(:post_id))
+        end
+      when :exclude_tag
+        query = query.where.not(posts: { id: PostsTag.where(tag_id: qualifier[:tag_id]).select(:post_id) })
+      when :exclude_tags
+        query = query.where.not(id: PostsTag.where(tag_id: qualifier[:tag_ids]).select(:post_id))
+      when :category
+        query = query.where("category_id #{qualifier[:operator]} ?", qualifier[:category_id])
+      when :post_type
+        query = query.where("post_type_id #{qualifier[:operator]} ?", qualifier[:post_type_id])
+      when :answers
         post_types_with_answers = PostType.where(has_answers: true)
-        query = query.where("answer_count #{operator.presence || '='} ?", val.to_i)
+        query = query.where("answer_count #{qualifier[:operator]} ?", qualifier[:value])
                      .where(post_type_id: post_types_with_answers)
+      when :status
+        case qualifier[:value]
+        when 'open'
+          query = query.where(closed: false)
+        when 'closed'
+          query = query.where(closed: true)
+        end
       end
     end
 
diff --git a/app/helpers/users/avatar_helper.rb b/app/helpers/users/avatar_helper.rb
index af86fc68950bf2e957ab8d3abfdd1615cd064c59..6a6686f2785c8d4f4f1801ac08d04d7231b41452 100644
--- a/app/helpers/users/avatar_helper.rb
+++ b/app/helpers/users/avatar_helper.rb
@@ -35,8 +35,8 @@ module Users::AvatarHelper
       let.font = './app/assets/imgfonts/Roboto.ttf'
       let.pointsize = size * 0.75
       let.gravity = CenterGravity
-      let.annotate ava, size, size * 1.16, 0, 0, letter.upcase do
-        self.fill = text_color
+      let.annotate ava, size, size * 1.16, 0, 0, letter.upcase do |s|
+        s.fill = text_color
       end
 
       ava.format = 'PNG'
diff --git a/app/helpers/users_helper.rb b/app/helpers/users_helper.rb
index 2ebc8c27cacda0954185d3f31676527881ec197d..a689060d0dff293d0379c67f84e8905327d8e0f1 100644
--- a/app/helpers/users_helper.rb
+++ b/app/helpers/users_helper.rb
@@ -30,6 +30,21 @@ module UsersHelper
     end
   end
 
+  def default_filter(user_id, category_id)
+    CategoryFilterDefault.find_by(user_id: user_id, category_id: category_id)&.filter
+  end
+
+  def set_filter_default(user_id, filter_id, category_id, keep)
+    if keep
+      CategoryFilterDefault.create_with(filter_id: filter_id)
+                           .find_or_create_by(user_id: user_id, category_id: category_id)
+                           .update(filter_id: filter_id)
+    else
+      CategoryFilterDefault.where(user_id: user_id, category_id: category_id)
+                           .destroy_all
+    end
+  end
+
   def user_preference(name, community: false)
     return nil if current_user.nil?
 
@@ -44,11 +59,37 @@ module UsersHelper
     deleted_user?(user) ? 'deleted user' : user.rtl_safe_username
   end
 
-  def user_link(user, **link_opts)
-    if deleted_user?(user)
-      link_to 'deleted user', '#', { dir: 'ltr' }.merge(link_opts)
+  def user_link(user, url_opts = nil, **link_opts)
+    url_opts ||= {}
+    anchortext = link_opts[:anchortext]
+    link_opts_reduced = { dir: 'ltr' }.merge(link_opts).except(:anchortext)
+    if !anchortext.nil?
+      link_to anchortext, user_url(user, **url_opts), { dir: 'ltr' }.merge(link_opts)
+    elsif deleted_user?(user)
+      link_to 'deleted user', '#', link_opts_reduced
+    else
+      link_to user.rtl_safe_username, user_url(user, **url_opts), link_opts_reduced
+    end
+  end
+
+  def sso_sign_in_enabled?
+    SiteSetting['SsoSignIn']
+  end
+
+  def devise_sign_in_enabled?
+    SiteSetting['MixedSignIn'] || !sso_sign_in_enabled?
+  end
+
+  ##
+  # Returns a user corresponding to the ID provided, with the caveat that if +user_id+ is 'me' and there is a user
+  # signed in, the signed in user will be returned. Use for /users/me links.
+  # @param [String] user_id The user ID to find, from +params+
+  # @return [User] The User object
+  def user_with_me(user_id)
+    if user_id == 'me' && user_signed_in?
+      current_user
     else
-      link_to user.rtl_safe_username, user_url(user), { dir: 'ltr' }.merge(link_opts)
+      User.find(user_id)
     end
   end
 end
diff --git a/app/jobs/application_job.rb b/app/jobs/application_job.rb
new file mode 100644
index 0000000000000000000000000000000000000000..d394c3d106230849e4ff6f4020d41b3e62589061
--- /dev/null
+++ b/app/jobs/application_job.rb
@@ -0,0 +1,7 @@
+class ApplicationJob < ActiveJob::Base
+  # Automatically retry jobs that encountered a deadlock
+  # retry_on ActiveRecord::Deadlocked
+
+  # Most jobs are safe to ignore if the underlying records are no longer available
+  # discard_on ActiveJob::DeserializationError
+end
diff --git a/app/jobs/clean_up_spammy_users_job.rb b/app/jobs/clean_up_spammy_users_job.rb
new file mode 100644
index 0000000000000000000000000000000000000000..001afda5f5678ca4e0654e68a544a330afb7abec
--- /dev/null
+++ b/app/jobs/clean_up_spammy_users_job.rb
@@ -0,0 +1,22 @@
+class CleanUpSpammyUsersJob < ApplicationJob
+  queue_as :default
+
+  def perform(created_after: 1.month.ago)
+    # Select potential spammers: users created within timeframe, who are not deleted, who have posted but all posts have
+    # since been deleted (no live posts).
+    possible_spammers = User.joins('inner join posts on users.id = posts.user_id')
+                            .where('users.created_at >= ?', created_after)
+                            .where(users: { deleted: false }).group('users.id').having('count(posts.id) > 0')
+                            .having('count(distinct if(posts.deleted = true, null, posts.id)) = 0')
+    possible_spammers.each do |spammer|
+      all_posts_spam = spammer.posts.all? do |post|
+        # A post is considered spam if there are any helpful spam flags on it.
+        post.flags.any? { |flag| flag.post_flag_type.name == "it's spam" && flag.status == 'helpful' }
+      end
+      if all_posts_spam
+        spammer.block('automatic block from spam cleanup job', length: 2.years)
+        spammer.do_soft_delete(User.find(-1))
+      end
+    end
+  end
+end
diff --git a/app/jobs/cleanup_votes_job.rb b/app/jobs/cleanup_votes_job.rb
new file mode 100644
index 0000000000000000000000000000000000000000..b54b26be7c2f817056474af7ae57c09cbfe91a1a
--- /dev/null
+++ b/app/jobs/cleanup_votes_job.rb
@@ -0,0 +1,32 @@
+class CleanupVotesJob < ApplicationJob
+  queue_as :default
+
+  def perform
+    Community.all.each do |c|
+      RequestContext.community = c
+      orphan_votes = Vote.all.reject { |v| v.post.present? }
+
+      puts "[#{c.name}] destroying #{orphan_votes.length} #{'orphan vote'.pluralize(orphan_votes.length)}"
+
+      system_user = User.find(-1)
+
+      orphan_votes.each do |v|
+        result = v.destroy
+
+        if result
+          AuditLog.admin_audit(
+            comment: "Deleted orphaned vote for user ##{v.recv_user_id} " \
+                     "on post ##{v.post_id} " \
+                     "in community ##{c.id} (#{c.name})",
+            event_type: 'vote_delete',
+            related: v,
+            user: system_user
+          )
+        else
+          puts "[#{c.name}] failed to destroy vote \"#{v.id}\""
+          v.errors.each { |e| puts e.full_message }
+        end
+      end
+    end
+  end
+end
diff --git a/app/mailers/admin_mailer.rb b/app/mailers/admin_mailer.rb
index a68f9c55c2c0aa6026a96efeeeab838cdf961377..f4f0d2606bb89bb3652405e637620e2e370013f5 100644
--- a/app/mailers/admin_mailer.rb
+++ b/app/mailers/admin_mailer.rb
@@ -1,5 +1,8 @@
 class AdminMailer < ApplicationMailer
-  default from: 'Codidact Admins <moderators-noreply@codidact.com>'
+  default from: lambda {
+                  "#{SiteSetting['ModeratorDistributionListSenderName']} " \
+                    "<#{SiteSetting['ModeratorDistributionListSenderEmail']}>"
+                }
 
   def to_moderators
     @subject = params[:subject]
@@ -8,6 +11,19 @@ class AdminMailer < ApplicationMailer
             "INNER JOIN community_users cu ON cu.user_id = u.id WHERE s.type = 'moderators' AND " \
             '(u.is_global_admin = 1 OR u.is_global_moderator = 1 OR cu.is_admin = 1 OR cu.is_moderator = 1)'
     emails = ActiveRecord::Base.connection.execute(query).to_a.flatten
-    mail subject: "Codidact Moderators: #{@subject}", to: 'moderators-noreply@codidact.org', bcc: emails
+    from = "#{SiteSetting['ModeratorDistributionListSenderName']} " \
+           "<#{SiteSetting['ModeratorDistributionListSenderEmail']}>"
+    to = SiteSetting['ModeratorDistributionListSenderEmail']
+    mail subject: "Codidact Moderators: #{@subject}", to: to, from: from, bcc: emails
+  end
+
+  def to_all_users
+    @subject = params[:subject]
+    @body_markdown = params[:body_markdown]
+    @users = User.where('email NOT LIKE ?', '%localhost').select(:email).map(&:email)
+    to = SiteSetting['AllUsersSenderEmail']
+    from = "#{SiteSetting['AllUsersSenderName']} <#{SiteSetting['AllUsersSenderEmail']}>"
+    reply_to = SiteSetting['AllUsersReplyToEmail']
+    mail subject: @subject, to: to, from: from, reply_to: reply_to, bcc: @users
   end
 end
diff --git a/app/mailers/application_mailer.rb b/app/mailers/application_mailer.rb
index 59f7ed59f28d7fa1dda12bea2986ad23fdba3c37..45f5f0f87aaf4248d7cb606a8d0c8d313a52f0c4 100644
--- a/app/mailers/application_mailer.rb
+++ b/app/mailers/application_mailer.rb
@@ -1,4 +1,4 @@
 class ApplicationMailer < ActionMailer::Base
-  default from: 'Codidact <noreply@codidact.com>'
+  default from: -> { "#{SiteSetting['NoReplySenderName']} <#{SiteSetting['NoReplySenderEmail']}>" }
   layout 'mailer'
 end
diff --git a/app/mailers/donation_mailer.rb b/app/mailers/donation_mailer.rb
index ee6b79854b0007e56d6b2c22ac876981ddce1a9a..587ae6009b5aef75e1b21d65b591d046e0374327 100644
--- a/app/mailers/donation_mailer.rb
+++ b/app/mailers/donation_mailer.rb
@@ -4,9 +4,9 @@ class DonationMailer < ApplicationMailer
     @amount = params[:amount]
     @email = params[:email]
     @name = params[:name]
-    mail from: 'Codidact Donations <donations-support@codidact.com>',
-         reply_to: 'Codidact Support <support@codidact.com>',
-         to: @email, subject: 'Thanks for your donation!'
+    from = "#{SiteSetting['DonationSenderName']} <#{SiteSetting['DonationSenderEmail']}>"
+    reply_to = "#{SiteSetting['DonationSupportReceiverName']} <#{SiteSetting['DonationSupportReceiverEmail']}>"
+    mail from: from, reply_to: reply_to, to: @email, subject: 'Thanks for your donation!'
   end
 
   def donation_uncaptured
@@ -15,8 +15,8 @@ class DonationMailer < ApplicationMailer
     @email = params[:email]
     @name = params[:name]
     @intent = params[:intent]
-    mail from: 'Codidact Donations <donations-support@codidact.com>',
-         reply_to: 'Codidact Support <support@codidact.com>',
-         to: @email, subject: 'Your donation is unfinished - was everything okay?'
+    from = "#{SiteSetting['DonationSenderName']} <#{SiteSetting['DonationSenderEmail']}>"
+    reply_to = "#{SiteSetting['DonationSupportReceiverName']} <#{SiteSetting['DonationSupportReceiverEmail']}>"
+    mail from: from, reply_to: reply_to, to: @email, subject: 'Your donation is unfinished - was everything okay?'
   end
 end
diff --git a/app/mailers/flag_mailer.rb b/app/mailers/flag_mailer.rb
index 9eb0f716815b41f80690fc33be168dd9e0cfe19e..b43d60310562ddfaddb92c0e9b93b62fa3cb0d77 100644
--- a/app/mailers/flag_mailer.rb
+++ b/app/mailers/flag_mailer.rb
@@ -7,7 +7,8 @@ class FlagMailer < ApplicationMailer
                  .or(User.joins(:community_user)
                          .where(community_users: { is_admin: true, community_id: @flag.community_id }))
                  .select(:email).map(&:email)
-    mail from: 'Codidact <noreply@codidact.com>', to: 'noreply@codidact.com', bcc: emails,
-         subject: "New flag escalation on #{@flag.community.name}"
+    from = "#{SiteSetting['NoReplySenderName']} <#{SiteSetting['NoReplySenderEmail']}>"
+    to = SiteSetting['NoReplySenderEmail']
+    mail from: from, to: to, bcc: emails, subject: "New flag escalation on #{@flag.community.name}"
   end
 end
diff --git a/app/mailers/subscription_mailer.rb b/app/mailers/subscription_mailer.rb
index 2c6687890c4dbe19d11644f1b4136fa730b0482d..6261e13bf365b2f7fd2183c43ae307a2b97e004c 100644
--- a/app/mailers/subscription_mailer.rb
+++ b/app/mailers/subscription_mailer.rb
@@ -1,4 +1,6 @@
 class SubscriptionMailer < ApplicationMailer
+  helper UsersHelper
+
   def subscription
     @subscription = params[:subscription]
     @questions = @subscription.questions&.includes(:user) || []
@@ -9,14 +11,20 @@ class SubscriptionMailer < ApplicationMailer
       return
     end
 
+    # Load request community to ensure we can access the settings/posts of the correct community
+    RequestContext.community = @subscription.community
+    name = @subscription.name
     site_name = @subscription.community.name
-    subject = if @subscription.name.present?
-                "Latest questions from your '#{@subscription.name}' subscription on #{site_name}"
+    subject = if name.present?
+                "Latest questions from your '#{name}' subscription on #{site_name}"
               else
                 "Latest questions from your subscription on #{site_name}"
               end
 
     @subscription.update(last_sent_at: DateTime.now)
-    mail from: 'Codidact Subscriptions <subscriptions@codidact.com>', to: @subscription.user.email, subject: subject
+    from = "#{SiteSetting['SubscriptionSenderName']} <#{SiteSetting['SubscriptionSenderEmail']}>"
+    to = @subscription.user.email
+    mail from: from, to: to, subject: subject
+    Rails.logger.info "Sent subscription mail (sub ID ##{@subscription.id}, to: '#{to}', name: '#{name}'"
   end
 end
diff --git a/app/mailers/two_factor_mailer.rb b/app/mailers/two_factor_mailer.rb
index de656dd7c5731846ca80d4f9684d0791fe2201e8..2784ab9be7961c16c4053ded1785b97a703fc4b1 100644
--- a/app/mailers/two_factor_mailer.rb
+++ b/app/mailers/two_factor_mailer.rb
@@ -1,5 +1,5 @@
 class TwoFactorMailer < ApplicationMailer
-  default from: 'Codidact <noreply@codidact.com>'
+  default from: -> { "#{SiteSetting['NoReplySenderName']} <#{SiteSetting['NoReplySenderEmail']}>" }
 
   def disable_email
     user = params[:user]
@@ -16,4 +16,10 @@ class TwoFactorMailer < ApplicationMailer
     user.update(login_token: @token, login_token_expires_at: 5.minutes.from_now)
     mail to: user.email, subject: 'Your sign in link for Codidact'
   end
+
+  def backup_code
+    @user = params[:user]
+    @host = params[:host]
+    mail to: @user.email, subject: 'Your 2FA backup code for Codidact'
+  end
 end
diff --git a/app/models/application_record.rb b/app/models/application_record.rb
index 0e552701f6f5cd9c2a9182f716b16f79ad79b75f..2f0d184b0756bfaa7c6d3d36babadfdc66a17114 100644
--- a/app/models/application_record.rb
+++ b/app/models/application_record.rb
@@ -107,11 +107,11 @@ module UserSortable
   end
 end
 
-klasses = [::ActiveRecord::Relation]
-klasses << if defined? ::ActiveRecord::Associations::CollectionProxy
-             ::ActiveRecord::Associations::CollectionProxy
+klasses = [ActiveRecord::Relation]
+klasses << if defined? ActiveRecord::Associations::CollectionProxy
+             ActiveRecord::Associations::CollectionProxy
            else
-             ::ActiveRecord::Associations::AssociationCollection
+             ActiveRecord::Associations::AssociationCollection
            end
 
 ActiveRecord::Base.extend UserSortable
diff --git a/app/models/category.rb b/app/models/category.rb
index 9cdaea7650374ce3646ce8240c7c98569511aa67..f5951519cd7ded62a96323dd016169eba8632578 100644
--- a/app/models/category.rb
+++ b/app/models/category.rb
@@ -9,6 +9,7 @@ class Category < ApplicationRecord
   has_many :posts
   belongs_to :tag_set
   belongs_to :license
+  belongs_to :default_filter, class_name: 'Filter', optional: true
 
   serialize :display_post_types, Array
 
diff --git a/app/models/category_filter_default.rb b/app/models/category_filter_default.rb
new file mode 100644
index 0000000000000000000000000000000000000000..540f1cf789ba384dff21a266e47f15282d89426f
--- /dev/null
+++ b/app/models/category_filter_default.rb
@@ -0,0 +1,5 @@
+class CategoryFilterDefault < ApplicationRecord
+  belongs_to :user
+  belongs_to :filter
+  belongs_to :category
+end
diff --git a/app/models/community_user.rb b/app/models/community_user.rb
index a62926dcc7b8e55872e1f63f3dd5fb7ad404b753..f0a80e60fc035ca7e38489593e46f3a8a0076b88 100644
--- a/app/models/community_user.rb
+++ b/app/models/community_user.rb
@@ -14,6 +14,12 @@ class CommunityUser < ApplicationRecord
 
   after_create :prevent_ulysses_case
 
+  delegate :url_helpers, to: 'Rails.application.routes'
+
+  def system?
+    user_id == -1
+  end
+
   def suspended?
     return true if is_suspended && !suspension_end.past?
 
@@ -24,6 +30,10 @@ class CommunityUser < ApplicationRecord
     false
   end
 
+  def latest_warning
+    mod_warnings&.order(created_at: 'desc')&.first&.created_at
+  end
+
   # Calculation functions for privilege scores
   # These are quite expensive, so we'll cache them for a while
   def post_score
@@ -54,8 +64,6 @@ class CommunityUser < ApplicationRecord
     end
   end
 
-  ## Privilege functions
-
   def privilege?(internal_id, ignore_suspension: false, ignore_mod: false)
     if internal_id != 'mod' && !ignore_mod && user.is_moderator
       return true # includes: privilege? 'mod'
@@ -73,11 +81,26 @@ class CommunityUser < ApplicationRecord
     UserAbility.joins(:ability).where(community_user_id: id, abilities: { internal_id: internal_id }).first
   end
 
-  def grant_privilege(internal_id)
+  ##
+  # Grant a specified ability to this CommunityUser.
+  # @param internal_id [String] The +internal_id+ of the ability to grant.
+  # @param notify [Boolean] Whether to send a notification to the user.
+  def grant_privilege!(internal_id, notify: true)
     priv = Ability.where(internal_id: internal_id).first
     UserAbility.create community_user_id: id, ability: priv
+    if notify
+      community_host = priv.community.host
+      user.create_notification("You've earned the #{priv.name} ability! Learn more.",
+                               url_helpers.ability_url(priv.internal_id, host: community_host))
+    end
   end
 
+  ##
+  # Recalculate a specified ability for this CommunityUser. Will not revoke abilities that have already been granted.
+  # @param internal_id [String] The +internal_id+ of the ability to be recalculated.
+  # @param sandbox [Boolean] Whether to run in sandbox mode - if sandboxed, the ability will not be granted but the
+  #   return value indicates whether it would have been.
+  # @return [Boolean] Whether or not the ability was granted.
   def recalc_privilege(internal_id, sandbox: false)
     # Do not recalculate privileges already granted
     return true if privilege?(internal_id, ignore_suspension: true, ignore_mod: false)
@@ -96,17 +119,27 @@ class CommunityUser < ApplicationRecord
     end
 
     # If not sandbox mode, create new privilege entry
-    grant_privilege(internal_id) unless sandbox
+    grant_privilege!(internal_id) unless sandbox
     recalc_trust_level unless sandbox
     true
   end
 
+  ##
+  # Recalculate a list of standard abilities for this CommunityUser.
+  # @param sandbox [Boolean] Whether to run in sandbox mode - see {#recalc_privilege}.
+  # @return [Array<Boolean>]
   def recalc_privileges(sandbox: false)
     [:everyone, :unrestricted, :edit_posts, :edit_tags, :flag_close, :flag_curate].map do |ability|
       recalc_privilege(ability, sandbox: sandbox)
     end
   end
 
+  alias ability? privilege?
+  alias ability privilege
+  alias grant_ability! grant_privilege!
+  alias recalc_ability recalc_privilege
+  alias recalc_abilities recalc_privileges
+
   # This check makes sure that every user gets the
   # 'everyone' permission upon creation. We do not want
   # to create a no permissions user by accident.
diff --git a/app/models/concerns/edits_validations.rb b/app/models/concerns/edits_validations.rb
new file mode 100644
index 0000000000000000000000000000000000000000..11b4b98fd5b46305584d3173c4bcf5e1e51a412f
--- /dev/null
+++ b/app/models/concerns/edits_validations.rb
@@ -0,0 +1,21 @@
+# Common validations for unilateral & suggested edits
+module EditsValidations
+  extend ActiveSupport::Concern
+
+  included do
+    validate :max_edit_comment_length
+  end
+
+  def max_edit_comment_length
+    if comment.nil?
+      return
+    end
+
+    max_edit_comment_length = SiteSetting['MaxEditCommentLength']
+    max_length = [(max_edit_comment_length || 255), 255].min
+    if comment.length > max_length
+      msg = I18n.t('edits.max_edit_comment_length', { count: max_length }).gsub(':length', max_length.to_s)
+      errors.add(:base, msg)
+    end
+  end
+end
diff --git a/app/models/concerns/post_validations.rb b/app/models/concerns/post_validations.rb
new file mode 100644
index 0000000000000000000000000000000000000000..a0986b345f9da939744c7702a0d73094d11e8b29
--- /dev/null
+++ b/app/models/concerns/post_validations.rb
@@ -0,0 +1,84 @@
+# Validations for posts which are shared between posts and suggested edits.
+module PostValidations
+  extend ActiveSupport::Concern
+
+  included do
+    validate :tags_in_tag_set, if: -> { post_type.has_tags }
+    validate :maximum_tags, if: -> { post_type.has_tags }
+    validate :maximum_tag_length, if: -> { post_type.has_tags }
+    validate :no_spaces_in_tags, if: -> { post_type.has_tags }
+    validate :stripped_minimum_body, if: -> { !body_markdown.nil? }
+    validate :stripped_minimum_title, if: -> { !title.nil? }
+    validate :maximum_title_length, if: -> { !title.nil? }
+    validate :required_tags?, if: -> { post_type.has_tags && post_type.has_category }
+  end
+
+  def maximum_tags
+    if tags_cache.length > 5
+      errors.add(:base, "Post can't have more than 5 tags.")
+    elsif tags_cache.empty?
+      errors.add(:base, 'Post must have at least one tag.')
+    end
+  end
+
+  def maximum_tag_length
+    tags_cache.each do |tag|
+      max_len = SiteSetting['MaxTagLength']
+      if tag.length > max_len
+        errors.add(:tags, "can't be more than #{max_len} characters long each")
+      end
+    end
+  end
+
+  def no_spaces_in_tags
+    tags_cache.each do |tag|
+      if tag.include?(' ') || tag.include?('_')
+        errors.add(:tags, 'may not include spaces or underscores - use hyphens for multiple-word tags')
+      end
+    end
+  end
+
+  def stripped_minimum_body
+    min_body = category.nil? ? 30 : category.min_body_length
+    if (body_markdown&.gsub(/(?:^[\s\t\u2000-\u200F]+|[\s\t\u2000-\u200F]+$)/, '')&.length || 0) < min_body
+      errors.add(:body, "must be more than #{min_body} non-whitespace characters long")
+    end
+  end
+
+  def stripped_minimum_title
+    min_title = if ['HelpDoc', 'PolicyDoc'].include?(post_type.name)
+                  1
+                elsif category.nil?
+                  15
+                else
+                  category.min_title_length
+                end
+
+    if (title&.gsub(/(?:^[\s\t\u2000-\u200F]+|[\s\t\u2000-\u200F]+$)/, '')&.length || 0) < min_title
+      errors.add(:title, "must be more than #{min_title} non-whitespace characters long")
+    end
+  end
+
+  def maximum_title_length
+    max_title_len = SiteSetting['MaxTitleLength']
+    if title.length > [(max_title_len || 255), 255].min
+      errors.add(:title, "can't be more than #{max_title_len} characters")
+    end
+  end
+
+  def tags_in_tag_set
+    tag_set = category.tag_set
+    unless tags.all? { |t| t.tag_set_id == tag_set.id }
+      errors.add(:base, "Not all of this question's tags are in the correct tag set.")
+    end
+  end
+
+  def required_tags?
+    required = category&.required_tag_ids
+    return unless required.present? && !required.empty?
+
+    unless tag_ids.any? { |t| required.include? t }
+      errors.add(:tags, "must contain at least one required tag (#{category.required_tags.pluck(:name).join(', ')})")
+    end
+  end
+end
diff --git a/app/models/concerns/saml_init.rb b/app/models/concerns/saml_init.rb
new file mode 100644
index 0000000000000000000000000000000000000000..3709b87ecc559c5716bf3c0355ebee26514805a9
--- /dev/null
+++ b/app/models/concerns/saml_init.rb
@@ -0,0 +1,128 @@
+# Module for saml based initalization.
+#
+# The saml_init_email method is used to initialize the email address after a successful SSO sign in.
+# The saml_init_identifier method is used to
+module SamlInit
+  extend ActiveSupport::Concern
+
+  included do
+    has_one :sso_profile, required: false, autosave: true, dependent: :destroy
+
+    before_validation :prepare_from_saml, if: -> { saml_identifier.present? }
+  end
+
+  # -----------------------------------------------------------------------------------------------
+  # Identifier
+  # -----------------------------------------------------------------------------------------------
+
+  # @return [String, Nil] the saml_identifier of this user, or nil if the user is not from SSO
+  def saml_identifier
+    sso_profile&.saml_identifier
+  end
+
+  # @param saml_identifier [String, Nil] sets (or clears) the saml_identifier of this user
+  def saml_identifier=(saml_identifier)
+    if saml_identifier.nil?
+      sso_profile&.destroy
+    else
+      build_sso_profile if sso_profile.nil?
+      sso_profile.saml_identifier = saml_identifier
+    end
+  end
+
+  # This method is added as a fallback to support the Single Logout Service.
+  #
+  # @return [String, Nil] the saml_identifier of this user, or nil if the user is not from SSO
+  # @see #saml_identifier
+  def saml_init_identifier
+    saml_identifier
+  end
+
+  # Sets the saml_identifier to the given saml_identifier upon initialization. In contrast to
+  # #saml_identifier=, this method does not delete the SSO profile in case the saml_identifier is
+  # not present (safety in case of SSO issues).
+  #
+  # @param saml_identifier [String, Nil] the saml_identifier
+  # @return [String, Nil] the saml_identifier of this user, should never be nil
+  def saml_init_identifier=(saml_identifier)
+    build_sso_profile if sso_profile.nil?
+
+    # Only update if non-empty
+    sso_profile.saml_identifier = saml_identifier if saml_identifier.present?
+  end
+
+  # -----------------------------------------------------------------------------------------------
+  # Email
+  # -----------------------------------------------------------------------------------------------
+
+  # This method is added as a fallback to support the Single Logout Service.
+  # @return [String, Nil] the email address of this user, or nil if the user is not from SSO
+  def saml_init_email
+    return nil if sso_profile.nil?
+
+    email
+  end
+
+  # Initializes email address, and prevents (re)confirmation in case it is changed.
+  #
+  # @param email [String] the email address
+  def saml_init_email=(email)
+    self.email = email
+    skip_confirmation!
+    skip_reconfirmation!
+  end
+
+  # -----------------------------------------------------------------------------------------------
+  # Email is identifier
+  # -----------------------------------------------------------------------------------------------
+
+  # Used in the case that email is the unique identifier from saml.
+  # @return [String, Nil] the email address of the user, or nil in the case the user is not from SSO
+  def saml_init_email_and_identifier
+    return nil if sso_profile.nil?
+
+    email
+  end
+
+  # Used in the case that email is the unique identifier from saml.
+  #
+  # @param email [String] the email address (and saml identifier)
+  def saml_init_email_and_identifier=(email)
+    self.saml_init_email = email
+    self.saml_init_identifier = email
+  end
+
+  # -----------------------------------------------------------------------------------------------
+  # Username
+  # -----------------------------------------------------------------------------------------------
+
+  # This method is added as fallback to support the Single Logout Service.
+  # @return [String] the username
+  def saml_init_username_no_update
+    username
+  end
+
+  # Sets the username from SAML in case it was not already set.
+  # This prevents overriding the user set username with the one from SAML all the time, while
+  # allowing for email updates to be applied.
+  #
+  # @param username [String] the username to set
+  def saml_init_username_no_update=(username)
+    self.username = username unless self.username.present?
+  end
+
+  # -----------------------------------------------------------------------------------------------
+  # Creation
+  # -----------------------------------------------------------------------------------------------
+
+  protected
+
+  # Prepare a (potentially) new user from saml for creation. If the user is actually new, a random
+  # password is created for them and email confirmation is skipped.
+  def prepare_from_saml
+    return unless new_record?
+
+    self.password = SecureRandom.hex
+    skip_confirmation!
+  end
+end
diff --git a/app/models/email_log.rb b/app/models/email_log.rb
new file mode 100644
index 0000000000000000000000000000000000000000..9b171ddbc28ab47d54c5a038ccc3b32cff38a25b
--- /dev/null
+++ b/app/models/email_log.rb
@@ -0,0 +1,2 @@
+class EmailLog < ApplicationRecord
+end
diff --git a/app/models/filter.rb b/app/models/filter.rb
new file mode 100644
index 0000000000000000000000000000000000000000..fb11e1d47779ab658d296eb0e5752903446afbaf
--- /dev/null
+++ b/app/models/filter.rb
@@ -0,0 +1,7 @@
+class Filter < ApplicationRecord
+  belongs_to :user
+  has_many :category_filter_defaults, dependent: :destroy
+  validates :name, uniqueness: { scope: :user }
+  serialize :include_tags, Array
+  serialize :exclude_tags, Array
+end
diff --git a/app/models/post.rb b/app/models/post.rb
index 458b91df77669d4f2798a1c314f407d5a6c880cf..4a1d1ba9943e6daed4a472c68f22afdae6a942f9 100644
--- a/app/models/post.rb
+++ b/app/models/post.rb
@@ -1,5 +1,6 @@
 class Post < ApplicationRecord
   include CommunityRelated
+  include PostValidations
 
   belongs_to :user, optional: true
   belongs_to :post_type
@@ -24,22 +25,20 @@ class Post < ApplicationRecord
   has_many :reactions
 
   counter_culture :parent, column_name: proc { |model| model.deleted? ? nil : 'answer_count' }
+  counter_culture [:user, :community_user], column_name: proc { |model| model.deleted? ? nil : 'post_count' }
 
   serialize :tags_cache, Array
 
-  validates :body, presence: true, length: { minimum: 30, maximum: 30_000 }
+  validates :body, presence: true, length: { maximum: 30_000 }
   validates :doc_slug, uniqueness: { scope: [:community_id], case_sensitive: false }, if: -> { doc_slug.present? }
-  validates :title, :body, :tags_cache, presence: true, if: -> { post_type.has_tags }
-  validate :tags_in_tag_set, if: -> { post_type.has_tags }
-  validate :maximum_tags, if: -> { post_type.has_tags }
-  validate :maximum_tag_length, if: -> { post_type.has_tags }
-  validate :no_spaces_in_tags, if: -> { post_type.has_tags }
-  validate :stripped_minimum, if: -> { post_type.has_tags }
-  validate :maximum_title_length, if: -> { post_type.has_tags }
+  validates :title, presence: true, if: -> { post_type.is_top_level? }
+  validates :tags_cache, presence: true, if: -> { post_type.has_tags }
+
   validate :category_allows_post_type, if: -> { category_id.present? }
   validate :license_valid, if: -> { post_type.has_license }
-  validate :required_tags?, if: -> { post_type.has_tags && post_type.has_category }
-  validate :moderator_tags, if: -> { post_type.has_tags && post_type.has_category }
+  validate :moderator_tags, if: -> { post_type.has_tags && post_type.has_category && tags_cache_changed? }
+
+  # Other validations (shared with suggested edits) are in concerns/PostValidations
 
   scope :undeleted, -> { where(deleted: false) }
   scope :deleted, -> { where(deleted: true) }
@@ -55,13 +54,32 @@ class Post < ApplicationRecord
   after_save :modify_author_reputation
   after_save :copy_last_activity_to_parent
   after_save :break_description_cache
-  after_save :update_category_activity, if: -> { post_type.has_category }
+  after_save :update_category_activity, if: -> { post_type.has_category && !destroyed? }
   after_save :recalc_score
 
+  # @param term [String] the search term
+  # @return [ActiveRecord::Relation<Post>]
   def self.search(term)
     match_search term, posts: :body_markdown
   end
 
+  def self.by_slug(slug, user)
+    post = Post.unscoped.where(
+      doc_slug: slug,
+      community_id: [RequestContext.community_id, nil]
+    ).first
+
+    if post&.help_category == '$Disabled'
+      return nil
+    end
+
+    if post&.help_category == '$Moderator' && !user&.is_moderator
+      return nil
+    end
+
+    post
+  end
+
   # Double-define: initial definitions are less efficient, so if we have a record of the post type we'll
   # override them later with more efficient methods.
   ['Question', 'Answer', 'PolicyDoc', 'HelpDoc', 'Article'].each do |pt|
@@ -76,14 +94,18 @@ class Post < ApplicationRecord
     end
   end
 
+  # @return [TagSet]
   def tag_set
     parent.nil? ? category.tag_set : parent.category.tag_set
   end
 
+  # @return [Boolean]
   def meta?
     false
   end
 
+  # Used in the transfer of content from SE to reassign the owner of a post to the given user.
+  # @param new_user [User]
   def reassign_user(new_user)
     new_user.ensure_community_user!
 
@@ -94,10 +116,13 @@ class Post < ApplicationRecord
     update!(deleted: false, deleted_at: nil, deleted_by: nil)
   end
 
+  # Removes the attribution notice from this post
+  # @return [Boolean] whether the action was successful
   def remove_attribution_notice!
     update(att_source: nil, att_license_link: nil, att_license_name: nil)
   end
 
+  # @return [String] the type of the last activity on this post
   def last_activity_type
     case last_activity
     when closed_at
@@ -121,37 +146,51 @@ class Post < ApplicationRecord
     end
   end
 
+  # @return [String] the body with all markdown stripped
   def body_plain
     ApplicationController.helpers.strip_markdown(body_markdown)
   end
 
+  # @return [Boolean] whether this post is a question
   def question?
     post_type_id == Question.post_type_id
   end
 
+  # @return [Boolean] whether this post is an answer
   def answer?
     post_type_id == Answer.post_type_id
   end
 
+  # @return [Boolean] whether this post is an article
   def article?
     post_type_id == Article.post_type_id
   end
 
+  # @return [Boolean] whether there is a suggested edit pending for this post
   def pending_suggested_edit?
     SuggestedEdit.where(post_id: id, active: true).any?
   end
 
+  # @return [SuggestedEdit, Nil] the suggested edit pending for this post (if any)
   def pending_suggested_edit
     SuggestedEdit.where(post_id: id, active: true).last
   end
 
+  # Recalculates the score of this post based on its up and downvotes
   def recalc_score
     variable = SiteSetting['ScoringVariable'] || 2
     sql = 'UPDATE posts SET score = (upvote_count + ?) / (upvote_count + downvote_count + (2 * ?)) WHERE id = ?'
     sanitized = ActiveRecord::Base.sanitize_sql_array([sql, variable, variable, id])
     ActiveRecord::Base.connection.execute sanitized
+
+    # ensures the updated score is immediately available
+    self.score = (upvote_count + variable).to_f / (upvote_count + downvote_count + (2 * variable))
+    # prevents AR from accidentally saving the dirty state
+    clear_attribute_changes([:score])
   end
 
+  # This method will update the locked status of this post if locked_until is in the past.
+  # @return [Boolean] whether this post is locked
   def locked?
     return true if locked && locked_until.nil? # permanent lock
     return true if locked && !locked_until.past?
@@ -159,14 +198,27 @@ class Post < ApplicationRecord
     if locked
       update(locked: false, locked_by: nil, locked_at: nil, locked_until: nil)
     end
+
+    false
+  end
+
+  # The test here is for flags that are pending (no status). A spam flag
+  # could be marked helpful but the post wouldn't be deleted, and
+  # we don't necessarily want the post to be treated like it's a spam risk
+  # if that happens.
+  def spam_flag_pending?
+    flags.any? { |flag| flag.post_flag_type&.name == "it's spam" && !flag.status }
   end
 
+  # @param user [User, Nil]
+  # @return [Boolean] whether the given user can view this post
   def can_access?(user)
     (!deleted? || user&.has_post_privilege?('flag_curate', self)) &&
       (!category.present? || !category.min_view_trust_level.present? ||
         category.min_view_trust_level <= (user&.trust_level || 0))
   end
 
+  # @return [Hash] a hash with as key the reaction type and value the amount of reactions for that type
   def reaction_list
     reactions.includes(:reaction_type).group_by(&:reaction_type_id)
              .to_h { |_k, v| [v.first.reaction_type, v] }
@@ -174,12 +226,21 @@ class Post < ApplicationRecord
 
   private
 
+  ##
+  # Before-validation callback. Update the tags association from the tags_cache.
   def update_tag_associations
     tags_cache.each do |tag_name|
-      tag = Tag.find_or_create_by name: tag_name, tag_set: category.tag_set
+      tag, name_used = Tag.find_or_create_synonymized name: tag_name, tag_set: category.tag_set
       unless tags.include? tag
         tags << tag
       end
+
+      # If the tags_cache doesn't include name_used then tag_name was a synonym - remove the synonym from tags_cache
+      # and add the primary for it instead.
+      unless tags_cache.include? name_used
+        tags_cache.delete tag_name
+        tags_cache << name_used
+      end
     end
     tags.each do |tag|
       unless tags_cache.include? tag.name
@@ -188,11 +249,20 @@ class Post < ApplicationRecord
     end
   end
 
+  ##
+  # Helper method for #check_attribution_notice validator. Produces a text-only attribution notice either based on
+  # values given or the current state of the post for use in post histories.
+  # @param source [String, Nil] where the post originally came from
+  # @param name [String, Nil] the name of the license
+  # @param url [String, Nil] the url of the license
+  # @return [String] an attribution notice corresponding to this post
   def attribution_text(source = nil, name = nil, url = nil)
     "Source: #{source || att_source}\nLicense name: #{name || att_license_name}\n" \
       "License URL: #{url || att_license_link}"
   end
 
+  # Intended to be called as callback after a save.
+  # If changes were made to the licensing of this post, this will insert the correct history items.
   def check_attribution_notice
     sc = saved_changes
     attributes = ['att_source', 'att_license_name', 'att_license_link']
@@ -210,6 +280,8 @@ class Post < ApplicationRecord
     end
   end
 
+  # Intended to be called as callback after a save.
+  # If the last activity of this post was changed and it has a parent, also updates the parent activity
   def copy_last_activity_to_parent
     sc = saved_changes
     if parent.present? && (sc.include?('last_activity') || sc.include?('last_activity_by_id')) \
@@ -218,22 +290,28 @@ class Post < ApplicationRecord
     end
   end
 
+  # Intended to be called as callback after a save.
+  # If this deletion status of this post was changed, then remove or re-add the reputation.
   def modify_author_reputation
     sc = saved_changes
     if sc.include?('deleted') && sc['deleted'][0] != sc['deleted'][1] && created_at >= 60.days.ago
       deleted = !!saved_changes['deleted']&.last
       if deleted
-        user.update(reputation: user.reputation - Vote.total_rep_change(votes))
+        user&.update(reputation: (user&.reputation || 1) - Vote.total_rep_change(votes))
       else
-        user.update(reputation: user.reputation + Vote.total_rep_change(votes))
+        user&.update(reputation: (user&.reputation || 1) + Vote.total_rep_change(votes))
       end
     end
   end
 
+  # Intended to be called as callback after a save.
+  # @return [PostHistory] creates an initial revision for this post
   def create_initial_revision
     PostHistory.initial_revision(self, user, after: body_markdown, after_title: title, after_tags: tags)
   end
 
+  # Intended to be used as validation.
+  # Will add an error if this post's post type is not allowed in the associated category.
   def category_allows_post_type
     return if category.nil?
 
@@ -242,6 +320,8 @@ class Post < ApplicationRecord
     end
   end
 
+  # Intended to be called as callback after a save.
+  # Deletes this posts description from the cache such that it will be regenerated next time it is needed.
   def break_description_cache
     Rails.cache.delete "posts/#{id}/description"
     if parent_id.present?
@@ -249,6 +329,8 @@ class Post < ApplicationRecord
     end
   end
 
+  # Intended to be used as a validation.
+  # Checks whether the associated license is present and enabled.
   def license_valid
     # Don't validate license on edits
     return unless id.nil?
@@ -263,63 +345,9 @@ class Post < ApplicationRecord
     end
   end
 
-  def maximum_tags
-    if tags_cache.length > 5
-      errors.add(:base, "Post can't have more than 5 tags.")
-    elsif tags_cache.empty?
-      errors.add(:base, 'Post must have at least one tag.')
-    end
-  end
-
-  def maximum_tag_length
-    tags_cache.each do |tag|
-      max_len = SiteSetting['MaxTagLength']
-      if tag.length > max_len
-        errors.add(:tags, "can't be more than #{max_len} characters long each")
-      end
-    end
-  end
-
-  def no_spaces_in_tags
-    tags_cache.each do |tag|
-      if tag.include?(' ') || tag.include?('_')
-        errors.add(:tags, 'may not include spaces or underscores - use hyphens for multiple-word tags')
-      end
-    end
-  end
-
-  def stripped_minimum
-    if (body&.gsub(/(?:^[\s\t\u2000-\u200F]+|[\s\t\u2000-\u200F]+$)/, '')&.length || 0) < 30
-      errors.add(:body, 'must be more than 30 non-whitespace characters long')
-    end
-    if (title&.gsub(/(?:^[\s\t\u2000-\u200F]+|[\s\t\u2000-\u200F]+$)/, '')&.length || 0) < 15
-      errors.add(:title, 'must be more than 15 non-whitespace characters long')
-    end
-  end
-
-  def maximum_title_length
-    max_title_len = SiteSetting['MaxTitleLength']
-    if title.length > [(max_title_len || 255), 255].min
-      errors.add(:title, "can't be more than #{max_title_len} characters")
-    end
-  end
-
-  def tags_in_tag_set
-    tag_set = category.tag_set
-    unless tags.all? { |t| t.tag_set_id == tag_set.id }
-      errors.add(:base, "Not all of this question's tags are in the correct tag set.")
-    end
-  end
-
-  def required_tags?
-    required = category&.required_tag_ids
-    return unless required.present? && !required.empty?
-
-    unless tag_ids.any? { |t| required.include? t }
-      errors.add(:tags, "must contain at least one required tag (#{category.required_tags.pluck(:name).join(', ')})")
-    end
-  end
-
+  # Intended to be used as validation.
+  # Checks whether there are any moderator tags present added, and if so whether the current user is allowed to add
+  # those.
   def moderator_tags
     mod_tags = category&.moderator_tags&.map(&:name)
     return unless mod_tags.present? && !mod_tags.empty?
@@ -333,6 +361,8 @@ class Post < ApplicationRecord
     end
   end
 
+  # Intended to be used as callback after a save.
+  # Updates the category activity indicator if the last activity of this post changed.
   def update_category_activity
     if saved_changes.include? 'last_activity'
       category.update_activity(last_activity)
diff --git a/app/models/post_history.rb b/app/models/post_history.rb
index 2a742a99f2a4e8578f210a60d2da333e5bbbe1b0..990518da28a1db62a8d75fe7ba75f1f5bf461ddc 100644
--- a/app/models/post_history.rb
+++ b/app/models/post_history.rb
@@ -1,5 +1,7 @@
 class PostHistory < ApplicationRecord
   include PostRelated
+  include EditsValidations
+
   belongs_to :post_history_type
   belongs_to :user
   has_many :post_history_tags
@@ -13,13 +15,30 @@ class PostHistory < ApplicationRecord
     tags.where(post_history_tags: { relationship: 'after' })
   end
 
+  # @param user [User]
+  # @return [Boolean] whether the given user is allowed to see the details of this history item
+  def allowed_to_see_details?(user)
+    !hidden || user&.is_admin || user_id == user&.id || post.user_id == user&.id
+  end
+
+  # Hides all previous history
+  # @param post [Post]
+  # @param user [User]
+  def self.redact(post, user)
+    where(post: post).update_all(hidden: true)
+    history_hidden(post, user, after: post.body_markdown,
+                                    after_title: post.title,
+                                    after_tags: post.tags,
+                                    comment: 'Detailed history before this event is hidden because of a redaction.')
+  end
+
   def self.method_missing(name, *args, **opts)
     unless args.length >= 2
       raise NoMethodError
     end
 
     object, user = args
-    fields = [:before, :after, :comment, :before_title, :after_title, :before_tags, :after_tags]
+    fields = [:before, :after, :comment, :before_title, :after_title, :before_tags, :after_tags, :hidden]
     values = fields.to_h { |f| [f, nil] }.merge(opts)
 
     history_type_name = name.to_s
@@ -31,7 +50,7 @@ class PostHistory < ApplicationRecord
 
     params = { post_history_type: history_type, user: user, post: object, community_id: object.community_id }
     { before: :before_state, after: :after_state, comment: :comment, before_title: :before_title,
-      after_title: :after_title }.each do |arg, attr|
+      after_title: :after_title, hidden: :hidden }.each do |arg, attr|
       next if values[arg].nil?
 
       params = params.merge(attr => values[arg])
@@ -47,7 +66,10 @@ class PostHistory < ApplicationRecord
       end
     end.values.compact.flatten
 
-    history.post_history_tags = PostHistoryTag.create(post_history_tags)
+    # do not create post history tags if post history validations failed
+    unless history.errors.any?
+      history.post_history_tags = PostHistoryTag.create(post_history_tags)
+    end
 
     history
   end
diff --git a/app/models/post_type.rb b/app/models/post_type.rb
index 7f3e7570b3fc372bf027820ce6d903a7c0a1d1de..2cf37fdfda9d7b74af0f62bad21d99f6129ded13 100644
--- a/app/models/post_type.rb
+++ b/app/models/post_type.rb
@@ -19,6 +19,11 @@ class PostType < ApplicationRecord
     end
   end
 
+  # @return [Boolean] whether the post type is a system type
+  def system?
+    ['HelpDoc', 'PolicyDoc'].include?(name)
+  end
+
   def self.mapping
     Rails.cache.fetch 'network/post_types/post_type_ids', include_community: false do
       PostType.all.to_h { |pt| [pt.name, pt.id] }
diff --git a/app/models/site_setting.rb b/app/models/site_setting.rb
index ab5c9504b75538d42a1c38b675afbb82eb4b6791..160bbd05b1a224e4bcf46c6b370ff3619ca5cd50 100644
--- a/app/models/site_setting.rb
+++ b/app/models/site_setting.rb
@@ -11,14 +11,14 @@ class SiteSetting < ApplicationRecord
 
   def self.[](name)
     key = "SiteSettings/#{RequestContext.community_id}/#{name}"
-    cached = Rails.cache.fetch key do
+    cached = Rails.cache.fetch key, include_community: false do
       SiteSetting.applied_setting(name)&.typed
     end
 
     if cached.nil?
-      Rails.cache.delete key
+      Rails.cache.delete key, include_community: false
       value = SiteSetting.applied_setting(name)&.typed
-      Rails.cache.write key, value
+      Rails.cache.write key, value, include_community: false
       value
     else
       cached
@@ -26,10 +26,16 @@ class SiteSetting < ApplicationRecord
   end
 
   def self.exist?(name)
-    Rails.cache.exist?("SiteSettings/#{RequestContext.community_id}/#{name}") ||
+    Rails.cache.exist?("SiteSettings/#{RequestContext.community_id}/#{name}", include_community: false) ||
       SiteSetting.where(name: name).count.positive?
   end
 
+  # Checks whether the setting is a global site setting
+  # @return [Boolean]
+  def global?
+    community_id.nil?
+  end
+
   def typed
     SettingConverter.new(value).send("as_#{value_type.downcase}")
   end
@@ -45,14 +51,14 @@ class SiteSetting < ApplicationRecord
   def self.all_communities(name)
     communities = Community.all
     keys = (communities.map { |c| [c.id, "SiteSetting/#{c.id}/#{name}"] } + [[nil, "SiteSetting//#{name}"]]).to_h
-    cached = Rails.cache.read_multi(*keys.values)
+    cached = Rails.cache.read_multi(*keys.values, include_community: false)
     missing = keys.reject { |_k, v| cached.include?(v) }.map { |k, _v| k }
     settings = if missing.empty?
                  {}
                else
                  SiteSetting.where(name: name, community_id: missing).to_h { |s| [s.community_id, s] }
                end
-    Rails.cache.write_multi(missing.to_h { |cid| [keys[cid], settings[cid]&.typed] })
+    Rails.cache.write_multi(missing.to_h { |cid| [keys[cid], settings[cid]&.typed] }, include_community: false)
     communities.to_h do |c|
       [
         c.id,
diff --git a/app/models/sso_profile.rb b/app/models/sso_profile.rb
new file mode 100644
index 0000000000000000000000000000000000000000..0559a0f0f2b9fc30de0f6a03d325091ad8f9c16e
--- /dev/null
+++ b/app/models/sso_profile.rb
@@ -0,0 +1,5 @@
+class SsoProfile < ApplicationRecord
+  belongs_to :user, inverse_of: :sso_profile
+
+  validates :saml_identifier, uniqueness: true, presence: true
+end
diff --git a/app/models/suggested_edit.rb b/app/models/suggested_edit.rb
index 7730bbc61d8d643fce3667ac9095ad4cd5d72778..0afbe4d906bc85d12d4414288423f38187712c12 100644
--- a/app/models/suggested_edit.rb
+++ b/app/models/suggested_edit.rb
@@ -1,5 +1,7 @@
 class SuggestedEdit < ApplicationRecord
   include PostRelated
+  include PostValidations
+  include EditsValidations
 
   belongs_to :user
 
@@ -10,6 +12,15 @@ class SuggestedEdit < ApplicationRecord
   has_and_belongs_to_many :tags
   has_and_belongs_to_many :before_tags, class_name: 'Tag', join_table: 'suggested_edits_before_tags'
 
+  has_one :post_type, through: :post
+  has_one :category, through: :post
+
+  after_save :clear_pending_cache, if: proc { saved_change_to_attribute?(:active) }
+
+  def clear_pending_cache
+    Rails.cache.delete "pending_suggestions/#{post.category_id}"
+  end
+
   def pending?
     active
   end
diff --git a/app/models/tag.rb b/app/models/tag.rb
index 1b78a95fe534ac04448e7b86570e3df5995fdf04..873a0ea9525717e1cc1187106b16ab9f66a87dff 100644
--- a/app/models/tag.rb
+++ b/app/models/tag.rb
@@ -5,6 +5,8 @@ class Tag < ApplicationRecord
   has_many :children, class_name: 'Tag', foreign_key: :parent_id
   has_many :children_with_paths, class_name: 'TagWithPath', foreign_key: :parent_id
   has_many :post_history_tags
+  has_many :tag_synonyms, dependent: :destroy
+  accepts_nested_attributes_for :tag_synonyms, allow_destroy: true
   belongs_to :tag_set
   belongs_to :parent, class_name: 'Tag', optional: true
 
@@ -13,12 +15,57 @@ class Tag < ApplicationRecord
   validates :name, presence: true, format: { with: /[^ \t]+/, message: 'Tag names may not include spaces' }
   validate :parent_not_self
   validate :parent_not_own_child
+  validate :synonym_unique
   validates :name, uniqueness: { scope: [:tag_set_id], case_sensitive: false }
 
   def self.search(term)
-    where('name LIKE ?', "%#{sanitize_sql_like(term)}%")
-      .or(where('excerpt LIKE ?', "%#{sanitize_sql_like(term)}%"))
-      .order(Arel.sql(sanitize_sql_array(['name LIKE ? DESC, name', "#{sanitize_sql_like(term)}%"])))
+    stripped = term.strip
+    # Query to search on tags, the name is used for sorting.
+    q1 = where('tags.name LIKE ?', "%#{sanitize_sql_like(stripped)}%")
+         .or(where('tags.excerpt LIKE ?', "%#{sanitize_sql_like(stripped)}%"))
+         .select(Arel.sql('name AS sortname, tags.*'))
+
+    # Query to search on synonyms, the synonym name is used for sorting.
+    # The order clause here actually applies to the union of q1 and q2 (so not just q2).
+    q2 = joins(:tag_synonyms)
+         .where('tag_synonyms.name LIKE ?', "%#{sanitize_sql_like(stripped)}%")
+         .select(Arel.sql('tag_synonyms.name AS sortname, tags.*'))
+         .order(Arel.sql(sanitize_sql_array(['sortname LIKE ? DESC, sortname', "#{sanitize_sql_like(stripped)}%"])))
+
+    # Select from the union of the above queries, select only the tag columns such that we can distinct them
+    from(Arel.sql("(#{q1.to_sql} UNION #{q2.to_sql}) tags"))
+      .select(Tag.column_names.map { |c| "tags.#{c}" })
+      .distinct
+  end
+
+  ##
+  # Find or create a tag within a given tag set, considering synonyms. If a synonym is given as +name+ then the primary
+  # tag for it is returned instead.
+  # @param name [String] A tag name to find or create.
+  # @param tag_set [TagSet] The tag set within which to search for or create the tag.
+  # @return [Array(Tag, String)] The found or created tag, and the final name used. If a synonymized name was given as
+  #   +name+ then this will be the primary tag name.
+  #
+  # @example +name+ does not yet exist: a new Tag is created
+  #   Tag.find_or_create_synonymized name: 'new-tag', tag_set: ...
+  #   # => [Tag, 'new-tag']
+  #
+  # @example +name+ already exists: the existing Tag is returned
+  #   Tag.find_or_create_synonymized name: 'existing-tag', tag_set: ...
+  #   # => [Tag, 'existing-tag']
+  #
+  # @example +name+ is a synonym of 'other-tag': the Tag for 'other-tag' is returned
+  #   Tag.find_or_create_synonymized name: 'synonym', tag_set: ...
+  #   # => [Tag, 'other-tag']
+  def self.find_or_create_synonymized(name:, tag_set:)
+    existing = Tag.find_by(name: name, tag_set: tag_set)
+    if existing.nil?
+      synonyms = TagSynonym.joins(:tag).where(name: name, tags: { tag_set: tag_set })
+      synonymized_name = synonyms.exists? ? synonyms.first.tag.name : name
+      [Tag.find_or_create_by(name: synonymized_name, tag_set: tag_set), synonymized_name]
+    else
+      [existing, name]
+    end
   end
 
   def all_children
@@ -54,4 +101,10 @@ class Tag < ApplicationRecord
       errors.add(:base, "The #{parent.name} tag is already a child of this tag.")
     end
   end
+
+  def synonym_unique
+    if TagSynonym.joins(:tag).where(tags: { community_id: community_id }).exists?(name: name)
+      errors.add(:base, "A tag synonym with the name #{name} already exists.")
+    end
+  end
 end
diff --git a/app/models/tag_set.rb b/app/models/tag_set.rb
index e71e33c89923e0366f39940f3e3e9eee39d280e7..150dab46a8ccef93d8f87ba8f5c5dd11fa1cbd75 100644
--- a/app/models/tag_set.rb
+++ b/app/models/tag_set.rb
@@ -13,4 +13,12 @@ class TagSet < ApplicationRecord
   def self.main
     where(name: 'Main').first
   end
+
+  def with_paths(no_excerpt = false)
+    if no_excerpt
+      tags_with_paths.where(excerpt: ['', nil])
+    else
+      tags_with_paths
+    end
+  end
 end
diff --git a/app/models/tag_synonym.rb b/app/models/tag_synonym.rb
new file mode 100644
index 0000000000000000000000000000000000000000..da0aec259157325a836e0a429eda17e261a2b6d9
--- /dev/null
+++ b/app/models/tag_synonym.rb
@@ -0,0 +1,18 @@
+class TagSynonym < ApplicationRecord
+  belongs_to :tag
+  has_one :community, through: :tag
+
+  validates :name, presence: true, format: { with: /[^ \t]+/, message: 'Tag names may not include spaces' }
+  validate :name_unique
+
+  private
+
+  # Checks whether the name of this synonym is not already taken by a tag or synonym in the same community.
+  def name_unique
+    if TagSynonym.joins(:tag).where(tags: { community_id: tag.community_id }).exists?(name: name)
+      errors.add(:base, "A tag synonym with the name #{name} already exists.")
+    elsif Tag.unscoped.where(community_id: tag.community_id).exists?(name: name)
+      errors.add(:base, "A tag with the name #{name} already exists.")
+    end
+  end
+end
diff --git a/app/models/user.rb b/app/models/user.rb
index 84d79db011cd7b4b5fa01166e390f893027c31eb..04e1219f786f5e535612a64f7cd665250dd6631b 100644
--- a/app/models/user.rb
+++ b/app/models/user.rb
@@ -2,10 +2,11 @@
 # application code (i.e. excluding Devise) is concerned, has many questions, answers, and votes.
 class User < ApplicationRecord
   include ::UserMerge
+  include ::SamlInit
 
   devise :database_authenticatable, :registerable, :confirmable,
          :recoverable, :rememberable, :trackable, :validatable,
-         :lockable, :omniauthable
+         :lockable, :omniauthable, :saml_authenticatable
 
   has_many :posts, dependent: :nullify
   has_many :votes, dependent: :destroy
@@ -24,6 +25,10 @@ class User < ApplicationRecord
   has_many :comments, dependent: :nullify
   has_many :comment_threads_deleted, class_name: 'CommentThread', foreign_key: :deleted_by_id, dependent: :nullify
   has_many :comment_threads_locked, class_name: 'CommentThread', foreign_key: :locked_by_id, dependent: :nullify
+  has_many :category_filter_defaults, dependent: :destroy
+  has_many :filters, dependent: :destroy
+  has_many :user_websites, dependent: :destroy
+  accepts_nested_attributes_for :user_websites
   belongs_to :deleted_by, required: false, class_name: 'User'
 
   validates :username, presence: true, length: { minimum: 3, maximum: 50 }
@@ -40,7 +45,7 @@ class User < ApplicationRecord
   scope :active, -> { where(deleted: false) }
   scope :deleted, -> { where(deleted: true) }
 
-  after_create :send_welcome_tour_message
+  after_create :send_welcome_tour_message, :ensure_websites
 
   def self.list_includes
     includes(:posts, :avatar_attachment)
@@ -58,6 +63,12 @@ class User < ApplicationRecord
     community_user.trust_level
   end
 
+  # Checks whether this user is the same as a given user
+  # @param [User] user user to compare with
+  def same_as?(user)
+    id == user.id
+  end
+
   # This class makes heavy use of predicate names, and their use is prevalent throughout the codebase
   # because of the importance of these methods.
   # rubocop:disable Naming/PredicateName
@@ -69,6 +80,22 @@ class User < ApplicationRecord
     end
   end
 
+  # Checks if the user can push a given post type to network
+  # @param post_type [PostType] type of the post to be pushed
+  # @return [Boolean]
+  def can_push_to_network(post_type)
+    post_type.system? && (is_global_moderator || is_global_admin)
+  end
+
+  # Checks if the user can directly update a given post
+  # @param post [Post] updated post (owners can unilaterally update)
+  # @param post_type [PostType] type of the post (some are freely editable)
+  # @return [Boolean]
+  def can_update(post, post_type)
+    privilege?('edit_posts') || is_moderator || self == post.user || \
+      (post_type.is_freely_editable && privilege?('unrestricted'))
+  end
+
   def metric(key)
     Rails.cache.fetch("community_user/#{community_user.id}/metric/#{key}", expires_in: 24.hours) do
       case key
@@ -111,6 +138,18 @@ class User < ApplicationRecord
     website.nil? ? website : URI.parse(website).hostname
   end
 
+  def valid_websites_for
+    user_websites.where.not(url: [nil, '']).order(position: :asc)
+  end
+
+  def ensure_websites
+    pos = user_websites.size
+    while pos < UserWebsite::MAX_ROWS
+      pos += 1
+      UserWebsite.create(user_id: id, position: pos)
+    end
+  end
+
   def is_moderator
     is_global_moderator || community_user&.is_moderator || is_admin || community_user&.privilege?('mod') || false
   end
@@ -119,6 +158,42 @@ class User < ApplicationRecord
     is_global_admin || community_user&.is_admin || false
   end
 
+  # Used by network profile: does this user have a profile on that other comm?
+  def has_profile_on(community_id)
+    cu = community_users.where(community_id: community_id).first
+    !cu&.user_id.nil? || false
+  end
+
+  def reputation_on(community_id)
+    cu = community_users.where(community_id: community_id).first
+    cu&.reputation || 1
+  end
+
+  def post_count_on(community_id)
+    cu = community_users.where(community_id: community_id).first
+    cu&.post_count || 0
+  end
+
+  def is_moderator_on(community_id)
+    cu = community_users.where(community_id: community_id).first
+    # is_moderator is a DB check, not a call to is_moderator()
+    is_global_moderator || is_admin || cu&.is_moderator || cu&.privilege?('mod') || false
+  end
+
+  def has_ability_on(community_id, ability_internal_id)
+    cu = community_users.where(community_id: community_id).first
+    if cu&.is_moderator || cu&.is_admin || is_global_moderator || is_global_admin || cu&.privilege?('mod')
+      true
+    elsif cu.nil?
+      false
+    else
+      Ability.unscoped do
+        UserAbility.joins(:ability).where(community_user_id: cu&.id, is_suspended: false,
+                                          ability: { internal_id: ability_internal_id }).exists?
+      end
+    end
+  end
+
   def rtl_safe_username
     "#{username}\u202D"
   end
@@ -177,7 +252,7 @@ class User < ApplicationRecord
 
   def email_not_bad_pattern
     return unless File.exist?(Rails.root.join('../.qpixel-email-patterns.txt'))
-    return unless saved_changes.include? 'email'
+    return unless changes.include? 'email'
 
     patterns = File.read(Rails.root.join('../.qpixel-email-patterns.txt')).split("\n")
     matched = patterns.select { |p| email.match? Regexp.new(p) }
@@ -214,7 +289,7 @@ class User < ApplicationRecord
                         'how this site works.', '/tour')
   end
 
-  def block(reason)
+  def block(reason, length: 180.days)
     user_email = email
     user_ip = [last_sign_in_ip]
 
@@ -222,10 +297,10 @@ class User < ApplicationRecord
       user_ip << current_sign_in_ip
     end
 
-    BlockedItem.create(item_type: 'email', value: user_email, expires: DateTime.now + 180.days,
+    BlockedItem.create(item_type: 'email', value: user_email, expires: length.from_now,
                        automatic: true, reason: "#{reason}: #" + id.to_s)
     user_ip.compact.uniq.each do |ip|
-      BlockedItem.create(item_type: 'ip', value: ip, expires: 180.days.from_now,
+      BlockedItem.create(item_type: 'ip', value: ip, expires: length.from_now,
                          automatic: true, reason: "#{reason}: #" + id.to_s)
     end
   end
@@ -234,13 +309,19 @@ class User < ApplicationRecord
     global_key = "prefs.#{id}"
     community_key = "prefs.#{id}.community.#{RequestContext.community_id}"
     {
-      global: AppConfig.preferences.reject { |_, v| v['community'] }.transform_values { |v| v['default'] }
+      global: AppConfig.preferences.select { |_, v| v['global'] }.transform_values { |v| v['default'] }
                        .merge(RequestContext.redis.hgetall(global_key)),
       community: AppConfig.preferences.select { |_, v| v['community'] }.transform_values { |v| v['default'] }
                           .merge(RequestContext.redis.hgetall(community_key))
     }
   end
 
+  def category_preference(category_id)
+    category_key = "prefs.#{id}.category.#{RequestContext.community_id}.category.#{category_id}"
+    AppConfig.preferences.select { |_, v| v['category'] }.transform_values { |v| v['default'] }
+             .merge(RequestContext.redis.hgetall(category_key))
+  end
+
   def validate_prefs!
     global_key = "prefs.#{id}"
     community_key = "prefs.#{id}.community.#{RequestContext.community_id}"
@@ -260,5 +341,24 @@ class User < ApplicationRecord
   def preference(name, community: false)
     preferences[community ? :community : :global][name]
   end
+
+  def has_active_flags?(post)
+    !post.flags.where(user: self, status: nil).empty?
+  end
+
+  def active_flags(post)
+    post.flags.where(user: self, status: nil)
+  end
+
+  def do_soft_delete(attribute_to)
+    AuditLog.moderator_audit(event_type: 'user_delete', related: self, user: attribute_to,
+                             comment: attributes_print(join: "\n"))
+    assign_attributes(deleted: true, deleted_by_id: attribute_to.id, deleted_at: DateTime.now,
+                      username: "user#{id}", email: "#{id}@deleted.localhost",
+                      password: SecureRandom.hex(32))
+    skip_reconfirmation!
+    save
+  end
+
   # rubocop:enable Naming/PredicateName
 end
diff --git a/app/models/user_website.rb b/app/models/user_website.rb
new file mode 100644
index 0000000000000000000000000000000000000000..6b50d909f50dce78f3c31653597d2c277411baeb
--- /dev/null
+++ b/app/models/user_website.rb
@@ -0,0 +1,6 @@
+class UserWebsite < ApplicationRecord
+  belongs_to :user
+  default_scope { order(:position) }
+
+  MAX_ROWS = 3
+end
diff --git a/app/models/vote.rb b/app/models/vote.rb
index a7a3505ab4474cafe34ec05de066661c2ee0bb1c..a7c42dcb3dea0128eb63503938d171b0a291bd01 100644
--- a/app/models/vote.rb
+++ b/app/models/vote.rb
@@ -34,37 +34,45 @@ class Vote < ApplicationRecord
   end
 
   def rep_change(direction)
+    return unless post.present?
+
     change = CategoryPostType.rep_changes[[post.category_id, post.post_type_id]][vote_type] || 0
     recv_user.update!(reputation: recv_user.reputation + (direction * change))
   end
 
   def post_not_deleted
-    if post.deleted?
+    if post&.deleted?
       errors.add(:base, 'Votes are locked on deleted posts')
     end
   end
 
   def check_valid
-    throw :abort unless valid?
+    throw :abort unless valid? || post.blank?
   end
 
   def add_counter
+    return unless post.present?
+
     case vote_type
     when 1
       post.update(upvote_count: post.upvote_count + 1)
     when -1
       post.update(downvote_count: post.downvote_count + 1)
     end
+
     post.recalc_score
   end
 
   def remove_counter
+    return unless post.present?
+
     case vote_type
     when 1
       post.update(upvote_count: [post.upvote_count - 1, 0].max)
     when -1
       post.update(downvote_count: [post.downvote_count - 1, 0].max)
     end
+
     post.recalc_score
   end
 end
diff --git a/app/tasks/maintenance/initialize_user_websites_task.rb b/app/tasks/maintenance/initialize_user_websites_task.rb
new file mode 100644
index 0000000000000000000000000000000000000000..80a91b9483f0a45db59489138f6752dbad332f26
--- /dev/null
+++ b/app/tasks/maintenance/initialize_user_websites_task.rb
@@ -0,0 +1,33 @@
+# frozen_string_literal: true
+
+module Maintenance
+  class InitializeUserWebsitesTask < MaintenanceTasks::Task
+    def collection
+      User.all
+    end
+
+    def process(user)
+      unless user.user_websites.exists?(position: 1)
+        if user.website.present?
+          UserWebsite.create!(user_id: user.id, position: 1, label: 'website', url: user.website)
+        else
+          UserWebsite.create!(user_id: user.id, position: 1)
+        end
+      end
+
+      unless user.user_websites.exists?(position: 2)
+        if user.twitter.present?
+          UserWebsite.create!(user_id: user.id, position: 2, label: 'Twitter',
+                              url: "https://twitter.com/#{user.twitter}")
+        else
+          UserWebsite.create!(user_id: user.id, position: 2)
+        end
+      end
+
+      # This check *should* be superfluous, but just in case...
+      unless user.user_websites.exists?(position: 3)
+        UserWebsite.create!(user_id: user.id, position: 3)
+      end
+    end
+  end
+end
diff --git a/app/views/abilities/show.html.erb b/app/views/abilities/show.html.erb
index c06d2131e89cc79782752fb7d4ca4913747a7995..a34e29ca211e8b25abc4fac31b85889ab10d5118 100644
--- a/app/views/abilities/show.html.erb
+++ b/app/views/abilities/show.html.erb
@@ -49,20 +49,23 @@
     <% unless @ability.manual? %>
       <% unless params[:thresholds].nil? %>
         <div class="widget--body">
-          <p>You need to reach these thresholds to earn the ability:</p>
-
+          <% if @user.id == current_user&.id %>
+            <p>You need to reach these thresholds to earn the ability:</p>
+	  <% else %>
+            <p><%= @user.username %> needs to reach these thresholds to earn the ability:</p>
+	  <% end %>
           <% unless @ability.post_score_threshold.nil? %>
             <% post_score_percent = (linearize_progress(@user.community_user.post_score) / linearize_progress(@ability.post_score_threshold) * 100).to_i %>
 
             <p><strong>Post score threshold</strong></p>
             <% if post_score_percent < 100 %>
-              <p>You need to have more well-received posts.</p>
+              <p>Need to have more well-received posts.</p>
               <div class="meter is-progress">
                 <div class="meter--bar is-<%= post_score_percent %>%"></div>
                 <div class="meter--label"><%= post_score_percent %>%</div>
               </div>
             <% else %>
-              <p>You need to have many well-received posts.</p>
+              <p>Need to have many well-received posts.</p>
               <div class="meter is-progress">
                 <div class="meter--bar is-100%"></div>
                 <div class="meter--label">100%</div>
@@ -75,13 +78,13 @@
 
             <p><strong>Edit score threshold</strong></p>
             <% if edit_score_percent < 100 %>
-              <p>You need to have more accepted suggested edits.</p>
+              <p>Need to have more accepted suggested edits.</p>
               <div class="meter is-progress">
                 <div class="meter--bar is-<%= edit_score_percent %>%"></div>
                 <div class="meter--label"><%= edit_score_percent %>%</div>
               </div>
             <% else %>
-              <p>You need to have many accepted suggested edits.</p>
+              <p>Need to have many accepted suggested edits.</p>
               <div class="meter is-progress">
                 <div class="meter--bar is-100%"></div>
                 <div class="meter--label">100%</div>
@@ -94,13 +97,13 @@
 
             <p><strong>Flag score threshold</strong></p>
             <% if flag_score_percent < 100 %>
-              <p>You need to have more helpful flags.</p>
+              <p>Need to have more helpful flags.</p>
               <div class="meter is-progress">
                 <div class="meter--bar is-<%= flag_score_percent %>%"></div>
                 <div class="meter--label"><%= flag_score_percent %>%</div>
               </div>
             <% else %>
-              <p>You need to have many helpful flags.</p>
+              <p>Need to have many helpful flags.</p>
               <div class="meter is-progress">
                 <div class="meter--bar is-100%"></div>
                 <div class="meter--label">100%</div>
diff --git a/app/views/admin/all_email.html.erb b/app/views/admin/all_email.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..ade37ea360fde04cf36635ebb9e681f32ee1c48a
--- /dev/null
+++ b/app/views/admin/all_email.html.erb
@@ -0,0 +1,24 @@
+<%= render 'posts/markdown_script' %>
+
+<div class="notice is-danger">
+  <p>
+    <i class="fas fa-exclamation-triangle"></i> Please be careful, as this tool sends a lot of emails.
+  </p>
+</div>
+
+<h1><%= t 'admin.tools.email_all' %></h1>
+<p><%= t 'admin.email_all_blurb' %></p>
+
+<%= form_with url: send_all_email_path do |f| %>
+  <div class="form-group">
+    <%= f.label :subject, t('g.subject').capitalize, class: 'form-element' %>
+    <%= f.text_field :subject,  class: 'form-element', required: true %>
+  </div>
+
+  <%= render 'shared/body_field', f: f, field_name: :body_markdown, field_label: t('g.body').capitalize, post: nil %>
+
+  <div class="post-preview"></div>
+
+  <%= f.submit t('g.send').capitalize, class: 'button is-filled',
+               onclick: "return confirm('Are you sure you want to send this email to all users?')" %>
+<% end %>
diff --git a/app/views/admin/audit_log.html.erb b/app/views/admin/audit_log.html.erb
index cb6003e1b8e8cc19d8e3a1587dd24947442032db..ea2b467af21ea95148d42fec5bd04225d5e1c5e8 100644
--- a/app/views/admin/audit_log.html.erb
+++ b/app/views/admin/audit_log.html.erb
@@ -4,15 +4,15 @@
   <h3><%= pluralize(@logs.count, t('g.log')) %></h3>
   <div class="button-list is-gutterless">
     <% classes = 'button is-outlined is-muted' %>
-    <%= link_to t('g.age'), query_url(sort: 'age'),
+    <%= link_to t('g.age'), request.params.merge(sort: 'age'),
                 class: "#{classes} #{params[:sort] == 'age' || params[:sort].nil? ? 'is-active' : ''}" %>
-    <%= link_to t('g.type'), query_url(sort: 'type'),
+    <%= link_to t('g.type'), request.params.merge(sort: 'type'),
                 class: "#{classes} #{params[:sort] == 'type' ? 'is-active' : ''}" %>
-    <%= link_to t('g.event'), query_url(sort: 'event'),
+    <%= link_to t('g.event'), request.params.merge(sort: 'event'),
                 class: "#{classes} #{params[:sort] == 'event' ? 'is-active' : ''}" %>
-    <%= link_to t('g.related'), query_url(sort: 'related'),
+    <%= link_to t('g.related'), request.params.merge(sort: 'related'),
                 class: "#{classes} #{params[:sort] == 'related' ? 'is-active' : ''}" %>
-    <%= link_to t('g.user'), query_url(sort: 'user'),
+    <%= link_to t('g.user'), request.params.merge(sort: 'user'),
                 class: "#{classes} #{params[:sort] == 'user' ? 'is-active' : ''}" %>
   </div>
 </div>
diff --git a/app/views/admin/change_back.html.erb b/app/views/admin/change_back.html.erb
index 26e9e77c047fcb4d2f63810ab4cf9d40e903e2e6..33cf3c7ea89ca61a2a247844b0bc85ba02891b7c 100644
--- a/app/views/admin/change_back.html.erb
+++ b/app/views/admin/change_back.html.erb
@@ -1,14 +1,20 @@
 <h1>Stop impersonating</h1>
 <p>
-  You (<%= @impersonator.username %>) are currently impersonating <%= current_user.username %>. To stop
-  impersonating them, verify your password below
+  You (<%= @impersonator.username %>) are currently impersonating <%= current_user.username %>.
+  <% if @impersonator.sso_profile.present? %>
+    You can stop impersonating them with the button below, after which you will have to sign in again.
+  <% else %>
+    To stop impersonating them, verify your password below.
+  <% end %>
 </p>
 
 <%= form_tag stop_impersonating_path, class: 'form-horizontal' do %>
-  <div class="form-group">
-    <%= label_tag :password, 'Password', class: 'form-element' %>
-    <%= password_field_tag :password, '', class: 'form-element' %>
-  </div>
+  <% unless @impersonator.sso_profile.present? %>
+    <div class="form-group">
+      <%= label_tag :password, 'Password', class: 'form-element' %>
+      <%= password_field_tag :password, '', class: 'form-element' %>
+    </div>
+  <% end %>
 
   <div class="actions">
     <%= submit_tag 'Verify & Stop Impersonating', class: 'button is-primary is-filled' %>
diff --git a/app/views/admin/index.html.erb b/app/views/admin/index.html.erb
index 89f299e13a479f06417ae798746467c71ec5c5fc..2f4c5266dcef59f0b3290f2402247c5b3fbebd2b 100644
--- a/app/views/admin/index.html.erb
+++ b/app/views/admin/index.html.erb
@@ -30,17 +30,28 @@
     <div class="grid--cell is-4-lg is-6-md is-12-sm" data-ckb-list-item data-ckb-item-type="link">
       <div class="widget">
         <div class="widget--body">
-          <i class="fas fa-globe has-color-red-700"></i> <i class="fas fa-envelope"></i>
-          <%= link_to t('admin.tools.email_moderators'), moderator_email_path, 'data-ckb-item-link' => '' %>
+          <i class="fas fa-globe has-color-red-700"></i> <i class="fas fa-mail-bulk"></i>
+          <%= link_to t('admin.tools.post_types'), post_types_path, 'data-ckb-item-link' => '' %>
         </div>
       </div>
     </div>
 
+    <% if current_user.developer? %>
+      <div class="grid--cell is-4-lg is-6-md is-12-sm" data-ckb-list-item data-ckb-item-type="link">
+        <div class="widget">
+          <div class="widget--body">
+            <i class="fas fa-globe has-color-red-700"></i> <i class="fas fa-folder-plus"></i>
+            <%= link_to 'Email All Users', email_all_path, 'data-ckb-item-link' => '' %>
+          </div>
+        </div>
+      </div>
+    <% end %>
+
     <div class="grid--cell is-4-lg is-6-md is-12-sm" data-ckb-list-item data-ckb-item-type="link">
       <div class="widget">
         <div class="widget--body">
-          <i class="fas fa-globe has-color-red-700"></i> <i class="fas fa-mail-bulk"></i>
-          <%= link_to t('admin.tools.post_types'), post_types_path, 'data-ckb-item-link' => '' %>
+          <i class="fas fa-globe has-color-red-700"></i> <i class="fas fa-envelope"></i>
+          <%= link_to t('admin.tools.email_moderators'), moderator_email_path, 'data-ckb-item-link' => '' %>
         </div>
       </div>
     </div>
@@ -108,6 +119,7 @@
       </div>
     </div>
   </div>
+
   <div class="grid--cell is-4-lg is-6-md is-12-sm" data-ckb-list-item data-ckb-item-type="link">
     <div class="widget">
       <div class="widget--body">
@@ -116,6 +128,7 @@
       </div>
     </div>
   </div>
+
   <% if current_user.is_global_admin %>
     <div class="grid--cell is-4-lg is-6-md is-12-sm" data-ckb-list-item data-ckb-item-type="link">
       <div class="widget">
diff --git a/app/views/admin_mailer/to_all_users.html.erb b/app/views/admin_mailer/to_all_users.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..790db26894d6f6ad18862d9ab03dc0a09ceaae9f
--- /dev/null
+++ b/app/views/admin_mailer/to_all_users.html.erb
@@ -0,0 +1,2 @@
+<%= raw(sanitize(ApplicationController.helpers.render_markdown(@body_markdown),
+                 scrubber: ApplicationController.helpers.scrubber)) %>
diff --git a/app/views/admin_mailer/to_all_users.text.erb b/app/views/admin_mailer/to_all_users.text.erb
new file mode 100644
index 0000000000000000000000000000000000000000..636df4adf321037939e0b66a42c0ceb83fa92d8a
--- /dev/null
+++ b/app/views/admin_mailer/to_all_users.text.erb
@@ -0,0 +1 @@
+<%= ApplicationController.helpers.strip_markdown @body_markdown %>
diff --git a/app/views/application/dashboard.html.erb b/app/views/application/dashboard.html.erb
index f818dfaba71c5d2854b70a89687dd316a50da735..397e020fcc092571cd5fe638b14c7137e7ca7058 100644
--- a/app/views/application/dashboard.html.erb
+++ b/app/views/application/dashboard.html.erb
@@ -1,6 +1,7 @@
 <%= content_for :title, 'Dashboard' %>
 
 <h1>Dashboard</h1>
+<p>All communities, with their categories. You can also see the communities from the dropdown arrow at the top right of each page.</p>
 <div class="grid community-list">
   <% @communities.each do |c| %>
     <% categories = Category.unscoped.where(community: c).order(sequence: :asc, id: :asc) %>
@@ -19,7 +20,6 @@
             <% end %>
           </div>
         </div>
-        <div class="widget--body h-fw-bold h-bg-tertiary-050"><i class="fa fa-file-alt"></i> Posts</div>
         <% categories.each do |cat| %>
           <% next if (cat.min_view_trust_level || -1) > (current_user&.trust_level || 0) %>
           <div class="widget--body">
@@ -29,6 +29,15 @@
                 <span class="badge is-status" title="unread activity in category"></span>
               <% end %>
             <% end %>
+
+            <% if current_user&.has_ability_on(c.id, 'edit_posts') %>
+              <% sug_edits = @edits[cat.id] || 0 %>
+              <% if sug_edits > 0 %>
+                <%= link_to suggested_edits_queue_url(cat, host: c.host), class: 'widget--body-extra' do %>
+                (<%= sug_edits %> pending <%= "edit".pluralize(sug_edits) %>)
+                <% end %>
+              <% end %>
+            <% end %>
           </div>
         <% end %>
         <% if current_user&.is_global_moderator || current_user&.is_global_admin %>
diff --git a/app/views/categories/_form.html.erb b/app/views/categories/_form.html.erb
index 7b172c7ebc7b46932efc668c09be8415860b8f3a..55e04b314cb8c52a2c51d0be6505eb25c09c2d23 100644
--- a/app/views/categories/_form.html.erb
+++ b/app/views/categories/_form.html.erb
@@ -1,4 +1,14 @@
 <%= form_for @category, url: submit_path, method: :post do |f| %>
+  <% if @category.errors.any? %>
+    <div class="notice is-danger">
+      <ul>
+        <% @category.errors.full_messages.each do |m| %>
+          <li><%= m %></li>
+        <% end %>
+      </ul>
+    </div>
+  <% end %>
+
   <details open>
     <summary>
       Basic Information
@@ -51,6 +61,18 @@
       <%= f.select :tag_set_id, options_for_select(TagSet.all.map { |ts|  [ts.name, ts.id] }, selected: @category.tag_set_id),
                    { include_blank: true }, class: 'form-element js-category-tag-set-select' %>
     </div>
+
+    <div class="form-group">
+      <%= f.label :min_title_length, 'Minimum title length', class: 'form-element' %>
+      <span class="form-caption">Titles of posts in this category must be at least this many characters long.</span>
+      <%= f.number_field :min_title_length, class: 'form-element' %>
+    </div>
+
+    <div class="form-group">
+      <%= f.label :min_body_length, 'Minimum body length', class: 'form-element' %>
+      <span class="form-caption">Posts in this category must be at least this many characters long.</span>
+      <%= f.number_field :min_body_length, class: 'form-element' %>
+    </div>
   </details>
 
   <details>
@@ -123,6 +145,13 @@
     </span>
       <%= f.number_field :sequence, class: 'form-element' %>
     </div>
+
+    <div class="form-group">
+      <%= f.label :default_filter_id, class: 'form-element' %>
+      <span class="form-caption">The default filter for this category, used for anonymous users.</span>
+      <% system_filters = User.find(-1).filters.to_h { |filter| [filter.name, filter.id] } %>
+      <%= f.select :default_filter_id, options_for_select(system_filters, selected: @category.default_filter_id), { include_blank: "No default" } %>
+    </div>
   </details>
 
   <details>
@@ -192,7 +221,7 @@
       <% disabled = @category.tag_set.nil? %>
       <%= f.select :required_tag_ids, options_for_select(@category.required_tags.map { |t| [t.name, t.id] },
                                                          selected: @category.required_tag_ids),
-                   { include_blank: true }, multiple: true, class: 'form-element js-tag-select js-required-tags',
+                   {}, multiple: true, class: 'form-element js-tag-select js-required-tags',
                    data: { tag_set: @category.tag_set&.id, create: 'false', use_ids: 'true' }, disabled: disabled %>
     </div>
 
@@ -210,7 +239,7 @@
 
       <%= f.select :topic_tag_ids, options_for_select(@category.topic_tags.map { |t| [t.name, t.id] },
                                                       selected: @category.topic_tag_ids),
-                   { include_blank: true }, multiple: true, class: 'form-element js-tag-select js-topic-tags',
+                   {}, multiple: true, class: 'form-element js-tag-select js-topic-tags',
                    data: { tag_set: @category.tag_set&.id, create: 'false', use_ids: 'true' }, disabled: disabled %>
     </div>
 
@@ -227,7 +256,7 @@
 
       <%= f.select :moderator_tag_ids, options_for_select(@category.moderator_tags.map { |t| [t.name, t.id] },
                                                           selected: @category.moderator_tag_ids),
-                   { include_blank: true }, multiple: true, class: 'form-element js-tag-select js-moderator-tags',
+                   {}, multiple: true, class: 'form-element js-tag-select js-moderator-tags',
                    data: { tag_set: @category.tag_set&.id, create: 'false', use_ids: 'true' }, disabled: disabled %>
     </div>
   </details>
diff --git a/app/views/categories/show.html.erb b/app/views/categories/show.html.erb
index c5bfcce11cc560fe104cec1289f402d164693b0c..62a18a3162b77caba85de4a0880970ed6884ea11 100644
--- a/app/views/categories/show.html.erb
+++ b/app/views/categories/show.html.erb
@@ -11,45 +11,62 @@
 <% end %>
 
 <% post_count = @posts.count %>
-<div class="has-color-tertiary-500 category-meta" title="<%= post_count %>">
-  <div>
-    <%= short_number_to_human post_count, precision: 1, significant: false %>
-    <%= 'post'.pluralize(post_count) %> &middot;
-
-    <% if current_user&.is_admin %>
-      <%= link_to 'Edit Category', edit_category_path(@category) %>
-      &middot;
-    <% end %>
-
-    <%= link_to 'Subscribe',
-                new_subscription_path(type: 'category', qualifier: @category.id, return_to: request.path),
-                class: 'button is-outlined' %>
-  </div>
+<div class="has-color-tertiary-500 category-meta">
+  <span class=category-meta--start>
+    <span title="<%= post_count %> posts">
+      <%= short_number_to_human post_count, precision: 1, significant: false %>
+      <%= 'post'.pluralize(post_count) %>
+    </span>
+    <%= render 'shared/rss_link', url: category_feed_path(@category, format: 'rss') %>
+    <span class="has-margin-4">
+      <% if current_user&.is_admin %>
+        <%= link_to 'Edit Category', edit_category_path(@category) %>
+      <% end %>
+    </span>
+  </span>
 
   <div class="button-list is-gutterless has-margin-2">
-    <%= link_to 'Activity', query_url(sort: 'activity'),
+    <%= link_to 'Activity', request.params.merge(sort: 'activity'),
                 class: "button is-muted is-outlined #{(params[:sort].nil?) && !current_page?(questions_lottery_path) ||
                     params[:sort] == 'activity' ? 'is-active' : ''}",
                 title: 'most recent changes: new posts, edits, close/open, delete/undelete' %>
-    <%= link_to 'Age', query_url(sort: 'age'),
+    <%= link_to 'Age', request.params.merge(sort: 'age'),
                 class: "button is-muted is-outlined #{params[:sort] == 'age' ? 'is-active' : ''}",
                 title: 'newest posts (ignores other activity)' %>
-    <%= link_to 'Score', query_url(sort: 'score'),
+    <%= link_to 'Score', request.params.merge(sort: 'score'),
                 class: "button is-muted is-outlined #{params[:sort] == 'score' ? 'is-active' : ''}",
                 title: 'highest score first (not the same as net votes)' %>
     <% if SiteSetting['AllowContentTransfer'] %>
-      <%= link_to 'Native', query_url(sort: 'native'),
+      <%= link_to 'Native', request.params.merge(sort: 'native'),
                   class: "button is-muted is-outlined #{params[:sort] == 'native' ? 'is-active' : ''}",
                   title: 'exclude imported posts' %>
     <% end %>
-    <% if @category.name == 'Q&A' %>
-      <%= link_to 'Random', query_url(sort: 'lottery'),
-                  class: "button is-muted is-outlined #{params[:sort] == 'lottery' ? 'is-active' : ''}",
-                  title: 'random set of questions, usually older ones' %>
-    <% end %>
+    <%= link_to 'Random', request.params.merge(sort: 'lottery'),
+        class: "button is-muted is-outlined #{params[:sort] == 'lottery' ? 'is-active' : ''}",
+        title: 'random set of questions, usually older ones' %>
   </div>
 </div>
 
+<details>
+  <summary>Filters (<%= @filtered ? @active_filter[:name].empty? ? 'Custom' : @active_filter[:name] : 'None' %>)</summary>
+  <% if @active_filter[:default] == :user %>
+    <div class="notice is-info">
+      You are currently filtering by <strong><%= @active_filter[:name] %></strong> because it is set as your default for this category
+    </div>
+  <% elsif @active_filter[:default] == :category and not user_signed_in? %>
+    <div class="notice is-info">
+      You are currently filtering by <strong><%= @active_filter[:name] %></strong> because it is the default for this category
+    </div>
+  <% elsif @active_filter[:default] == :category and user_signed_in? %>
+    <div class="notice is-info">
+      You are currently filtering by <strong><%= @active_filter[:name] %></strong> because it is the default for this category and you do not have a personal default set
+    </div>
+  <% end %>
+  <%= form_tag request.original_url, method: :get do %>
+    <%= render 'search/filters' %>
+  <% end %>
+</details>
+
 <div class="item-list">
   <% @posts.each do |post| %>
     <%= render 'posts/type_agnostic', post: post %>
@@ -61,7 +78,5 @@
 </div>
 
 <div class="has-padding-top-4">
-  <%= link_to category_feed_path(@category, format: 'rss') do %>
-    <i class="fas fa-rss"></i> Category RSS feed
-  <% end %>
-</div>
\ No newline at end of file
+  <%= render 'shared/rss_link', url: category_feed_path(@category, format: 'rss'), text: 'Category RSS feed' %>
+</div>
diff --git a/app/views/close_reasons/_form.html.erb b/app/views/close_reasons/_form.html.erb
index e691772a650ec8225028dea02a19d44e86541567..03522077dc1d246ec94b77c6f298594526e89ff9 100644
--- a/app/views/close_reasons/_form.html.erb
+++ b/app/views/close_reasons/_form.html.erb
@@ -42,5 +42,5 @@
   </div>
 
   <%= f.submit 'Save', class: 'button is-filled' %>
-  <%= link_to 'Cancel', close_reasons_path(global: @close_reason.community.nil? ? 1 : 0), class: 'button' %>
+  <%= link_to 'Cancel', close_reasons_path(global: @close_reason.community.nil? ? 1 : 0), class: 'button', role: 'button' %>
 <% end %>
diff --git a/app/views/close_reasons/index.html.erb b/app/views/close_reasons/index.html.erb
index 324b7ecbc7d91af904b2f6175980f62e1a4c42f6..1297a027970954504c60a4235f20be17c94d06ae 100644
--- a/app/views/close_reasons/index.html.erb
+++ b/app/views/close_reasons/index.html.erb
@@ -33,7 +33,7 @@
       <td><%= reason.requires_other_post ? "yes" : "no" %></td>
       <td><%= reason.active ? "yes" : "no" %></td>
       <td>
-        <%= link_to "edit", close_reason_path(id: reason.id), class: "button is-outlined" %>
+        <%= link_to "edit", close_reason_path(id: reason.id), class: "button is-outlined", 'aria-label': "Edit close reason #{reason.name}" %>
       </td>
     </tr>
   <% end %>
@@ -46,5 +46,5 @@
     Do not add reasons where you're not sure that your community needs or wants them.
   </p>
   <p>It's not possible to remove reasons, once created. They can only be deactivated.</p>
-  <%= link_to "add reason", new_close_reason_path(global: params[:global]), class: "button is-outlined" %>
+  <%= link_to "add reason", new_close_reason_path(global: params[:global]), class: "button is-outlined", 'aria-label': 'Add new close reason' %>
 </div>
diff --git a/app/views/comments/_comment.html.erb b/app/views/comments/_comment.html.erb
index 7dc7caad9e7ca6f856b5368c6a2f7351375fda11..4d8e4f2c2b1aad65b379fea1e5b458e8e2e1bf44 100644
--- a/app/views/comments/_comment.html.erb
+++ b/app/views/comments/_comment.html.erb
@@ -45,27 +45,27 @@
       <% end %>
     </div>
     <div class="comment--links">
-      <%= link_to comment_link(comment), class: 'js-comment-permalink' do %>
+      <%= link_to comment_link(comment), class: 'js-comment-permalink', role: 'button' do %>
         <span class="js-text">copy link</span>
       <% end %>
       <% if with_post_link %>
         <%= link_to 'post', generic_share_link(comment.post) %>
       <% end %>
       <% if user_signed_in? && (comment.user == current_user || current_user.is_moderator) && params[:inline] != 'true' %>
-        <a href="#" class="js-comment-edit">edit</a>
+        <a href="#" class="js-comment-edit" role="button" aria-label="Edit comment">edit</a>
         <% if comment.deleted %>
-          <a href="#" class="is-red js-comment-undelete">undelete</a>
+          <a href="#" class="is-red js-comment-undelete" role="button" aria-label="Undelete comment">undelete</a>
         <% else %>
-          <a href="#" class="is-red js-comment-delete">delete</a>
+          <a href="#" class="is-red js-comment-delete" role="button" aria-label="Delete comment">delete</a>
         <% end %>
       <% end %>
       <% if user_signed_in? && params[:inline] != 'true' %>
-        <a href="javascript:void(0)" data-drop="#flag-comment-<%= comment.id %>">flag</a>
+        <a href="javascript:void(0)" data-drop="#flag-comment-<%= comment.id %>" role="button" aria-label="Flag comment">flag</a>
         <div class="droppanel is-large h-c-black" id="flag-comment-<%= comment.id %>">
           <label for="flag-post-<%= comment.id %>">Flag reason</label>
           <input type="text" id="flag-post-<%= comment.id %>" class="form-element" />
           <a href="javascript:void(0)" class="flag-link js-comment-flag button is-danger" data-post-id="<%= comment.id %>"
-             data-requires-details="true">Flag</a>
+             data-requires-details="true" role="button" aria-label="Flag comment">Flag</a>
         </div>
       <% end %>
     </div>
diff --git a/app/views/comments/_new_thread_modal.html.erb b/app/views/comments/_new_thread_modal.html.erb
index 5ba122f28671e033001b1d6854f98f2bfc558359..8ea24575f81b375e68688951f91bb26c61183916 100644
--- a/app/views/comments/_new_thread_modal.html.erb
+++ b/app/views/comments/_new_thread_modal.html.erb
@@ -20,11 +20,7 @@
       <div class="form-caption">Start the thread with a comment.</div>
       <%= text_area_tag :body, '', class: 'form-element js-comment-field', required: true,
                         data: { post: post.id, thread: '-1', character_count: ".js-character-count-#{post.id}" } %>
-      <span class="has-float-right has-font-size-caption js-character-count-<%= post.id %>"
-            data-max="1000" data-min="15">
-        <i class="fas fa-ellipsis-h js-character-count__icon"></i>
-        <span class="js-character-count__count">0 / 1000</span>
-      </span>
+      <%= render 'shared/char_count', type: post.id, min: 15, max: 1000 %>
 
       <%= label_tag :title, 'Comment thread title (optional)', class: 'form-element' %>
       <span class="form-caption">
@@ -32,12 +28,7 @@
         be shown.
       </span>
       <%= text_field_tag :title, '', class: 'form-element', data: { character_count: ".js-character-count-thread-title" } %>
-
-      <span class="has-float-right has-font-size-caption js-character-count-thread-title hide"
-            data-max="255" data-min="0" data-display-at="0.75">
-        <i class="fas fa-ellipsis-h js-character-count__icon"></i>
-        <span class="js-character-count__count">0 / 255</span>
-      </span>
+      <%= render 'shared/char_count', type: 'thread-title' %>
 
       <%= submit_tag 'Create thread', class: 'button is-filled', id: "create_thread_button_#{post.id}", disabled: true %>
     <% end %>
diff --git a/app/views/comments/_post.html.erb b/app/views/comments/_post.html.erb
index 4949f50123b0157af420da191e0fd5263ba10dc0..4d7c00fcfcf4718ac3c554adb38153096c237c3c 100644
--- a/app/views/comments/_post.html.erb
+++ b/app/views/comments/_post.html.erb
@@ -3,14 +3,14 @@
 %>
 
 <% comment_threads.each do |thread| %>
-  <div class="post--comments-thread-wrapper">
+  <div class="post--comments-thread-wrapper" role="listitem">
     <div class="post--comments-thread is-inline <%= thread.deleted ? 'is-deleted' : '' %> <%= thread.archived ? 'is-archived' : '' %>">
       <% if thread.deleted %>
         <i class="fas fa-trash h-c-red-600 fa-fw" title="Deleted thread" aria-label="Deleted thread"></i>
       <% elsif thread.archived %>
         <i class="fas fa-archive fa-fw" title="Archived thread" aria-label="Archived thread"></i>
       <% end %>
-      <%= link_to thread.title, comment_thread_path(thread), class: 'js--comment-link', data: { thread: thread.id } %>
+      <%= link_to thread.title, comment_thread_path(thread), class: 'js--comment-link', data: { thread: thread.id }, role: 'button' %>
       (<%= pluralize(thread.reply_count, 'comment') %>)
     </div>
   </div>
diff --git a/app/views/comments/thread.html.erb b/app/views/comments/thread.html.erb
index dda35581cfc144bbaa1e2526b7d2cbb91f7cb693..98aa32ebdc4ebd28d05fab944f08e93b5cc36db2 100644
--- a/app/views/comments/thread.html.erb
+++ b/app/views/comments/thread.html.erb
@@ -24,9 +24,9 @@
 <!-- THREAD STARTS BELOW -->
 <div class="<%= @comment_thread.deleted ? 'h-bg-red-050' : '' %> <%= params[:inline] == 'true' ? 'post--comments-thread is-embedded' : '' %>">
 
-<div class="widget <%= @comment_thread.deleted ? 'is-red' : '' %>" data-deleted="<%= @comment_thread.deleted %>"
+<div class="widget thread <%= @comment_thread.deleted ? 'is-red' : '' %>" data-deleted="<%= @comment_thread.deleted %>"
      data-archived="<%= @comment_thread.archived %>" data-thread="<%= @comment_thread.id %>"
-     data-comments="<%= @comment_thread.reply_count %>">
+     data-comments="<%= @comment_thread.reply_count %>" data-post="<%= @post.id %>">
   <div class="widget--header">
     <% if params[:inline] == 'true' %>
       <a href="<%= comment_thread_path(@comment_thread.id) %>" class="widget--header-link">
@@ -37,18 +37,23 @@
       </a>
     <% else %>
       <% if current_user&.privilege? 'flag_curate' %>
-        <a href="#" class="widget--header-link" data-drop=".js--tools-thread-<%= @comment_thread.id %>"><i class="fa fa-cog fa-fw"></i>tools</a>
+        <a href="#" class="widget--header-link" data-drop=".js--tools-thread-<%= @comment_thread.id %>"
+          role="button" aria-label="Show more thread tools">
+          <i class="fa fa-cog fa-fw"></i>tools
+        </a>
       <% end %>
       <% unless current_user.nil? %>
         <% if @comment_thread.followed_by? current_user %>
           <a href="#" class="widget--header-link js--unrestrict-thread" data-action="follow"
-            data-thread="<%= @comment_thread.id %>" title="You are following this thread and will be notified of every response. You can unfollow at any time.">
-            <i class="fa fa-minus fa-fw"></i>unfollow
+            data-thread="<%= @comment_thread.id %>" title="You are following this thread and will be notified of every response. You can unfollow at any time."
+            role="button" aria-label="Unfollow this thread">
+            <i class="fas fa-fw fa-bell-slash"></i> unfollow
           </a>
         <% else %>
           <a href="#" class="widget--header-link js--restrict-thread" data-action="follow"
-            data-thread="<%= @comment_thread.id %>" title="Follow this thread to be notified of every response.">
-            <i class="fa fa-plus fa-fw"></i>follow
+            data-thread="<%= @comment_thread.id %>" title="Follow this thread to be notified of every response."
+            role="button" aria-label="Follow this thread">
+            <i class="fas fa-fw fa-bell"></i> follow
           </a>
         <% end %>
       <% end %>
@@ -71,26 +76,28 @@
       <% count <= 5 %>
     <% end %>
   <% end %>
-  <% comments.each do |comment| %>
-    <% if comment.deleted && !(current_user&.is_moderator && params[:show_deleted_comments] == "1") %>
-      <% skipped_deleted += 1%>
-      <% next %>
-    <% elsif skipped_deleted > 0 %>
-      <div class="widget--body">
+  <div role="list">
+    <% comments.each do |comment| %>
+      <% if comment.deleted && !(current_user&.is_moderator && params[:show_deleted_comments] == "1") %>
+        <% skipped_deleted += 1%>
+        <% next %>
+      <% elsif skipped_deleted > 0 %>
+        <div class="widget--body" role="listitem">
+          <%= render 'comments/skip_deleted', skipped_deleted: skipped_deleted%>
+        </div>
+        <% skipped_deleted = 0 %>
+      <% end %>
+      <div class="widget--body" role="listitem">
+        <%= render 'comments/comment', comment: comment, pingable: pingable %>
+      </div>
+    <% end %>
+    <% if skipped_deleted > 0 %>
+      <div class="widget--body" role="listitem">
         <%= render 'comments/skip_deleted', skipped_deleted: skipped_deleted%>
       </div>
       <% skipped_deleted = 0 %>
     <% end %>
-    <div class="widget--body">
-      <%= render 'comments/comment', comment: comment, pingable: pingable %>
-    </div>
-  <% end %>
-  <% if skipped_deleted > 0 %>
-    <div class="widget--body">
-      <%= render 'comments/skip_deleted', skipped_deleted: skipped_deleted%>
-    </div>
-    <% skipped_deleted = 0 %>
-  <% end %>
+  </div>
   <% unless current_user.nil? || params[:inline] == 'true' %>
     <div class="widget--footer">
       <% if !@comment_thread.read_only? %>
@@ -114,12 +121,8 @@
             <%= text_area_tag :content, '', class: 'form-element js-comment-field',
                               data: { thread: @comment_thread.id, post: @comment_thread.post_id,
                                       character_count: ".js-character-count-#{@post.id}" } %>
-            <span class="has-float-right has-font-size-caption js-character-count-<%= @post.id %>"
-                  data-max="1000" data-min="15">
-              <i class="fas fa-ellipsis-h js-character-count__icon"></i>
-              <span class="js-character-count__count">0 / 1000</span>
-            </span>
-            
+            <%= render 'shared/char_count', type: @post.id, min: 15, max: 1000 %>
+
             <%= submit_tag 'Add reply', class: 'button is-muted is-filled', disabled:true %>
           <% end %>
         <% end %>
@@ -149,36 +152,50 @@
     <div class="droppanel--header">thread options</div>
     <div class="droppanel--menu">
       <% if current_user.is_moderator || !@comment_thread.read_only? %>
-        <a href="#" data-modal=".js--rename-thread-<%= @comment_thread.id %>" data-drop=".js--tools-thread-<%= @comment_thread.id %>"><i class="fas fa-pen fa-fw"></i> rename</a>
+        <a href="#" data-modal=".js--rename-thread-<%= @comment_thread.id %>" role="button" aria-label="Rename thread">
+          <i class="fas fa-pen fa-fw"></i> rename
+        </a>
       <% end %>
 
       <% unless @comment_thread.archived || @comment_thread.deleted %>
         <% if @comment_thread.locked? %>
-          <a href="#" class="js--unrestrict-thread" data-action="lock" data-drop=".js--tools-thread-<%= @comment_thread.id %>" data-thread="<%= @comment_thread.id %>"><i class="fas fa-lock fa-fw"></i> unlock</a>
+          <a href="#" class="js--unrestrict-thread" data-action="lock" data-thread="<%= @comment_thread.id %>" role="button" aria-label="Unlock thread">
+            <i class="fas fa-lock fa-fw"></i> unlock
+          </a>
         <% else %>
-          <a href="#" data-modal=".js--lock-thread-<%= @comment_thread.id %>" data-drop=".js--tools-thread-<%= @comment_thread.id %>"><i class="fas fa-lock fa-fw"></i> lock</a>
+          <a href="#" data-modal=".js--lock-thread-<%= @comment_thread.id %>" role="button" aria-label="Lock thread">
+            <i class="fas fa-lock fa-fw"></i> lock
+          </a>
         <% end %>
       <% end %>
       
       <% unless @comment_thread.locked? || @comment_thread.deleted %>
         <% if @comment_thread.archived %>
-          <a href="#" class="js--unrestrict-thread" data-action="archive" data-drop=".js--tools-thread-<%= @comment_thread.id %>" data-thread="<%= @comment_thread.id %>"><i class="fas fa-archive fa-fw"></i> restore</a>
+          <a href="#" class="js--unrestrict-thread" data-action="archive" data-thread="<%= @comment_thread.id %>" role="button" aria-label="Restore thread">
+            <i class="fas fa-archive fa-fw"></i> restore
+          </a>
         <% else %>
-          <a href="#" class="js--restrict-thread" data-action="archive" data-drop=".js--tools-thread-<%= @comment_thread.id %>" data-thread="<%= @comment_thread.id %>"><i class="fas fa-archive fa-fw"></i> archive</a>
+          <a href="#" class="js--restrict-thread" data-action="archive" data-thread="<%= @comment_thread.id %>" role="button" aria-label="Archive thread">
+            <i class="fas fa-archive fa-fw"></i> archive
+          </a>
         <% end %>
       <% end %>
 
       <% unless @comment_thread.locked? || @comment_thread.archived %>
         <% if @comment_thread.deleted %>
-          <a href="#" class="js--unrestrict-thread" data-action="delete" data-drop=".js--tools-thread-<%= @comment_thread.id %>" data-thread="<%= @comment_thread.id %>"><i class="fas fa-trash fa-fw"></i> undelete</a>
+          <a href="#" class="js--unrestrict-thread" data-action="delete" data-thread="<%= @comment_thread.id %>" role="button" aria-label="Undelete thread">
+            <i class="fas fa-trash fa-fw"></i> undelete
+          </a>
         <% else %>
-          <a href="#" class="js--restrict-thread" data-action="delete" data-drop=".js--tools-thread-<%= @comment_thread.id %>" data-thread="<%= @comment_thread.id %>"><i class="fas fa-trash fa-fw"></i> delete</a>
+          <a href="#" class="js--restrict-thread" data-action="delete" data-thread="<%= @comment_thread.id %>" role="button" aria-label="Delete thread">
+            <i class="fas fa-trash fa-fw"></i> delete
+          </a>
         <% end %>
       <% end %>
 
       <% if current_user&.is_moderator || current_user&.is_admin %>
-        <a href="#" class="js--show-followers" data-modal="#js-followers-<%= @comment_thread.id %>"
-           data-drop=".js--tools-thread-<%= @comment_thread.id %>" data-thread="<%= @comment_thread.id %>">
+        <a href="#" class="js--show-followers" data-modal="#js-followers-<%= @comment_thread.id %>" data-thread="<%= @comment_thread.id %>"
+           role="button" aria-label="Show thread followers">
           <i class="fas fa-users"></i> followers
         </a>
       <% end %>
diff --git a/app/views/devise/registrations/edit.html.erb b/app/views/devise/registrations/edit.html.erb
index b1d1216f22bfca5661e432e18e49936bd6f18ad9..644611675a422c29e8c1524f4910791bb9d9661e 100644
--- a/app/views/devise/registrations/edit.html.erb
+++ b/app/views/devise/registrations/edit.html.erb
@@ -6,12 +6,27 @@
   Two-factor authentication &raquo;
 <% end %>
 
+<% sso = current_user.sso_profile.present? %>
+<% if sso %>
+  <% if devise_sign_in_enabled? && SiteSetting['AllowSsoDisconnect'] %>
+    <%= link_to user_disconnect_sso_path, class: 'button is-outlined is-danger' do %>
+      Disconnect Single Sign-On &raquo;
+    <% end %>
+  <% end %>
+
+  <div class="notice is-warning">
+    You sign in through a Single Sign-On provider.
+    Because of that, you cannot change your email address or password here.
+    Please contact your system administrator if you would like to change these.
+  </div>
+<% end %>
+
 <%= form_for(resource, as: resource_name, url: registration_path(resource_name), html: { method: :put }) do |f| %>
   <%= render "devise/shared/error_messages", resource: resource %>
 
   <div class="form-group">
     <%= f.label :email, class: "form-element" %>
-    <%= f.email_field :email, class: 'form-element', autofocus: true, autocomplete: "email" %>
+    <%= f.email_field :email, class: 'form-element', autofocus: true, autocomplete: "email", disabled: sso %>
   </div>
 
   <% if devise_mapping.confirmable? && resource.pending_reconfirmation? %>
@@ -19,26 +34,27 @@
   <% end %>
 
   <div class="form-group">
-    <%= f.label :password, class: "form-element" %>
+    <%= f.label :current_password, class: "form-element" %>
+    <div class="form-caption">We need your current password to confirm your changes.</div>
+    <%= f.password_field :current_password, class: 'form-element', autocomplete: "current-password", required: true,
+                         disabled: sso %>
+  </div>
+
+  <div class="form-group">
+    <%= f.label :password, "New password", class: "form-element" %>
     <div class="form-caption">
       Leave blank if you don't want to change it.
       <% if @minimum_password_length %>
         <em>(<%= @minimum_password_length %> characters minimum)</em>
       <% end %>
     </div>
-    <%= f.password_field :password, class: 'form-element', autocomplete: "new-password" %>
+    <%= f.password_field :password, class: 'form-element', autocomplete: "new-password", disabled: sso %>
   </div>
 
   <div class="form-group">
-    <%= f.label :password_confirmation, class: "form-element" %>
-    <%= f.password_field :password_confirmation, class: 'form-element', autocomplete: "new-password" %>
-  </div>
-
-  <div class="form-group">
-    <%= f.label :current_password, class: "form-element" %>
-    <div class="form-caption">We need your current password to confirm your changes.</div>
-    <%= f.password_field :current_password, class: 'form-element', autocomplete: "current-password", required: true %>
+    <%= f.label :password_confirmation, "Confirm new password", class: "form-element" %>
+    <%= f.password_field :password_confirmation, class: 'form-element', autocomplete: "new-password", disabled: sso %>
   </div>
 
-  <%= f.submit "Update", class: 'button is-filled is-very-large' %>
+  <%= f.submit "Update", class: 'button is-filled is-very-large', disabled: sso %>
 <% end %><br/>
diff --git a/app/views/devise/registrations/new.html.erb b/app/views/devise/registrations/new.html.erb
index 1679be35b260b43e0c21ee7e9ac0fd5f19d5d905..8fc1a62a7946e6a7065098fdca8862cbc8d199a4 100644
--- a/app/views/devise/registrations/new.html.erb
+++ b/app/views/devise/registrations/new.html.erb
@@ -1,5 +1,12 @@
 <h1>Sign up</h1>
 
+<% if sso_sign_in_enabled? %>
+  <div class="notice is-info has-margin-bottom-4">
+    <i class="fas fa-info-circle"></i> You can also sign in with your Single Sign-On provider.
+    <%= link_to "SSO Sign in", new_saml_user_session_path(resource_name), class: 'button is-primary is-filled' %>
+  </div>
+<% end %>
+
 <div class="notice is-info">
   <i class="fas fa-info-circle"></i> If you have an account on another <a href="https://codidact.com/">Codidact site</a>,
   don't create a new account here - <%= link_to 'sign in', new_user_session_path %> with your existing details instead.
diff --git a/app/views/devise/sessions/new.html.erb b/app/views/devise/sessions/new.html.erb
index a15f722062c8f2e9735ff88bb57b9dedb38868e4..b6a4569664d54cf3976969f718e21e8716645dde 100644
--- a/app/views/devise/sessions/new.html.erb
+++ b/app/views/devise/sessions/new.html.erb
@@ -1,8 +1,15 @@
 <h1>Sign in</h1>
 
+<% if sso_sign_in_enabled? %>
+  <div class="notice is-info has-margin-bottom-4">
+    <i class="fas fa-info-circle"></i> You can also sign in with your Single Sign-On provider.
+    <%= link_to "SSO Sign in", new_saml_user_session_path(resource_name), class: 'button is-primary is-filled' %>
+  </div>
+<% end %>
+
 <div class="notice is-info has-margin-bottom-4">
-  <i class="fas fa-info-circle"></i> Your sign-in information is the same on all
-  <a href="https://codidact.com/">Codidact communities</a>.
+  <p><i class="fas fa-info-circle"></i> Your sign-in information is the same on all communities on this network.</p>
+  <p>By signing in, you agree to abide by our <a href="/help/tos">Terms of Service</a>, <a href="/help/code-of-conduct">Code of Conduct</a>, <a href="/help/spam">Guidelines for promotional content</a>, and any other policies listed in the <a href="/help">Help Center</a>.</p>
 </div>
 
 <%= form_for(resource, as: resource_name, url: session_path(resource_name), method: :post) do |f| %>
@@ -18,7 +25,7 @@
 
   <% if devise_mapping.rememberable? %>
     <div class="form-group">
-      <%= f.label :remember_me, class: "form-element" %>
+      <%= f.label :remember_me, 'Stay signed in on this device', class: "form-element" %>
       <%= f.check_box :remember_me, class: "form-radio-element" %>
     </div>
   <% end %>
diff --git a/app/views/devise/shared/_links.html.erb b/app/views/devise/shared/_links.html.erb
index 18fdcd6ab2d31659fa02c1dff05351dd735a6af3..15b79838a6d9e2d1385c445f4128809f29d652d5 100644
--- a/app/views/devise/shared/_links.html.erb
+++ b/app/views/devise/shared/_links.html.erb
@@ -1,20 +1,24 @@
-<%- if controller_name != 'sessions' %>
+<%- if controller_name != 'sessions' && devise_sign_in_enabled? %>
   <%= link_to "Sign in", new_session_path(resource_name) %><br />
 <% end %>
 
-<%- if devise_mapping.registerable? && controller_name != 'registrations' %>
+<%- if controller_name != 'saml_sessions' && sso_sign_in_enabled? %>
+  <%= link_to "SSO Sign in", new_saml_user_session_path(resource_name) %><br />
+<% end %>
+
+<%- if devise_mapping.registerable? && controller_name != 'registrations' && devise_sign_in_enabled? %>
   <%= link_to "Sign up", new_registration_path(resource_name) %><br />
 <% end %>
 
-<%- if devise_mapping.recoverable? && controller_name != 'passwords' && controller_name != 'registrations' %>
+<%- if devise_mapping.recoverable? && controller_name != 'passwords' && controller_name != 'registrations' && devise_sign_in_enabled? %>
   <%= link_to "Forgot your password?", new_password_path(resource_name) %><br />
 <% end %>
 
-<%- if devise_mapping.confirmable? && controller_name != 'confirmations' %>
+<%- if devise_mapping.confirmable? && controller_name != 'confirmations' && devise_sign_in_enabled? %>
   <%= link_to "Didn't receive confirmation instructions?", new_confirmation_path(resource_name) %><br />
 <% end %>
 
-<%- if devise_mapping.lockable? && resource_class.unlock_strategy_enabled?(:email) && controller_name != 'unlocks' %>
+<%- if devise_mapping.lockable? && resource_class.unlock_strategy_enabled?(:email) && controller_name != 'unlocks' && devise_sign_in_enabled? %>
   <%= link_to "Didn't receive unlock instructions?", new_unlock_path(resource_name) %><br />
 <% end %>
 
diff --git a/app/views/donation_mailer/donation_uncaptured.html.erb b/app/views/donation_mailer/donation_uncaptured.html.erb
index a37cfaaaffdc5b56af63730df9aa6788ba473139..4e7a89d57eb893edf65a939adb0a1902f4f7b5e4 100644
--- a/app/views/donation_mailer/donation_uncaptured.html.erb
+++ b/app/views/donation_mailer/donation_uncaptured.html.erb
@@ -12,7 +12,7 @@
 <p>
   We want to make sure that everything's okay. If you just changed your mind, that's absolutely fine! We're not here
   to push. If you ran into problems, however, please let us know if we can help &mdash; you can reply to this email to
-  reach us at <a href="mailto:support@codidact.org">support@codidact.org</a>, or you can post on our
+  reach us at <a href="mailto:<%= message.reply_to %>"><%= message.reply_to %></a>, or you can post on our
   <a href="https://meta.codidact.com/">Meta</a> site, and someone will lend a hand.
 </p>
 
diff --git a/app/views/donation_mailer/donation_uncaptured.text.erb b/app/views/donation_mailer/donation_uncaptured.text.erb
index 463355f7849b3914556146baad26a58f82d70f43..0b29da4f887ee10853729addd9d8f26322c18f73 100644
--- a/app/views/donation_mailer/donation_uncaptured.text.erb
+++ b/app/views/donation_mailer/donation_uncaptured.text.erb
@@ -10,7 +10,7 @@ You started the process to make a donation of <%= number_to_currency(@amount, un
 
 We want to make sure that everything's okay. If you just changed your mind, that's absolutely fine! We're not here
 to push. If you ran into problems, however, please let us know if we can help — you can reply to this email to
-reach us at support@codidact.org, or you can post on our Meta site (https://meta.codidact.com/), and someone will
+reach us at <%= message.reply_to %>, or you can post on our Meta site (https://meta.codidact.com/), and someone will
 lend a hand.
 
 Whatever happened, thanks for thinking of us!
diff --git a/app/views/errors/restricted_content.html.erb b/app/views/errors/restricted_content.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..1b9c5f2eab81424c50f6b2d848184a99114e1894
--- /dev/null
+++ b/app/views/errors/restricted_content.html.erb
@@ -0,0 +1,20 @@
+<%= content_for :title, "Welcome to #{SiteSetting['SiteName']}" %>
+
+<%- if request.fullpath != '/' %>
+  <div class="notice is-danger">
+    <div class="container">
+      <p>You need to sign in before you can access this site.</p>
+    </div>
+  </div>
+<% end %>
+
+<%= raw(sanitize(render_markdown(SiteSetting['RestrictedAccessFrontPageText']), scrubber: scrubber)) %>
+
+<%- if sso_sign_in_enabled? %>
+  <%= link_to "SSO Sign in", new_saml_user_session_path, class: 'button is-extremely-large is-filled' %><br />
+<% end %>
+
+<%- if devise_sign_in_enabled? %>
+  <%= link_to "Sign in", new_user_session_path, class: 'button is-extremely-large is-muted is-outlined' %><br />
+  <%= link_to "Sign up", new_user_registration_path, class: 'button is-extremely-large is-muted is-filled' %><br />
+<% end %>
\ No newline at end of file
diff --git a/app/views/flags/_flag.html.erb b/app/views/flags/_flag.html.erb
index 3ce6aea42a26d635866d7de8b46eea31ed3af9db..d3fb85779c8456224a5c1baa47a2023ee3685e81 100644
--- a/app/views/flags/_flag.html.erb
+++ b/app/views/flags/_flag.html.erb
@@ -21,7 +21,12 @@
   <div class="widget--body">
     <p>
       <strong><%= flag.post_flag_type&.name || 'Flag reason' %></strong>: <%= flag.reason %> &mdash;
-      <%= user_link flag.user %>
+      <%= user_link flag.user %> at <%= flag.created_at.iso8601 %>
+      <% if flag.post_type == 'Post' && flag.post.updated_at > flag.created_at %>
+      <%= link_to post_history_path(flag.post) do %>
+	(post modified after flag)
+      <% end %>
+      <% end %>
     </p>
   </div>
   <% if escalation %>
@@ -30,6 +35,9 @@
         <strong>Escalation reason</strong>: <%= flag.escalation_comment %> &mdash;
         <%= user_link flag.escalated_by %> <%= flag.escalated_at.iso8601 %>
       </p>
+      <p>
+	<strong>Attention</strong>: The reply to the flag will be shown to the flagger, not the mod escalating this flag. Do not share sensitive, mod-only information.
+      </p>
     </div>
   <% end %>
   <% if controls %>
@@ -50,4 +58,4 @@
       </div>
     </div>
   <% end %>
-</div>
\ No newline at end of file
+</div>
diff --git a/app/views/flags/_handled.html.erb b/app/views/flags/_handled.html.erb
index 8922132267986e92c5a912cfcad7a984f4026b41..e9e65c75ceeaa0637d4bd9444aa5d30eafdee8a4 100644
--- a/app/views/flags/_handled.html.erb
+++ b/app/views/flags/_handled.html.erb
@@ -9,7 +9,7 @@
   <div class="widget--body">
     <p>
       <strong><%= flag.post_flag_type&.name || 'Flag reason' %>: </strong><%= flag.reason %> &mdash;
-      <%= user_link flag.user %>
+      <%= user_link flag.user %>  at <%= flag.created_at.iso8601 %>
       (<%= link_to 'history', flag_history_path(flag.user) %>)
     </p>
     <p>
@@ -20,12 +20,13 @@
       <% if flag.message.present? %>
         <strong>response:</strong> <%= flag.message %>
       <% end %>
-      <% if flag.handled_by_id.present? %>
+      <% if flag.handled_by_id.present? && current_user.is_moderator %>
         &mdash;
         <%= link_to user_path(flag.handled_by) do %>
           <span dir="ltr"><%= rtl_safe_username(flag.handled_by) %></span>
-        <% end %>
+          <% end %>
+	  handled at <%= flag.handled_at.iso8601 %>
       <% end %>
     </p>
   </div>
-</div>
\ No newline at end of file
+</div>
diff --git a/app/views/flags/handled.html.erb b/app/views/flags/handled.html.erb
index 320ac30ddaa1d7523ceb917e8cb36950719fcdcb..9e0c3d0ed793df56dde3bb828220a8a72e368483 100644
--- a/app/views/flags/handled.html.erb
+++ b/app/views/flags/handled.html.erb
@@ -13,4 +13,6 @@
 
 <% @flags.each do |flag| %>
   <%= render 'handled', flag: flag %>
-<% end %>
\ No newline at end of file
+<% end %>
+
+<%= will_paginate @flags, renderer: BootstrapPagination::Rails %>
diff --git a/app/views/flags/queue.html.erb b/app/views/flags/queue.html.erb
index 9939b4d7db9361855af278150cc5b99f7aca730a..3cee195918a574216a26255ac9ad31154bfcbbdd 100644
--- a/app/views/flags/queue.html.erb
+++ b/app/views/flags/queue.html.erb
@@ -6,6 +6,7 @@
 <h1>Moderator Flag Queue</h1>
 <p>Below is a list of posts that users have flagged. Users are asked to provide a reason when flagging posts for
   moderator attention; use that to help you determine what needs to be done.</p>
+<p>You can mark a flag helpful even if you take no action. If a post was edited after the flag was raised, for example, the problem might already be fixed.</p>
 
 <div class="button-list is-gutterless">
   <%= link_to 'Active', flag_queue_path,
diff --git a/app/views/layouts/_footer.html.erb b/app/views/layouts/_footer.html.erb
index b00c2c03f59917f8c5e2b720153b422c4e2435a3..909be8b490ae25f937927e1d19329386fdb3ef7c 100644
--- a/app/views/layouts/_footer.html.erb
+++ b/app/views/layouts/_footer.html.erb
@@ -2,28 +2,23 @@
   <div class="footer--container container">
     <div class="grid">
       <div class="grid--cell is-6 is-12-md is-12-sm">
-        <h3>The Codidact Project</h3>
         <ul>
-          <li><%= link_to 'Terms of Service', '/policy/tos' %></li>
-          <li><%= link_to 'Privacy Policy', '/policy/privacy-policy' %></li>
-          <li><%= link_to 'Code of Conduct', '/policy/code-of-conduct' %></li>
-          <li><%= link_to 'About us', 'https://codidact.org/' %></li>
-          <li><%= link_to 'Contact us', 'https://codidact.com/contact' %></li>
-          <li><%= link_to 'Other communities', 'https://codidact.com/' %></li>
+          <li><%= link_to 'Our Communities', '/dashboard' %></li>
+          <li><%= link_to 'About Us', '/policy/network-faq' %></li>
         </ul>
       </div>
       <div class="grid--cell is-6 is-12-md is-12-sm">
-        <h3>Other Codidact Communities</h3>
         <ul>
-          <% Rails.cache.persistent('codidact_sites').each do |site| %>
-            <li><%= link_to site['name'], site['canonical_url'] %></li>
-          <% end %>
+          <li><%= link_to 'Terms of Service', '/policy/tos' %></li>
+          <li><%= link_to 'Privacy Policy', '/policy/privacy-policy' %></li>
+          <li><%= link_to 'Code of Conduct', '/policy/code-of-conduct' %></li>
         </ul>
       </div>
     </div>
-    <% commit = Rails.cache.persistent('current_commit') %>
-    <p>
-      Version <%= link_to commit[0][0..7], "https://github.com/codidact/qpixel/commit/#{commit[0]}" %> (<%= commit[1] %>)
+    <% sha, date = current_commit %>
+    <p>Powered by <%= link_to 'Codidact', 'https://codidact.org/' %>. <% if sha&.present? %>Version <%= 
+      link_to sha[0..7], "https://github.com/codidact/qpixel/commit/#{sha}" 
+    %> (<%= date.to_time.utc.strftime('%F %TZ') %>)<% end %>
     </p>
   </div>
 </footer>
diff --git a/app/views/layouts/_head.html.erb b/app/views/layouts/_head.html.erb
index c9bf20abdb269627aadabdb8a082736c8d9053f6..8a88d427ac3425bd93c55f6122763d5cf2510678 100644
--- a/app/views/layouts/_head.html.erb
+++ b/app/views/layouts/_head.html.erb
@@ -19,7 +19,7 @@
 <link rel="preconnect" href="https://cdnjs.cloudflare.com" />
 
 <%= stylesheet_link_tag "https://cdn.jsdelivr.net/npm/@fortawesome/fontawesome-free@5.11.2/css/all.min.css" %>
-<%= stylesheet_link_tag "https://cdn.jsdelivr.net/npm/@codidact/co-design@0.12.5/dist/codidact.css" %>
+<%= stylesheet_link_tag "https://cdn.jsdelivr.net/npm/@codidact/co-design@latest/dist/codidact.css" %>
 <%= stylesheet_link_tag "https://cdn.jsdelivr.net/npm/select2@4.0.12/dist/css/select2.min.css" %>
 <%= stylesheet_link_tag "/assets/community/#{@community.host.split('.')[0]}.css" %>
 <%= stylesheet_link_tag 'application', media: 'all' %>
@@ -27,13 +27,13 @@
 <%= javascript_include_tag "https://cdn.jsdelivr.net/npm/jquery@2.2.2/dist/jquery.min.js" %>
 <%= javascript_include_tag "https://cdn.jsdelivr.net/npm/moment@2.13.0/min/moment.min.js" %>
 <%= javascript_include_tag "https://cdn.jsdelivr.net/npm/select2@4.0.12/dist/js/select2.min.js" %>
-<% if SiteSetting['DonationsEnabled'] %>
+<% if SiteSetting['DonationsEnabled'] && (SiteSetting['LoadStripeEverywhere'] || controller_name == 'donations') %>
   <%= javascript_include_tag "https://js.stripe.com/v3/" %>
 <% end %>
 <%= javascript_include_tag "https://cdn.jsdelivr.net/npm/dompurify@2.2.9/dist/purify.min.js" %>
 <%= javascript_include_tag "/assets/community/#{@community.host.split('.')[0]}.js" %>
 <%= javascript_include_tag 'application' %>
-<script src="https://cdn.jsdelivr.net/npm/@codidact/co-design@0.12.5/js/co-design.js" defer></script>
+<script src="https://cdn.jsdelivr.net/npm/@codidact/co-design@latest/js/co-design.js" defer></script>
 
 <% if SiteSetting['SyntaxHighlightingEnabled'] %>
   <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.0.1/styles/default.min.css">
@@ -49,7 +49,7 @@
       }
     };
   </script>
-  <script src="https://polyfill.io/v3/polyfill.min.js?features=es6"></script>
+  <script src="https://cdnjs.cloudflare.com/polyfill/v3/polyfill.min.js?version=3.27.4&features=es7%2Ces6%2Ces5%2Ces2018%2Ces2017%2Ces2016%2Ces2015%2Cdefault"></script>
   <script id="MathJax-script" async src="https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-mml-chtml.js"></script>
 <% end %>
 
diff --git a/app/views/layouts/_header.html.erb b/app/views/layouts/_header.html.erb
index 0de78a20deae4ac1e5c3f840a012af3b2a49d596..bf3f6c612d9f553d95b3aa37e14900365f0effc9 100644
--- a/app/views/layouts/_header.html.erb
+++ b/app/views/layouts/_header.html.erb
@@ -2,6 +2,23 @@
 <% mobile_logo_path = SiteSetting['MobileLogoPath'] %>
 <% sticky_header = user_preference('sticky_header', community: false) == 'true' %>
 
+<script>
+const toggleSearchFocus = () => {
+  const $input = $('#search');
+
+  if ($input.is(':focus')) {
+    $input.blur();
+  }
+  else {
+      $input.focus();
+  }
+};
+
+$(() => {
+  $('a[data-header-slide="#search-slide"]').on('click', toggleSearchFocus);
+});
+</script>
+
 <header class="header is-small has-margin-0<%=' is-dark' if SiteSetting['SiteHeaderIsDark'] %><%= ' sticky' if sticky_header %>">
   <div class="container header--container">
     <div class="header--brand">
@@ -22,10 +39,12 @@
     </div>
     <nav class="header--menu">
       <% unless @community.is_fake %>
-        <%= link_to users_path, class: 'header--item' do %>
+        <%= link_to users_path, class: 'header--item', title: 'Open users list' do %>
           <i class="fas fa-fw fa-users"></i> Users
         <% end %>
-        <%= link_to search_path, class: 'header--item', data: { 'header-slide' => '#search-slide' } do %>
+        <%= link_to search_path, class: 'header--item', 
+                                 data: { 'header-slide' => '#search-slide' },
+                                 title: 'Show search' do %>
           <i class="fas fa-fw fa-search"></i> Search
         <% end %>
       <% end %>
@@ -33,7 +52,9 @@
       <% if user_signed_in? %>
         <% unless @community.is_fake %>
           <% if current_user.is_moderator %>
-            <%= link_to (@open_flags > 0) ? flag_queue_path : moderator_path, class: 'header--item' do %>
+            <%= link_to (@open_flags > 0) ? flag_queue_path : moderator_path, 
+                class: 'header--item',
+                title: 'Open moderator dashboard' do %>
               <i class="fas fa-fw fa-wrench"></i> Mod
               <% if @open_flags > 0 %>
                 <span class="header--alert"><%= @open_flags %></span>
@@ -43,34 +64,47 @@
         <% end %>
       <% end %>
       <% unless @community.is_fake %>
-        <%= link_to help_center_path, class: 'header--item' do %>
+        <%= link_to help_center_path, class: 'header--item', title: 'Open Help Center' do %>
           <i class="fas fa-fw fa-question-circle"></i> Help
         <% end %>
-        <%= link_to dashboard_path, class: 'header--item' do %>
-          <i class="fas fa-fw fa-th" title='Dashboard'></i>
+        <%= link_to dashboard_path, class: 'header--item', title: 'Open dashboard' do %>
+          <i class="fas fa-fw fa-th"></i>
         <% end %>
       <% end %>
       <% if user_signed_in? %>
-        <a href="#" class="header--item inbox-toggle is-visible-on-mobile" data-header-slide="#js-inbox">
+        <a href="#" class="header--item inbox-toggle is-visible-on-mobile" 
+                    data-header-slide="#js-inbox"
+                    title='Open notifications'
+                    >
           <% unread = current_user.unread_count %>
           <% if unread > 0 %>
             <span class="header--alert inbox-count"><%= unread %></span>
           <% end %>
-          <i class='fas fa-fw fa-inbox' title='Notifications'></i>
+          <i class='fas fa-fw fa-inbox'></i>
         </a>
-        <%= link_to qr_login_code_path, class: 'header--item' do %>
-          <i class="fas fa-fw fa-mobile-alt" aria-label="Mobile Sign In" title="Mobile Sign In"></i>
-        <% end %>
-        <%= link_to user_path(current_user), class: 'header--item is-complex is-visible-on-mobile' do %>
+        <%= link_to user_path(current_user), class: 'header--item is-complex is-visible-on-mobile',
+                                             title: 'Manage profile' do %>
           <img alt="user avatar" src="<%= avatar_url(current_user, 40) %>" class="header--item-image avatar-40">&nbsp;&nbsp;
           <span class="<%= SiteSetting['SiteHeaderIsDark'] ? 'has-color-white' : 'has-color-tertiary-600' %>"><%= current_user.reputation %></span>
         <% end %>
       <% end %>
-      <a href="#" class="header--item is-visible-on-mobile" aria-label="Show Communities" data-header-slide="#communities-slide"><i class="far fa-fw fa-caret-square-down"></i></a>
+      <a href="#" 
+         class="header--item is-visible-on-mobile" 
+         aria-label="Show Communities" 
+         data-header-slide="#communities-slide"
+         title="Show Communities"
+         >
+          <i class="far fa-fw fa-caret-square-down"></i>
+      </a>
       <% unless user_signed_in? %>
         <div class="header--item">
-          <%= link_to 'Sign Up', new_user_registration_path, class: 'button is-muted is-filled' %>
-          <%= link_to 'Sign In', new_user_session_path, class: 'button is-muted is-outlined' %>
+          <% if devise_sign_in_enabled? %>
+            <%= link_to 'Sign Up', new_user_registration_path, class: 'button is-muted is-filled' %>
+            <%= link_to 'Sign In', new_user_session_path, class: 'button is-muted is-outlined' %>
+          <% end %>
+          <% if sso_sign_in_enabled? %>
+            <%= link_to 'SSO Sign In', new_saml_user_session_path, class: 'button is-muted is-outlined' %>
+          <% end %>
         </div>
       <% end %>
       <% unless @community.is_fake %>
@@ -88,7 +122,11 @@
 
 <div class="header-slide" id="communities-slide">
   <% if user_signed_in? %>
-    <%= link_to 'Sign Out', destroy_user_session_path, method: :delete, class: 'button is-muted has-float-right' %>
+    <% if devise_sign_in_enabled? %>
+      <%= link_to 'Sign Out', destroy_user_session_path, method: :delete, class: 'button is-muted has-float-right', role: 'button' %>
+    <% else %>
+      <%= link_to 'Sign Out', destroy_saml_user_session_path, method: :delete, class: 'button is-muted has-float-right', role: 'button' %>
+    <% end %>
   <% end %>
   <h3 class="h-m-t-1">Communities</h3>
   <div class="community-header-list">
@@ -139,9 +177,9 @@
     </div>
   </div>
 
-  <%= form_tag search_path, method: :get do %>
+  <%= form_tag search_path, method: :get, role: 'search' do %>
     <div class="grid is-nowrap">
-        <%= text_field_tag :search, params[:search], class: 'form-element' %>
+        <%= search_field_tag :search, params[:search], class: 'form-element' %>
         <div class="h-m-1">
           <%= button_tag type: :submit, class: 'button is-filled is-outlined', name: nil do %>
             <i class="fa fa-search"></i>
@@ -158,6 +196,7 @@
   <div class="h-p-2 h-bg-tertiary-050 h-fw-bold">Notifications</div>
   <div class="h-p-2 h-bg-tertiary-050 h-fw-bold">
     <a class="no-unread js-read-all-notifs" href="#"><i class="fas fa-envelope-open"></i> Mark all as read</a>
+    <a href="/users/me/notifications" class="button is-muted is-small">See all your notifications &raquo;</a>
   </div>
   <div class="inbox--container h-p-2"></div>
 </div>
@@ -182,10 +221,19 @@
           <% end %>
         <% end %>
       <% end %>
-      <%= link_to 'Sign Out', destroy_user_session_path, method: :delete, class: 'menu--item' %>
+      <% if devise_sign_in_enabled? %>
+        <%= link_to 'Sign Out', destroy_user_session_path, method: :delete, class: 'menu--item', role: 'button' %>
+      <% else %>
+        <%= link_to 'Sign Out', destroy_saml_user_session_path, method: :delete, class: 'menu--item', role: 'button' %>
+      <% end %>
     <% else %>
-      <%= link_to 'Sign In', new_user_session_path, class: 'menu--item' %>
-      <%= link_to 'Sign Up', new_user_registration_path, class: 'menu--item' %>
+      <% if sso_sign_in_enabled? %>
+        <%= link_to 'SSO Sign In', new_saml_user_session_path, class: 'menu--item' %>
+      <% end %>
+      <% if devise_sign_in_enabled? %>
+        <%= link_to 'Sign In', new_user_session_path, class: 'menu--item' %>
+        <%= link_to 'Sign Up', new_user_registration_path, class: 'menu--item' %>
+      <% end %>
     <% end %>
   </div>
 </div>
@@ -226,8 +274,13 @@
                       class: "category-header--nav-item #{active?(current_cat) && !['tags', 'suggested_edit'].include?(controller_name) ? 'is-active' : ''}" %>
           <%= link_to 'Tags', category_tags_path(current_cat),
                       class: "category-header--nav-item #{active?(current_cat) && controller_name == 'tags' ? 'is-active' : ''}" %>
-          <%= link_to 'Edits', suggested_edits_queue_path(current_cat),
-                      class: "category-header--nav-item #{active?(current_cat) && controller_name == 'suggested_edit' ? 'is-active' : ''}" %>
+          <%= link_to suggested_edits_queue_path(current_cat),
+                      class: "category-header--nav-item #{active?(current_cat) && controller_name == 'suggested_edit' ? 'is-active' : ''}" do %>
+            Edits
+            <% if pending_suggestions?  && check_your_privilege('edit_posts') %>
+              <span class="badge is-status" title="Suggested edits pending"></span>
+            <% end %>
+          <% end %>
           <div class="category-header--nav-separator"></div>
           <%= link_to category_post_types_path(current_cat.id),
                       class: 'category-header--nav-item is-button' do %>
diff --git a/app/views/layouts/_matomo.html.erb b/app/views/layouts/_matomo.html.erb
index 8198772326ff64f2210e10abdbeb8cb929e21bf7..8ef62a3b08e5277cf7b97df3a54c1905172e437e 100644
--- a/app/views/layouts/_matomo.html.erb
+++ b/app/views/layouts/_matomo.html.erb
@@ -1,28 +1,28 @@
 <% unless SiteSetting['AnalyticsURL'].blank? || SiteSetting['AnalyticsSiteId'].blank? %>
-  <% cache [RequestContext.community, SiteSetting['AnalyticsURL'], SiteSetting['AnalyticsSiteId']] do %>
-    <script defer type="text/javascript">
-        var _paq = window._paq = window._paq || [];
-        /* tracker methods like "setCustomDimension" should be called before "trackPageView" */
+  <script defer type="text/javascript">
+    (async () => {
+      var _paq = window._paq = window._paq || [];
+      /* tracker methods like "setCustomDimension" should be called before "trackPageView" */
 
-        const user = await QPixel.user();
-        if (user) {
-          _paq.push(['setUserId', `${user.id}`]);
-        } else {
-          _paq.push(['resetUserId']);
-        }
+      const user = await QPixel.user();
+      if (user) {
+        _paq.push(['setUserId', `${user.id}`]);
+      } else {
+        _paq.push(['resetUserId']);
+      }
 
-        _paq.push(['trackPageView']);
-        _paq.push(['enableLinkTracking']);
-        (function() {
-          var u="<%= SiteSetting['AnalyticsURL'] %>";
-          _paq.push(['setTrackerUrl', u+'matomo.php']);
-          _paq.push(['setSiteId', '<%= SiteSetting['AnalyticsSiteId'] %>']);
-          var d=document, g=d.createElement('script'), s=d.getElementsByTagName('script')[0];
-          g.async=true; g.src=u+'matomo.js'; s.parentNode.insertBefore(g,s);
-        })();
-    </script>
-    <noscript><p>
-      <img referrerpolicy="no-referrer-when-downgrade" src="<%= SiteSetting['AnalyticsURL'] %>matomo.php?idsite=<%= SiteSetting['AnalyticsSiteId'] %>&amp;rec=1" style="border:0" alt="" />
-    </p></noscript>
-  <% end %>
+      _paq.push(['trackPageView']);
+      _paq.push(['enableLinkTracking']);
+      (function() {
+        var u="<%= SiteSetting['AnalyticsURL'] %>";
+        _paq.push(['setTrackerUrl', u+'matomo.php']);
+        _paq.push(['setSiteId', '<%= SiteSetting['AnalyticsSiteId'] %>']);
+        var d=document, g=d.createElement('script'), s=d.getElementsByTagName('script')[0];
+        g.async=true; g.src=u+'matomo.js'; s.parentNode.insertBefore(g,s);
+      })();
+    })();
+  </script>
+  <noscript><p>
+    <img referrerpolicy="no-referrer-when-downgrade" src="<%= SiteSetting['AnalyticsURL'] %>matomo.php?idsite=<%= SiteSetting['AnalyticsSiteId'] %>&amp;rec=1" style="border:0" alt="" />
+  </p></noscript>
 <% end %>
diff --git a/app/views/layouts/_sidebar.html.erb b/app/views/layouts/_sidebar.html.erb
index 33a713a42dd3289ecf05bd99af8a2a79894d2215..9f608f5f859f92002e71500fb939186c8c85ef43 100644
--- a/app/views/layouts/_sidebar.html.erb
+++ b/app/views/layouts/_sidebar.html.erb
@@ -1,4 +1,4 @@
-<div class="grid--cell is-4-lg is-12">
+<div class="grid--cell is-4-lg is-12" role="complementary">
   <%= yield(:sidebar) %>
 
   <% notice_text = SiteSetting['SidebarNoticeText'] %>
@@ -92,10 +92,10 @@
       <div class="widget has-margin-4">
         <div class="widget--header">
           <%= link_to 'edit', subscriptions_path, class: 'widget--header-link' %>
-          Subscribe to Questions
+          Subscribe by Email
         </div>
         <div class="widget--body">
-          <p>You can subscribe to
+          <p>You can subscribe by email to
             <%= link_to 'all new questions', new_subscription_path(type: 'all', return_to: request.path) %>
             (from all categories) or to
             <%= link_to 'interesting questions', new_subscription_path(type: 'interesting', return_to: request.path) %>.</p>
@@ -109,11 +109,11 @@
       <img src="/assets/codidact.png" alt="Codidact logo" class="codidact-logo" />
     </div>
     <div class="widget--body">
+      <% pitch = SiteSetting['NetworkPitch'] %>
       <% chat = SiteSetting['ChatLink'] %>
-      <p>
-        This community is part of the <a href="https://codidact.com">Codidact network</a>.
-        We have other communities too &mdash; take a look!
-      </p>
+      <% if pitch.present? %>
+        <%= raw(sanitize(render_markdown(pitch), scrubber: scrubber)) %>
+      <% end %>
       <% if chat.present? %>
         <p>
           You can also <%= link_to 'join us in chat', chat %>!
diff --git a/app/views/layouts/application.html.erb b/app/views/layouts/application.html.erb
index c259af3ea187f26a35ff862472f5bd2d4a8bc418..d4b6dd5c60312b81ec808cf5126069b48f3503bc 100644
--- a/app/views/layouts/application.html.erb
+++ b/app/views/layouts/application.html.erb
@@ -3,7 +3,9 @@
 <head>
   <%= render 'layouts/head' %>
 </head>
-<body class="<%= Rails.env.development? ? 'development' : '' %>">
+<body class="<%= Rails.env.development? ? 'development' : '' %>"
+      data-user-id="<%= user_signed_in? ? current_user.id : 'none' %>"
+      data-mathjax="<%= SiteSetting['MathJaxEnabled'] %>">
   <%= render 'layouts/header' %>
 
   <main class="container">
diff --git a/app/views/layouts/devise_mailer.html.erb b/app/views/layouts/devise_mailer.html.erb
index b3a260e52eef16986f15dbea5ea47367c3d23fde..9f8e4a39c567c923c1b5cafb8f71f003b6aeda21 100644
--- a/app/views/layouts/devise_mailer.html.erb
+++ b/app/views/layouts/devise_mailer.html.erb
@@ -3,7 +3,7 @@
 <head>
   <meta charset="utf-8" />
   <title><%= message.subject %></title>
-  <%= stylesheet_link_tag 'https://cdn.jsdelivr.net/npm/@codidact/co-design@0.12.5/dist/codidact.css' %>
+  <%= stylesheet_link_tag 'https://cdn.jsdelivr.net/npm/@codidact/co-design@latest/dist/codidact.css' %>
   <%= stylesheet_link_tag 'application', media: 'all' %>
 </head>
 <body>
@@ -26,7 +26,11 @@
         London, UK<br/>
         WC2H 9JQ
       </p>
-      <p><a href="mailto:info@codidact.org">info@codidact.org</a></p>
+      <p>
+        <a href="mailto:<%= SiteSetting['SupportReceiverEmail'] || 'info@codidact.org' %>">
+          <%= SiteSetting['SupportReceiverEmail'] || 'info@codidact.org' %>
+        </a>
+      </p>
     </div>
   </div>
 </body>
diff --git a/app/views/layouts/mailer.html.erb b/app/views/layouts/mailer.html.erb
index bdf01890b14b0945a932e0c771490e211cc4899a..fe13b1f7d17d500269a87010144c9b773b659258 100644
--- a/app/views/layouts/mailer.html.erb
+++ b/app/views/layouts/mailer.html.erb
@@ -3,7 +3,7 @@
 <head>
   <meta charset="utf-8" />
   <title><%= message.subject %></title>
-  <%= stylesheet_link_tag 'https://cdn.jsdelivr.net/npm/@codidact/co-design@0.12.5/dist/codidact.css' %>
+  <%= stylesheet_link_tag 'https://cdn.jsdelivr.net/npm/@codidact/co-design@latest/dist/codidact.css' %>
   <%= stylesheet_link_tag 'application', media: 'all' %>
 </head>
 <body>
@@ -26,7 +26,11 @@
       London, UK<br/>
       WC2H 9JQ
     </p>
-    <p><a href="mailto:info@codidact.org">info@codidact.org</a></p>
+    <p>
+      <a href="mailto:<%= SiteSetting['SupportReceiverEmail'] || 'info@codidact.org' %>">
+        <%= SiteSetting['SupportReceiverEmail'] || 'info@codidact.org' %>
+      </a>
+    </p>
   </div>
 </div>
 </body>
diff --git a/app/views/layouts/mailer.text.erb b/app/views/layouts/mailer.text.erb
index 37c98dc48b42e895f59282a2a6ad92515511aa0b..8b0a6ed004f254b3bc6a5e91326a605433d7b89c 100644
--- a/app/views/layouts/mailer.text.erb
+++ b/app/views/layouts/mailer.text.erb
@@ -6,4 +6,4 @@
 71-75 Shelton Street
 London, UK
 WC2H 9JQ
-info@codidact.org
+<%= SiteSetting['SupportReceiverEmail'] || 'info@codidact.org' %>
diff --git a/app/views/licenses/_form.html.erb b/app/views/licenses/_form.html.erb
index 3633062856ec617d18fc828d5d15f38869a5d1b9..e0a18295735221dd8becc53e0f5ea28a67792f87 100644
--- a/app/views/licenses/_form.html.erb
+++ b/app/views/licenses/_form.html.erb
@@ -20,9 +20,9 @@
     <%= f.check_box :default, class: 'form-checkbox-element' %>
     <%= f.label :default, 'Default license?' %>
     <span class="form-caption">
-      Set this license as the site-default license. A site can only have one default license.
+      Set this license as the community-default license. A community can only have one default license.
     </span>
   </div>
 
   <%= f.submit 'Save', class: 'button is-filled' %>
-<% end %>
\ No newline at end of file
+<% end %>
diff --git a/app/views/licenses/_license.html.erb b/app/views/licenses/_license.html.erb
index 616015f802ccd4843bfd120ac64960ae674e1cae..57544538fc7326d1d5e2176e71c0e5c543d0699e 100644
--- a/app/views/licenses/_license.html.erb
+++ b/app/views/licenses/_license.html.erb
@@ -15,7 +15,7 @@
     <p>
       <%= link_to 'Edit', edit_license_path(license), class: 'button is-outlined' %>
       <%= link_to license.enabled? ? 'Disable' : 'Enable', toggle_license_path(license), class: 'button is-outlined',
-                  method: :post %>
+                  method: :post, role: 'button' %>
     </p>
   </div>
 </div>
\ No newline at end of file
diff --git a/app/views/micro_auth/apps/index.html.erb b/app/views/micro_auth/apps/index.html.erb
index 8f9984adcd50dd463dcec96180827af4c6f66736..5e19d56f3ac09820827aaa20175ad764363f0060 100644
--- a/app/views/micro_auth/apps/index.html.erb
+++ b/app/views/micro_auth/apps/index.html.erb
@@ -3,7 +3,7 @@
 <h1>Your OAuth Apps</h1>
 
 <div class="flex-row jc-sb">
-  <%= link_to new_oauth_app_path, class: 'button is-filled' do %>
+  <%= link_to new_oauth_app_path, class: 'button is-filled', 'aria-label': 'Create new OAuth App' do %>
     <i class="fas fa-plus-circle"></i> Create
   <% end %>
 
@@ -11,9 +11,9 @@
     <div class="button-list is-gutterless">
       <% classes = 'button is-outlined is-muted' %>
       <%= link_to 'yours', oauth_apps_path(request.query_parameters.except(:admin)),
-                  class: "#{classes} #{params[:admin].nil? ? 'is-active' : ''}" %>
+                  class: "#{classes} #{params[:admin].nil? ? 'is-active' : ''}", 'aria-label': 'View your OAuth Apps' %>
       <%= link_to 'admin', oauth_apps_path(request.query_parameters.merge(admin: true)),
-                  class: "#{classes} #{params[:admin].present? ? 'is-active' : ''}" %>
+                  class: "#{classes} #{params[:admin].present? ? 'is-active' : ''}", 'aria-label': 'View Admin OAuth Apps' %>
     </div>
   <% end %>
 </div>
@@ -29,7 +29,9 @@
       <%= text_field_tag :search, params[:search], class: 'form-element' %>
     </div>
     <div class="actions has-padding-bottom-1">
-      <%= submit_tag 'Search', class: 'button' %>
+     <%= button_tag type: :submit, class: 'button is-medium is-filled is-outlined', name: nil do %>
+        <i class="fa fa-search"></i>
+      <% end %>
     </div>
   </div>
 <% end %>
diff --git a/app/views/micro_auth/authentication/initiate.html.erb b/app/views/micro_auth/authentication/initiate.html.erb
index ff4f9222c643b54bbaebee42bc9d7a763b54002e..bc1e9bebb0d83a7d76f53f1740328016a0031531 100644
--- a/app/views/micro_auth/authentication/initiate.html.erb
+++ b/app/views/micro_auth/authentication/initiate.html.erb
@@ -24,5 +24,5 @@
 
 <p>You can choose to allow or deny this request.</p>
 
-<%= link_to 'Allow', approve_oauth_path(request.query_parameters), method: :post, class: 'button is-green is-filled' %>
-<%= link_to 'Deny', reject_oauth_path, class: 'button is-danger is-outlined' %>
+<%= link_to 'Allow', approve_oauth_path(request.query_parameters), method: :post, class: 'button is-green is-filled', role: 'button' %>
+<%= link_to 'Deny', reject_oauth_path, class: 'button is-danger is-outlined', role: 'button' %>
diff --git a/app/views/mod_warning/current.html.erb b/app/views/mod_warning/current.html.erb
index c98f4bf9384c76d98b504413b118d67a1d51d0c5..f5f7eb3e92f6ef5328a7d24111967a5a308f5350 100644
--- a/app/views/mod_warning/current.html.erb
+++ b/app/views/mod_warning/current.html.erb
@@ -10,7 +10,11 @@
             <%= raw(sanitize(@warning.body_as_html, scrubber: scrubber)) %>
             <p>Your account has been <strong>temporarily suspended</strong> (ends <span title="<%= current_user.community_user.suspension_end.iso8601 %>">in <%= time_ago_in_words(current_user.community_user.suspension_end) %></span>). We look forward to your return and continued contributions to the site after this period. In the event of continued rule violations after this period, your account may be suspended for longer periods. If you have any questions about this suspension or would like to dispute it, <a href="mailto:<%= SiteSetting['AdministratorContactEmail'] %>">contact us</a>.</p>
 
-            <%= link_to 'Sign Out', destroy_user_session_path, method: :delete, class: 'button is-danger is-outlined' %>
+            <% if devise_sign_in_enabled? %>
+                <%= link_to 'Sign Out', destroy_user_session_path, method: :delete, class: 'button is-danger is-outlined', role: 'button' %>
+            <% else %>
+                <%= link_to 'Sign Out', destroy_saml_user_session_path, method: :delete, class: 'button is-danger is-outlined', role: 'button' %>
+            <% end %>
         </div>
     <% else %>
         <div class="notice is-danger">
@@ -27,7 +31,11 @@
                 </label>
 
                 <%= submit_tag 'Continue', class: 'button is-filled' %>
-                <%= link_to 'Sign Out', destroy_user_session_path, method: :delete, class: 'button is-danger is-outlined' %>
+                <% if devise_sign_in_enabled? %>
+                    <%= link_to 'Sign Out', destroy_user_session_path, method: :delete, class: 'button is-danger is-outlined', role: 'button' %>
+                <% else %>
+                    <%= link_to 'Sign Out', destroy_saml_user_session_path, method: :delete, class: 'button is-danger is-outlined', role: 'button' %>
+                <% end %>
             <% end %>
         </div>
     <% end %>
@@ -46,7 +54,11 @@
             </label>
 
             <%= submit_tag 'Continue', class: 'button is-filled' %>
-            <%= link_to 'Sign Out', destroy_user_session_path, method: :delete, class: 'button is-danger is-outlined' %>
+            <% if devise_sign_in_enabled? %>
+                <%= link_to 'Sign Out', destroy_user_session_path, method: :delete, class: 'button is-danger is-outlined', role: 'button' %>
+            <% else %>
+                <%= link_to 'Sign Out', destroy_saml_user_session_path, method: :delete, class: 'button is-danger is-outlined', role: 'button' %>
+            <% end %>
         <% end %>
     </div>
 <% end %>
\ No newline at end of file
diff --git a/app/views/notifications/index.html.erb b/app/views/notifications/index.html.erb
index 0672c4558c74b5a65afaf9846b60f0fe28a01810..33c2be2fbeef1853a3268278c139d0cb200f010c 100644
--- a/app/views/notifications/index.html.erb
+++ b/app/views/notifications/index.html.erb
@@ -1,3 +1,5 @@
+<%= render 'users/tabs', user: current_user %>
+
 <h1>Your Inbox</h1>
 <p>You'll find all your inbox messages here, as far back as your account goes.</p>
 
diff --git a/app/views/pinned_links/_form.html.erb b/app/views/pinned_links/_form.html.erb
index a69f010bbaf5f43c325cbec03d10074089acafb4..80a3191e6d31ac42c89482e8d897b557d21af403 100644
--- a/app/views/pinned_links/_form.html.erb
+++ b/app/views/pinned_links/_form.html.erb
@@ -48,5 +48,5 @@
   </div>
 
   <%= f.submit "Update", class: "button is-filled" %>
-  <%= link_to "Cancel", pinned_links_path(global: params[:global]), class: "button" %>
+  <%= link_to "Cancel", pinned_links_path(global: params[:global]), class: "button", role: 'button' %>
 <% end %>
\ No newline at end of file
diff --git a/app/views/post_history/post.html.erb b/app/views/post_history/post.html.erb
index 5e2b284d614e2a660617840da9db31091a1d0e70..c31b586ea6fc94e7efbfbd49a8f305e470dfd99c 100644
--- a/app/views/post_history/post.html.erb
+++ b/app/views/post_history/post.html.erb
@@ -1,11 +1,15 @@
+<% @show_content = !!defined?(show_content) ? show_content : true %>
+
 <h1>Post History</h1>
 
+<% if @show_content %>
 <div class="item-list">
   <%= render 'posts/type_agnostic', post: @post, show_category_tag: true, show_type_tag: true, last_activity: false %>
 </div>
+<% end %>
 
 <% @history.each.with_index do |event, index| %>
-  <details class="history-event">
+  <details class="history-event" id="<%= @history.size - index %>">
     <summary>
       <strong>#<%= @history.size - index %>: <%= event.post_history_type.name.humanize %></strong>
       <span class="has-color-tertiary-600 history-meta">
@@ -24,15 +28,45 @@
           <br/><em><%= event.comment %></em>
         <% end %>
       </span>
+      <%= render 'shared/copy_link', classes: ["button", "is-small", "is-muted", "is-outlined"],
+                                     desc: "Copy a permanent link to revision #{@history.size - index}",
+                                     id: "#{@post.id}-#{@history.size - index}", 
+                                     md: post_history_share_link_md(@post, @history, index), 
+                                     raw: post_history_share_link(@post, @history, index) 
+                                     %>
     </summary>
-    <% if (event.before_title.present? || event.after_title.present?) && event.before_title != event.after_title %>
-      <%= render 'diff', before: event.before_title, after: event.after_title, post: @post %>
-    <% end %>
-    <% if (event.before_state.present? || event.after_state.present?) && event.before_state != event.after_state %>
-      <%= render 'diff', before: event.before_state, after: event.after_state, post: @post %>
-    <% end %>
-    <% if (event.before_tags.present? || event.after_tags.present?) && event.before_tags != event.after_tags %>
-      <%= render 'diff', before: event.before_tags, after: event.after_tags, post: @post %>
+    <% if event.allowed_to_see_details?(current_user) %>
+      <% if event.hidden? %>
+        <div class="notice is-warning is-filled">
+          <p><i class="fas fa-info-circle"></i> <strong>Hidden revision</strong></p>
+          <p>
+            This revision is hidden because of a redaction. You have access to the details because
+            <% if current_user == event.user %>
+              you performed the redaction,
+            <% elsif current_user == @post.user %>
+              you are the post author,
+            <% elsif current_user&.is_admin %>
+              you are an administrator,
+            <% end %>
+            but you should not share this revision with others.
+          </p>
+        </div>
+      <% end %>
+      <% if (event.before_title.present? || event.after_title.present?) && event.before_title != event.after_title %>
+        <%= render 'diff', before: event.before_title, after: event.after_title, post: @post %>
+      <% end %>
+      <% if (event.before_state.present? || event.after_state.present?) && event.before_state != event.after_state %>
+        <%= render 'diff', before: event.before_state, after: event.after_state, post: @post %>
+      <% end %>
+      <% if (event.before_tags.present? || event.after_tags.present?) && event.before_tags != event.after_tags %>
+        <%= render 'diff', before: event.before_tags, after: event.after_tags, post: @post %>
+      <% end %>
+    <% elsif [event.before_title, event.after_title,
+              event.before_state, event.after_state,
+              event.before_tags, event.after_tags].any?(&:present?) %>
+      <p>
+        <em>The detailed changes of this event are hidden because of a redaction.</em>
+      </p>
     <% end %>
   </details>
 <% end %>
diff --git a/app/views/posts/_article_list.html.erb b/app/views/posts/_article_list.html.erb
index d8deeaf561080a87ff044f6133d2283310110cdf..fb48d014d17a1148fadc47b5f7a702b2494063eb 100644
--- a/app/views/posts/_article_list.html.erb
+++ b/app/views/posts/_article_list.html.erb
@@ -3,7 +3,7 @@
 <% @show_category_tag = !!defined?(show_category_tag) ? show_category_tag : false %>
 <% @last_activity = !!defined?(last_activity) ? last_activity : true %>
 <div class="item-list--item <%= post.deleted ? 'deleted-content' : '' %>" data-ckb-list-item data-ckb-item-type="link">
-  <div class="item-list--number-value">
+  <div class="item-list--number-value" title="Score: <%= post.score %>">
     <div class="meter is-question-score">
       <div class="meter--bar is-<%= (post.score * 100).to_i %>%"><%= (post.score * 100).to_i %>%</div>
     </div>
@@ -17,7 +17,7 @@
   <div class="item-list--text-value is-primary">
     <div class="post-list--title">
       <% if @show_category_tag %>
-        <span class="badge is-tag is-filled"><%= defined?(@category) ? @category.name : post.category.name %></span>
+        <span class="badge is-tag is-filled"><%= post.category.name %></span>
       <% end %>
       <%= link_to post.title, generic_share_link(post), 'data-ckb-item-link' => '' %>
     </div>
diff --git a/app/views/posts/_edit_comment.html.erb b/app/views/posts/_edit_comment.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..80a952ccd6a17e1f2d98c702d131aa2f06017378
--- /dev/null
+++ b/app/views/posts/_edit_comment.html.erb
@@ -0,0 +1,39 @@
+<%#
+  Edit comment reusable partial.
+  Variables:
+    comment       : [String, Nil] optional, initial value of the field (default '')
+    cur_length    : [Integer, Nil] optional, current character length (default 0)
+    min_length    : [Integer, Nil] optional, the minimum allowed length (default 0)
+    max_length    : [Integer, Nil] optional, the maximum allowed length (default 255)
+%>
+
+<%
+  # Defaults
+  comment = (defined?(comment) ? comment : nil) || ''
+  cur_length = (defined?(cur_length) ? cur_length : nil) || 0
+  min_length = (defined?(min_length) ? min_length : nil) || 0
+  max_length = (defined?(max_length) ? max_length : nil) || 255
+%>
+
+<div class="form-group">
+  <%= label_tag :edit_comment, t('posts.edit_comment_label'), class: 'form-element' %>
+  <%= text_field_tag :edit_comment,
+                     nil,
+                     class: 'form-element',
+                     value: comment,
+                     data: { character_count: ".js-character-count-edit-comment" } %>
+  <div class="clearfix">
+    <%= render 'shared/char_count', type: 'edit-comment', cur: cur_length, min: min_length, max: max_length %>
+  </div>
+</div>
+<div class="form-group">
+  <div class="checkbox-setting">
+    <div class="checkbox-setting--desc">
+      <%= label_tag :redact, t('posts.redact_label'), class: 'form-element' %>
+      <span class="form-caption"><%= t('posts.redact_explanation') %></span>
+    </div>
+    <div class="checkbox-setting--value">
+      <%= check_box_tag :redact, true, false, class: 'form-checkbox-element' %>
+    </div>
+  </div>
+</div>
\ No newline at end of file
diff --git a/app/views/posts/_expanded.html.erb b/app/views/posts/_expanded.html.erb
index 82733701ff6fe23351d0afc933a5368438a71292..5f28e9d234df08f43cd2258f40d05e917bb3d647 100644
--- a/app/views/posts/_expanded.html.erb
+++ b/app/views/posts/_expanded.html.erb
@@ -18,7 +18,7 @@
       <% title = post.title +
                  (post.closed && !post.duplicate_post ? " [closed]" : "") +
                  (post.duplicate_post ? " [duplicate]" : "") %>
-      <span class="post--title-text"><%= title %></span>
+      <a href="<%= generic_share_link(post) %>" class="post--title-text"><%= title %></a>
       <% if category.display_post_types.reject { |e| e.to_s.empty? }.size > 1 %>
         <%= post_type_badge(post_type) %>
       <% end %>
@@ -36,33 +36,35 @@
   <% end %>
 
   <div class="post--container <%= 'deleted-content' if post.deleted? %> grid is-nowrap">
-    <% if post_type.has_votes %>
+    <% if post_type.has_votes || (user_signed_in? && post.post_type.has_reactions && post.post_type.reactions.any?) %>
       <div>
-        <div class="post--votes has-text-align-center" title="Score : <%= post.score %>">
-          <% existing_vote = my_vote(post) %>
-          <% unless post.locked? %>
-            <button class="vote-button button is-icon-only-button <%= (existing_vote&.vote_type == 1) ? 'is-active' : '' %>"
-                    data-vote-type="1" data-vote-id="<%= existing_vote&.id %>" id="post-<%= post.id %>-up" aria-label="Upvote">
-                <svg width="2em" height="1.33em" viewbox="0 0 100 50">
-                  <path d="M50,0 L100,50 L0,50 Z" fill="currentColor" />
-                </svg>
-            </button>
-          <% end %>
-          <div class="score has-font-size-subheading js-upvote-count">
-            +<%= post.upvote_count %>
-          </div>
-          <div class="score has-font-size-subheading js-downvote-count">
-            &minus;<%= post.downvote_count %>
+        <% if post_type.has_votes %>
+          <div class="post--votes has-text-align-center" title="Score: <%= post.score %>">
+            <% existing_vote = my_vote(post) %>
+            <% unless post.locked? %>
+              <button class="vote-button button is-icon-only-button <%= (existing_vote&.vote_type == 1) ? 'is-active' : '' %>"
+                      data-vote-type="1" data-vote-id="<%= existing_vote&.id %>" id="post-<%= post.id %>-up" aria-label="Upvote">
+                  <svg width="2em" height="1.33em" viewbox="0 0 100 50">
+                    <path d="M50,0 L100,50 L0,50 Z" fill="currentColor" />
+                  </svg>
+              </button>
+            <% end %>
+            <div class="score has-font-size-subheading js-upvote-count">
+              +<%= post.upvote_count %>
+            </div>
+            <div class="score has-font-size-subheading js-downvote-count">
+              &minus;<%= post.downvote_count %>
+            </div>
+            <% unless post.locked? %>
+              <button class="vote-button button is-icon-only-button <%= (existing_vote&.vote_type == -1) ? 'is-active' : '' %>"
+                      data-vote-type="-1" data-vote-id="<%= existing_vote&.id %>" id="post-<%= post.id %>-up" aria-label="Downvote">
+                  <svg width="2em" height="1.33em" viewbox="0 0 100 50">
+                    <path d="M0,0 L100,0 L50,50 Z" fill="currentColor" />
+                  </svg>
+              </button>
+            <% end %>
           </div>
-          <% unless post.locked? %>
-            <button class="vote-button button is-icon-only-button <%= (existing_vote&.vote_type == -1) ? 'is-active' : '' %>"
-                    data-vote-type="-1" data-vote-id="<%= existing_vote&.id %>" id="post-<%= post.id %>-up" aria-label="Downvote">
-                <svg width="2em" height="1.33em" viewbox="0 0 100 50">
-                  <path d="M0,0 L100,0 L50,50 Z" fill="currentColor" />
-                </svg>
-            </button>
-          <% end %>
-        </div>
+        <% end %>
         <% if user_signed_in? && post.post_type.has_reactions && post.post_type.reactions.any? %>
           <div class="post--react has-text-align-center">
             <% unless post.locked? %>
@@ -83,6 +85,15 @@
       <%= render('reactions/list', post: post) if post.reactions %>
 
       <% if post_type.is_closeable && post.closed %>
+        <% if current_user == post.user %>
+          <div class="notice is-info has-margin-2">
+            <p>
+              <i class="fas fa-info-circle"></i>
+              <%= t 'posts.post_closed_guidance' %>
+            </p>
+          </div>
+        <% end %>
+
         <div class="notice has-margin-2">
           <p class="has-font-weight-normal">
             <strong>Closed</strong>
@@ -123,7 +134,7 @@
             </p>
           <% end %>
 
-          <p>Users with the Curate privilege may vote to undelete this post if it has been deleted incorrectly.</p>
+          <p>Users with the Curate privilege may vote to restore this post if it has been deleted incorrectly.</p>
         </div>
       <% end %>
 
@@ -134,22 +145,42 @@
             on <%= post.locked_at.strftime('%b %e, %Y at %H:%M') %>.
           </p>
         </div>
-      <% end %>
+	<% end %>
 
+	<% if post.spam_flag_pending? && user_signed_in? %>
+	<div class="notice has-margin-2">
+	  <% if post.user == current_user %>
+	  <p>Your post has been flagged by members of our community. Please review our <a href="/help/spam">guidelines for promotional content</a>.</p>
+	  <% else %>
+	  <p><strong>Possible spam:</strong> this post has pending flags for spam.  Be careful when following links.</p>
+	  <% end %>
+	</div>
+	<% end %>
       <% if post_type.is_public_editable && post.pending_suggested_edit? %>
         <% if check_your_post_privilege(post, 'edit_posts') %>
           <div class="notice h-p-2">
-            <p class="h-m-0"><i class="fa fa-pencil-alt h-m-l-2 h-c-red-600"></i> There is a <strong>pending suggested edit</strong> on this post. <a href="<%= suggested_edit_url(post.pending_suggested_edit.id) %>" class="button is-outlined is-muted is-small">Review changes</a></p>
+            <p class="h-m-0">
+              <i class="fa fa-pencil-alt h-m-l-2 h-c-red-600"></i>
+              There is a <strong>pending suggested edit</strong> on this post.
+              <a href="<%= suggested_edit_url(post.pending_suggested_edit.id) %>" class="button is-outlined is-muted is-small" role="button" aria-label="Review changes of suggested edit">Review changes</a></p>
           </div>
-        <% elsif post.pending_suggested_edit.user.id == current_user&.id %>
+        <% elsif post.pending_suggested_edit.user&.id == current_user&.id %>
           <div class="notice h-p-2">
-            <p class="h-m-0"><i class="fa fa-pencil-alt h-m-l-2 h-c-tertiary-600"></i> Your suggested edit on this post is <strong><%= link_to 'pending review', suggested_edit_url(post.pending_suggested_edit.id) %></strong>.</p>
+            <p class="h-m-0">
+              <i class="fa fa-pencil-alt h-m-l-2 h-c-tertiary-600"></i>
+              Your suggested edit on this post is <strong>
+              <%= link_to 'pending review', suggested_edit_url(post.pending_suggested_edit.id) %></strong>.</p>
           </div>
         <% end %>
-      <% end %>
+	<% end %>
 
-      <div class="post--body">
-        <%= raw(sanitize(post.body, scrubber: scrubber)) %>
+	<div class="post--body">
+	  <% effective_post = raw(sanitize(post.body, scrubber: scrubber)) %>
+	  <% if post.spam_flag_pending? && !user_signed_in? %>
+	    <%= sanitize(effective_post, attributes: %w()) %>
+	  <% else %>
+            <%= effective_post %>
+	  <% end %>
 
         <% been_edited = post.last_edited_by_id != nil %>
         <% if been_edited then last_edited_by_self = post.user_id == post.last_edited_by_id end %>
@@ -200,11 +231,12 @@
 
         <div class="post--actions">
           <div class="tools">
-            <%= link_to generic_share_link(post), class: 'tools--item js-permalink' do %>
-              <i class="fa fa-link"></i>
-              <span class="js-text">Permalink</span>
-            <% end %>
-            <%= link_to post_history_path(post), class: 'tools--item' do %>
+            <%= render "shared/copy_link", classes: ["tools--item"],
+                                           desc: "Copy a permanent link to this post",
+                                           id: post.id, 
+                                           md: generic_share_link_md(post), 
+                                           raw: generic_share_link(post) %>
+            <%= link_to post_history_path(post), class: 'tools--item', 'aria-label': 'View history of this post' do %>
               <i class="fa fa-history"></i>
               History
             <% end %>
@@ -216,7 +248,7 @@
                     Review suggested edit
                   <% end %>
                 <% else %>
-                  <%= link_to edit_post_path(post), class: 'tools--item' do %>
+                  <%= link_to edit_post_path(post), class: 'tools--item', 'aria-label': 'Edit this post' do %>
                     <i class="fa fa-pencil-alt"></i>
                     Edit
                   <% end %>
@@ -224,8 +256,13 @@
               <% elsif !current_user.nil? %>
                 <% if post.pending_suggested_edit? %>
                   <span class="tools--item">suggested edit pending...</span>
+		<% elsif post_type.is_freely_editable %>
+                  <%= link_to edit_post_path(post), class: 'tools--item', 'aria-label': 'Edit this post' do %>
+                    <i class="fa fa-pencil-alt"></i>
+                    Edit
+		  <% end %>
                 <% else %>
-                  <%= link_to edit_post_path(post), class: 'tools--item' do %>
+                  <%= link_to edit_post_path(post), class: 'tools--item', 'aria-label': 'Suggest edit to this post' do %>
                     <i class="fa fa-pencil-alt"></i>
                     Suggest edit
                   <% end %>
@@ -233,7 +270,7 @@
               <% end %>
             <% end %>
             <% if user_signed_in? %>
-              <a href="#" class="flag-dialog-link tools--item">
+              <a href="#" class="flag-dialog-link tools--item" role="button" aria-label="Flag this post">
                 <i class="fa fa-flag"></i>
                 Flag
               </a>
@@ -241,12 +278,12 @@
             <% unless post.locked? || !post_type.is_closeable %>
               <% if check_your_privilege('flag_close') || (post.user_id === current_user&.id)%>
                 <% if !post.closed %>
-                  <a href="#" class="close-dialog-link tools--item">
+                  <a href="#" class="close-dialog-link tools--item" role="button" aria-label="Close this post">
                     <i class="fa fa-lock"></i>
                     Close
                   </a>
                 <% elsif post.closed %>
-                  <%= link_to reopen_post_path(post), method: :post, class: 'reopen-question tools--item' do %>
+                  <%= link_to reopen_post_path(post), method: :post, class: 'reopen-question tools--item', role: 'button', 'aria-label': 'Reopen this post' do %>
                     <i class="fa fa-unlock"></i>
                     Reopen
                   <% end %>
@@ -257,13 +294,15 @@
               <% unless post.locked? %>
                 <% if !post.deleted %>
                   <%= link_to delete_post_path(post), method: :post,
-                              data: { confirm: 'Are you sure you want to delete this post?' }, class: "tools--item is-danger" do %>
+                              data: { confirm: 'Are you sure you want to delete this post?' }, class: "tools--item is-danger",
+                              role: 'button', 'aria-label': 'Delete this post' do %>
                     <i class="fa fa-trash"></i>
                     Delete
                   <% end %>
                 <% else %>
                   <%= link_to restore_post_path(post), method: :post,
-                              data: { confirm: 'Restore this post, making it visible to regular users?' }, class: "tools--item is-danger is-filled" do %>
+                              data: { confirm: 'Restore this post, making it visible to regular users?' }, class: "tools--item is-danger is-filled",
+                              role: 'button', 'aria-label': 'Restore this post' do %>
                     <i class="fa fa-undo"></i>
                     Restore
                   <% end %>
@@ -271,7 +310,7 @@
               <% end %>
             <% end %>
             <% if check_your_privilege('flag_curate') %>
-              <a href="javascript:void(0);" data-modal="#mod-tools-<%= post.id %>" class="tools--item">
+              <a href="javascript:void(0);" data-modal="#mod-tools-<%= post.id %>" class="tools--item" role="button" aria-label="Open Moderator Tools">
                 <i class="fa fa-wrench"></i>
                 Tools
               </a>
@@ -288,7 +327,7 @@
                                    end %>
 
               <% if flags_count > 0 %>
-                <a href="#" class="show-all-flags-dialog-link tools--item">
+                <a href="#" class="show-all-flags-dialog-link tools--item" role="button">
                   <i class="fa fa-exclamation-triangle"></i>
                   Show <%= pluralize(flags_count - own_flags_count, 'flag') %>
                 </a>
@@ -299,9 +338,26 @@
 
         <div class="post--action-dialog js-flag-box">
           <div class="widget">
-            <div class="widget--header">Why does this post require moderator attention?</div>
+            <div class="widget--header">Why does this post require attention from curators or moderators?</div>
+            <% if current_user&.has_active_flags?(post) %>
+              <div class="widget--body">
+                <div class="notice is-warning">
+                  You already have active flags on this post:
+                  <ul>
+                    <% current_user.active_flags(post).each do |flag| %>
+                      <li>
+                        <strong><%= flag.post_flag_type&.name || 'other' %></strong>
+                        <% unless flag.reason.nil? %>
+                          <span class="has-color-tertiary-600"><%= flag.reason %></span>
+                        <% end %>
+                      </li>
+                    <% end %>
+                  </ul>
+                </div>
+              </div>
+            <% end %>
             <% unless post.locked? %>
-              <% PostFlagType.where(post_type_id: post.post_type.id).or(PostFlagType.where(post_type_id: nil)).each do |reason| %>
+	            <% PostFlagType.where(post_type_id: [post.post_type.id, nil]).where(active: 1).each do |reason| %>
                 <div class="widget--body">
                   <div class="grid">
                     <div class="grid--cell">
@@ -330,22 +386,38 @@
                 </div>
               <% end %>
             <% end %>
-              <div class="widget--body">
-                <div class="grid">
-                  <div class="grid--cell">
-                    <input class="form-radio-element" type="radio" name="flag-reason" value="-1" id="flag-reason-other_<%= post.id  %>"
-                           data-requires-details="true" />
-                  </div>
-                  <div class="grid--cell is-flexible">
-                    <label class="form-element has-margin-0" for="flag-reason-other_<%= post.id  %>">
-                      other reason
-                      <span class="form-caption">
-                        Please elaborate in the details field below.
-                      </span>
-                    </label>
-                  </div>
+            <div class="widget--body">
+              <div class="grid">
+                <div class="grid--cell">
+                  <input class="form-radio-element" type="radio" name="flag-reason" value="-1" id="flag-reason-curator_<%= post.id  %>"
+                         data-requires-details="true" />
+                </div>
+                <div class="grid--cell is-flexible">
+                  <label class="form-element has-margin-0" for="flag-reason-curator_<%= post.id  %>">
+                    change post status <br/>
+                    <span class="form-caption">
+		      This post should be reopened/closed, locked/unlocked, or deleted/undeleted. Please elaborate in the details field below.
+                    </span>
+                  </label>
                 </div>
               </div>
+            </div>
+            <div class="widget--body">
+              <div class="grid">
+                <div class="grid--cell">
+                  <input class="form-radio-element" type="radio" name="flag-reason" value="-1" id="flag-reason-other_<%= post.id  %>"
+                         data-requires-details="true" />
+                </div>
+                <div class="grid--cell is-flexible">
+                  <label class="form-element has-margin-0" for="flag-reason-other_<%= post.id  %>">
+                    moderator attention <br/>
+                    <span class="form-caption">
+                      Please elaborate in the details field below. This flag will be seen only by moderators and administrators, not curators.
+                    </span>
+                  </label>
+                </div>
+              </div>
+            </div>
             <div class="widget--body">
               <label class="form-element" for="flag-post-<%= post.id %>">
                 Details?
@@ -355,7 +427,7 @@
             </div>
             <div class="widget--footer">
               <button class="flag-link button is-filled is-muted" data-post-type="<%= is_question ? 'Question' : 'Answer' %>" data-post-id="<%= post.id %>">
-                Flag for moderator attention
+                Flag for attention
               </button>
             </div>
           </div>
@@ -424,7 +496,7 @@
           </div>
         <% end %>
 
-        <% if is_top_level && post.children.length >= SiteSetting["TableOfContentsThreshold"] && SiteSetting["TableOfContentsThreshold"] != -1 %>
+        <% if is_top_level && post.children.undeleted.length >= SiteSetting["TableOfContentsThreshold"] && SiteSetting["TableOfContentsThreshold"] != -1 %>
           <div class="toc has-margin-left-4" id="toc-toggle">
             <button class="toc--header" data-toggle="#toc-toggle" data-toggle-property="class" data-toggle-value="is-active">Table of Contents</button>
             <% sorted_answers = post.children.sort_by { |answer| answer.score }.reverse! %>
@@ -452,30 +524,36 @@
           <% public_count = comment_threads.count %>
           <% available_count = current_user&.has_post_privilege?('flag_curate', post) ?
                                  post.comment_threads.count : post.comment_threads.publicly_available.count %>
+          <div class="post--comments-header">
           <h4 class="has-margin-0">
             <%= pluralize(public_count, 'comment thread') %>
           </h4>
           <% if user_signed_in? %>
-            <p class="has-font-size-caption">
-              <% if CommentThread.post_followed?(post, current_user) %>
-                <%= link_to follow_post_comments_path(post_id: post.id), method: :post,
-                            title: 'Don\'t follow new comment threads on this post' do %>
-                  <i class="fas fa-fw fa-minus"></i> unfollow new
-                <% end %>
-              <% else %>
-                <%= link_to follow_post_comments_path(post_id: post.id), method: :post,
-                            title: 'Follow all new comment threads on this post' do %>
-                  <i class="fas fa-fw fa-plus"></i> follow new
-                <% end %>
+            <% if CommentThread.post_followed?(post, current_user) %>
+              <%= link_to follow_post_comments_path(post_id: post.id), method: :post,
+                          class: "button is-muted is-outlined is-small",
+                          title: 'Don\'t follow new comment threads on this post',
+                          role: 'button',
+                          'aria-label': 'Unfollow new comment threads on this post' do %>
+                  <i class="fas fa-fw fa-bell-slash"></i> Unfollow new
               <% end %>
-            </p>
+            <% else %>
+              <%= link_to follow_post_comments_path(post_id: post.id), method: :post,
+                          class: "button is-muted is-outlined is-small",
+                          title: 'Follow all new comment threads on this post',
+                          role: 'button',
+                          'aria-label': 'Follow all new comment threads on this post' do %>
+                <i class="fas fa-fw fa-bell"></i> Follow new
+              <% end %>
+            <% end %>
           <% end %>
-          <div class="post--comments-container">
+          </div>
+          <div class="post--comments-container" role="list">
             <%= render 'comments/post', comment_threads: comment_threads.first(5) %>
           </div>
           <div class="post--comments-links has-margin-top-1">
             <% if available_count > [comment_threads.count, 5].min %>
-              <a href="#" class="js-more-comments button is-muted is-small" data-post-id="<%= post.id %>">
+              <a href="#" class="js-more-comments button is-muted is-small" data-post-id="<%= post.id %>" role="button" aria-label="Show more comment threads">
                 Show more
               </a>
             <% end %>
@@ -492,8 +570,10 @@
                 <% elsif post.comments_disabled %>
                   <p class="has-color-red-500"><i class="fa fa-lock"></i> Comments have been disabled on this post, but as a moderator you are exempt from that block.</p>
                 <% end %>
-                <a class="button is-muted is-small js-new-thread-link" data-post="<%= post.id %>">Start new comment thread</a>
               <% end %>
+                <a class="button is-outlined is-small js-new-thread-link" data-post="<%= post.id %>" role="button">
+                  <i class="fas fa-fw fa-plus"></i> Start new comment thread
+                </a>
                 <%= render 'comments/new_thread_modal', post: post %>
             <% end %>
           </div>
diff --git a/app/views/posts/_form.html.erb b/app/views/posts/_form.html.erb
index 97c7a2d9d029c41e2f82516c3a42f40e848e4492..6002284bd71e3d3dd7de04fca72a3497db1b3617 100644
--- a/app/views/posts/_form.html.erb
+++ b/app/views/posts/_form.html.erb
@@ -54,21 +54,46 @@
     </p>
   <% end %>
 
-  <%= render 'shared/body_field', f: f, field_name: :body_markdown, field_label: t('posts.body_label'), post: post %>
+  <%= render 'shared/body_field', f: f, field_name: :body_markdown, field_label: t('posts.body_label'), post: post,
+            cur_length: post.body_markdown&.length, min_length: min_body_length(category), max_length: max_body_length(category) %>
 
+  <div class="rejected-elements notice is-warning hide">
+    <h3>Unsupported HTML detected</h3>
+    <p>The following HTML tags and attributes are unsupported and will be removed from the final post:</p>
+    <ul>
+    </ul>
+    <p>For a list of allowed HTML, see <a href="/help/advanced-formatting">this help article</a>.
+    If you meant to display the tags as code in the post, please enclose them in a code block.</p>
+  </div>
   <div class="post-preview"></div>
 
   <% unless post_type.has_parent? %>
+    <% value = {} %>
+    <% key = "saved_post.#{current_user&.id}.#{request.path}.title" %>
+    <% saved_at_key = "saved_post_at.#{current_user&.id}.#{request.path}" %>
+    <% saved_at = DateTime.parse(RequestContext.redis.get(saved_at_key) || '') rescue Date.new(2000, 1, 1) %>
+    <%
+      # Find the most recent between post-create, post-update, and draft-saved, and use the value corresponding to that.
+      value = [
+        [post.created_at || Date.new(2000, 1, 1), {}],
+        [post.updated_at || Date.new(2000, 1, 1), {}],
+        [saved_at || Date.new(2001, 1, 1), { value: RequestContext.redis.get(key) }]
+      ].max_by do |x|
+        x[0]
+      end[1]
+    %>
     <div class="form-group">
       <%= f.label :title, t('posts.post_title_label'), class: 'form-element' %>
-      <%= f.text_field :title, class: 'form-element post_title', data: { character_count: ".js-character-count-post-title" } %>
-    </div>
-    <div>
-      <span class="has-float-right has-font-size-caption js-character-count-post-title hide"
-            data-max="255" data-min="15" data-display-at="0.75">
-        <i class="fas fa-ellipsis-h js-character-count__icon"></i>
-        <span class="js-character-count__count">0 / 255</span>
-      </span>
+      <%= f.text_field :title,
+                        **({ class: 'form-element post_title' }).merge(value),
+                        data: { character_count: ".js-character-count-post-title" }
+      %>
+      <div class="clearfix">
+        <%= render 'shared/char_count', type: 'post-title',
+                                        cur: value[:value]&.length || post.title&.length,
+                                        max: max_title_length(category),
+                                        min: min_title_length(category, post_type) %>
+      </div>
     </div>
   <% end %>
 
@@ -87,8 +112,26 @@
           <% end %>
         </span>
       <% end %>
-      <%= f.select :tags_cache, options_for_select(post.tags_cache.map { |t| [t, t] }, selected: post.tags_cache),
-                   { include_blank: true }, multiple: true, class: "form-element js-tag-select",
+      <% tags = post.tags_cache %>
+      <% key = "saved_post.#{current_user&.id}.#{request.path}.tags" %>
+      <% saved_at_key = "saved_post_at.#{current_user&.id}.#{request.path}" %>
+      <% saved_at = DateTime.parse(RequestContext.redis.get(saved_at_key) || '') rescue Date.new(2000, 1, 1) %>
+      <%
+        # Find the most recent between post-create, post-update, and draft-saved, and use the value corresponding to that.
+        tags = [
+          [post.created_at || Date.new(2000, 1, 1), post.tags_cache],
+          [post.updated_at || Date.new(2000, 1, 1), post.tags_cache],
+          [saved_at || Date.new(2001, 1, 1), RequestContext.redis.smembers(key)]
+        ].max_by do |x|
+          x[0]
+        end[1]
+      %>
+      <%= f.select :tags_cache, options_for_select(
+                                  (tags || post.tags_cache).map { |t| [t, t] },
+                                  selected: tags || post.tags_cache),
+                   {},
+                   multiple: true,
+                   class: "form-element js-tag-select",
                    data: { tag_set: category.tag_set_id } %>
     </div>
   <% end %>
@@ -101,7 +144,7 @@
         <% category_default = category.license %>
         <% user_default = user_preference('default_license', community: true) %>
         <% if site_default.present? %>
-          site default: <a href="javascript:void(0)" class="js-license-autofill" data-license-id="<%= site_default.id %>">
+          community default: <a href="javascript:void(0)" class="js-license-autofill" data-license-id="<%= site_default.id %>">
             <%= site_default.name %>
           </a>
         <% end %>
@@ -122,9 +165,23 @@
           </a>
         <% end %>
       </span>
+      <% license = post.license_id %>
+      <% key = "saved_post.#{current_user&.id}.#{request.path}.license" %>
+      <% saved_at_key = "saved_post_at.#{current_user&.id}.#{request.path}" %>
+      <% saved_at = DateTime.parse(RequestContext.redis.get(saved_at_key) || '') rescue Date.new(2000, 1, 1) %>
+      <%
+        # Find the most recent between post-create, post-update, and draft-saved, and use the value corresponding to that.
+        license = [
+          [post.created_at || Date.new(2000, 1, 1), post.license_id],
+          [post.updated_at || Date.new(2000, 1, 1), post.license_id],
+          [saved_at || Date.new(2001, 1, 1), RequestContext.redis.get(key)]
+        ].max_by do |x|
+          x[0]
+        end[1]
+      %>
       <%= f.select :license_id, options_for_select(License.enabled.default_order(category, user_default)
                                                           .map { |l| [l.name, l.id, { 'data-title': l.description }] },
-                                                   selected: post.license_id),
+                                                   selected: license || post.license_id),
                    { include_blank: user_default == 'No default (make me choose)' },
                    class: 'form-element js-license-select' %>
     </div>
@@ -171,11 +228,25 @@
   <% end %>
 
   <% if edit_comment %>
+    <% comment = '' %>
+    <% key = "saved_post.#{current_user&.id}.#{request.path}.comment" %>
+    <% saved_at_key = "saved_post_at.#{current_user&.id}.#{request.path}" %>
+    <% saved_at = DateTime.parse(RequestContext.redis.get(saved_at_key) || '') rescue Date.new(2000, 1, 1) %>
+    <%
+      # Find the most recent between post-create, post-update, and draft-saved, and use the value corresponding to that.
+      comment = [
+        [post.created_at || Date.new(2000, 1, 1), ''],
+        [post.updated_at || Date.new(2000, 1, 1), ''],
+        [saved_at || Date.new(2001, 1, 1), RequestContext.redis.get(key)]
+      ].max_by do |x|
+        x[0]
+      end[1]
+    %>
     <hr/>
-    <div class="form-group">
-      <%= label_tag :edit_comment, t('posts.edit_comment_label'), class: 'form-element' %>
-      <%= text_field_tag :edit_comment, nil, class: 'form-element' %>
-    </div>
+    <%= render 'edit_comment', comment: comment,
+                               cur_length: comment&.length || 0,
+                               max_length: max_edit_comment_length
+    %>
   <% end %>
 
   <div class="actions">
diff --git a/app/views/posts/_list.html.erb b/app/views/posts/_list.html.erb
index 9caf81526bd137ceff01370a6e1ff7fe6363b545..456e75360f367f8953d9d46dc6552bce4da24002 100644
--- a/app/views/posts/_list.html.erb
+++ b/app/views/posts/_list.html.erb
@@ -8,7 +8,7 @@
 
 <div class="item-list--item <%= post.deleted ? 'deleted-content' : '' %>" data-ckb-list-item data-ckb-item-type="link">
   <% if post.post_type.has_votes %>
-    <div class="item-list--number-value">
+    <div class="item-list--number-value" title="Score: <%= post.score %>">
       <div class="meter is-question-score">
         <div class="meter--bar is-<%= (post.score * 100).to_i %>%"><%= (post.score * 100).to_i %>%</div>
       </div>
@@ -28,7 +28,7 @@
     <div class="post-list--title">
       <%= link_to generic_share_link(post), 'data-ckb-item-link' => '' do %>
         <% if @show_category_tag && post.post_type.has_category %>
-          <span class="badge is-tag is-filled"><%= defined?(@category) ? @category.name : post.category.name %></span>
+          <span class="badge is-tag is-filled"><%= post.category.name %></span>
         <% end %>
         <%= post.post_type.is_top_level ? post.title : post.parent.title %>
         <%= post.post_type.is_closeable && post.closed && !post.duplicate_post  ? "[closed]" : "" %>
@@ -37,7 +37,7 @@
     </div>
     <% if (SiteSetting['PostBodyListTruncateLength'] || 0) > 0 %>
       <p class="post-list--content">
-        <%= strip_tags(post.body).truncate(SiteSetting['PostBodyListTruncateLength'] || 200) %>
+        <%= sanitize(strip_tags(post.body).truncate(SiteSetting['PostBodyListTruncateLength'] || 200), scrubber: scrubber) %>
       </p>
     <% end %>
     <p class="has-color-tertiary-600 has-float-right post-list--meta">
diff --git a/app/views/posts/_mdhint.html.erb b/app/views/posts/_mdhint.html.erb
index 5f9536ca79aebbecf58802faf930faea85fc9add..27c264ddd7fa059f73b2613a8b3fc516966d73f9 100644
--- a/app/views/posts/_mdhint.html.erb
+++ b/app/views/posts/_mdhint.html.erb
@@ -1,9 +1,25 @@
-<div class="form-caption widget--footer js-post-field-footer">
-  We <a href="/help/formatting">support Markdown</a> for posts:
-  <strong>**bold**</strong>, <em>*italics*</em>, <code>`code`</code>, two newlines for paragraphs
-  <span class="has-float-right has-font-size-caption js-character-count-post-body hide"
-        data-max="30000" data-min="30" data-display-at="0.75">
-    <i class="fas fa-ellipsis-h js-character-count__icon"></i>
-    <span class="js-character-count__count">0 / 30000</span>
-  </span>
+<%#
+  Widget displaying that we support markdown, linking to the help article. This also applies the min/max length for the
+  post body.
+
+  Variables:
+    cur_length    : [Integer, Nil] optional, current character length (default 0)
+    min_length    : [Integer, Nil] optional, the minimum allowed length (default 30)
+    max_length    : [Integer, Nil] optional, the maximum allowed length (default 30_000)
+%>
+
+<%
+  # Defaults
+  cur_length = (defined?(cur_length) ? cur_length : nil) || 0
+  min_length = (defined?(min_length) ? min_length : nil) || 30
+  max_length = (defined?(max_length) ? max_length : nil) || 30_000
+%>
+
+<div class="mdhint form-caption widget--footer js-post-field-footer">
+  <span>Markdown <a href="/help/formatting">support</a> for posts:
+  <strong>**bold**</strong>, <em>*italics*</em>, <code>`code`</code>, 2 newlines for paragraphs</span>
+  <span class="draft-status transparent has-color-green-600 js-post-draft-status">draft saved</span>
+  <div class="clearfix">
+    <%= render 'shared/char_count', type: 'post-body', cur: cur_length, min: min_length, max: max_length %>
+  </div>
 </div>
diff --git a/app/views/posts/document.html.erb b/app/views/posts/document.html.erb
index a5cb84a167ba3c82b7ae75a9110be610f0b9a887..e08d748ab463574d8e861492bfc8e7cdcdf351a0 100644
--- a/app/views/posts/document.html.erb
+++ b/app/views/posts/document.html.erb
@@ -2,9 +2,15 @@
   &laquo; Back to help center
 <% end %>
 <% unless @post.nil? %>
-  <% if (moderator? && @post.post_type_id == HelpDoc.post_type_id) || (admin? && @post.post_type_id == PolicyDoc.post_type_id) %>
+  <% 
+    is_hc = @post.post_type_id == HelpDoc.post_type_id
+    is_policy = @post.post_type_id == PolicyDoc.post_type_id
+    history_path = is_hc ? help_post_history_path(@post.doc_slug) : policy_post_history_path(@post.doc_slug)
+  %>
+  <% if (moderator? && is_hc) || (admin? && is_policy) %>
     <%= link_to 'edit', edit_post_path(@post), class: "button is-outlined is-muted" %>
   <% end %>
+  <%= link_to 'history', history_path,  class: "button is-outlined is-muted" %>
 <% end %>
 
 <% if @post.help_category == '$Moderator' %>
diff --git a/app/views/posts/help_center.html.erb b/app/views/posts/help_center.html.erb
index 49fa62614adc74229c816fce8008a4f32f85b887..35dce5d6c342952263d7deb8164b0ff079185c64 100644
--- a/app/views/posts/help_center.html.erb
+++ b/app/views/posts/help_center.html.erb
@@ -1,7 +1,7 @@
 <% content_for :title, 'Help Center' %>
 
 <h1>Help Center</h1>
-
+<p>Community moderators can edit, reorganize, and create topics in the Help section. Topics in the Policy section are managed by the network administrators.</p>
 <div class="grid">
   <div class="grid--cell is-6 is-12-sm">
     <h2>Help</h2>
diff --git a/app/views/posts/new.html.erb b/app/views/posts/new.html.erb
index 94baae739b457f7282cd1776dd372ff673cadba2..b81569e90dcdea32820b7c30d8422dabdaecc981 100644
--- a/app/views/posts/new.html.erb
+++ b/app/views/posts/new.html.erb
@@ -34,4 +34,6 @@
   <div class="notice is-danger is-filled">
       You don't have a high enough trust level to post in the <%= @category.name %> category.
   </div>
+  <p>Not where you meant to post? See <%= link_to 'Categories', categories_path %></p>
+
 <% end %>
diff --git a/app/views/posts/show.html.erb b/app/views/posts/show.html.erb
index c25a17c035b2386a55a7f5e3c69879f860bdb56c..5846533fdb642ece56b39df260ce5914b0f2c897 100644
--- a/app/views/posts/show.html.erb
+++ b/app/views/posts/show.html.erb
@@ -1,4 +1,4 @@
-<% content_for :title, @post.title.truncate(50) %>
+<% content_for :title, @post.title.truncate(max_title_length(@post.category)) %>
 <% content_for :description do %>
   <% Rails.cache.fetch "posts/#{@post.id}/description" do %>
     <%= @post.body_plain[0..74].strip %>...
@@ -23,8 +23,15 @@
   <h2><%= pluralize(@post.children.where(deleted: false).count, 'answer') %></h2>
 
   <div class="button-list is-gutterless has-float-right">
-    <a href="<%= query_url(sort: 'score') %>" class="button is-muted is-outlined <%= params[:sort].nil? || params[:sort] == 'score' ? 'is-active' : '' %>">Score</a>
-    <a href="<%= query_url(sort: 'active') %>" class="button is-muted is-outlined <%= params[:sort] == 'active' ? 'is-active' : '' %>">Active</a>
+    <%= link_to 'Score', request.params.merge(sort: 'score'),
+        class: "button is-muted is-outlined #{params[:sort].nil? || params[:sort] == 'score' ? 'is-active' : ''}",
+	title: 'highest score first (not the same as net votes)' %>
+    <%= link_to 'Active', request.params.merge(sort: 'active'),
+        class: "button is-muted is-outlined #{params[:sort] == 'active' ? 'is-active' : ''}",
+	title: 'most recent changes first: new answers, edits, delete/undelete' %>
+    <%= link_to 'Age', request.params.merge(sort: 'age'),
+	class: "button is-muted is-outlined #{params[:sort] == 'age' ? 'is-active' : ''}",
+	title: 'newest posts first (ignores other activity)' %>
   </div>
 
   <div class="has-clear-clear"></div>
diff --git a/app/views/questions/_list.html.erb b/app/views/questions/_list.html.erb
index 7e387ee6d6220ddb345b0b0faaed1314ba684922..046ce39bc08a07abfa7ad6a4666d3af8301a1488 100644
--- a/app/views/questions/_list.html.erb
+++ b/app/views/questions/_list.html.erb
@@ -1,10 +1,14 @@
 <div class="button-list is-gutterless has-float-right has-margin-2">
   <%= link_to 'Activity', query_url(root_url, sort: 'activity'),
               class: "button is-muted is-outlined #{(params[:sort].nil?) && !current_page?(questions_lottery_path) || params[:sort] == 'activity' ?
-                                      'is-active' : ''}" %>
-  <%= link_to 'Age', query_url(root_url, sort: 'age'), class: "button is-muted is-outlined #{params[:sort] == 'age' ? 'is-active' : ''}" %>
-  <%= link_to 'Score', query_url(root_url, sort: 'score'), class: "button is-muted is-outlined #{params[:sort] == 'score' ? 'is-active' : ''}" %>
-  <%= link_to 'Random', questions_lottery_path, class: "button is-muted is-outlined #{current_page?(questions_lottery_path) ? 'is-active' : ''}" %>
+                                      'is-active' : ''}",
+              role: 'button', 'aria-label': 'Sort by activity' %>
+  <%= link_to 'Age', query_url(root_url, sort: 'age'), class: "button is-muted is-outlined #{params[:sort] == 'age' ? 'is-active' : ''}",
+              role: 'button', 'aria-label': 'Sort by age' %>
+  <%= link_to 'Score', query_url(root_url, sort: 'score'), class: "button is-muted is-outlined #{params[:sort] == 'score' ? 'is-active' : ''}",
+              role: 'button', 'aria-label': 'Sort by score' %>
+  <%= link_to 'Random', questions_lottery_path, class: "button is-muted is-outlined #{current_page?(questions_lottery_path) ? 'is-active' : ''}",
+              role: 'button', 'aria-label': 'Order randomly' %>
 </div>
 <div class="has-clear-clear">&nbsp;</div>
 
diff --git a/app/views/reactions/_dialog.html.erb b/app/views/reactions/_dialog.html.erb
index d6e75d5330a736a99849e56777f27506ee4e0dc7..64bedbde1a3754ce98d080cd45d1f5f280954e29 100644
--- a/app/views/reactions/_dialog.html.erb
+++ b/app/views/reactions/_dialog.html.erb
@@ -37,6 +37,7 @@
   <% end %>
   <label for="reaction-type-comment-<%= post.id %>">Explain in comment?</label>
   <textarea class="form-element reaction-comment-field is-small"
+            placeholder="Enter comment (minimum 15 characters)"
             id="reaction-type-comment-<%= post.id %>"></textarea>
   <button type="button" class="button is-filled reaction-submit" data-post-id="<%= post.id %>">
     Add reaction
diff --git a/app/views/reactions/_form.html.erb b/app/views/reactions/_form.html.erb
index ee8d82cf05cc14cd77590c3fca69589c1dba5c10..2341ba15e067b71f1aba61b74724039607b3aba7 100644
--- a/app/views/reactions/_form.html.erb
+++ b/app/views/reactions/_form.html.erb
@@ -84,5 +84,5 @@
   </div>
 
   <%= f.submit 'Save', class: 'button is-filled' %>
-  <%= link_to 'Cancel', reactions_path, class: 'button' %>
+  <%= link_to 'Cancel', reactions_path, class: 'button', role: 'button' %>
 <% end %>
\ No newline at end of file
diff --git a/app/views/reactions/index.html.erb b/app/views/reactions/index.html.erb
index b0fd6a4ee5c74148f1e60e2e70561dc6fb09ca95..bef08dc66967fccd8dbce2779a0467ebaceb6f70 100644
--- a/app/views/reactions/index.html.erb
+++ b/app/views/reactions/index.html.erb
@@ -37,7 +37,7 @@
         </td>
         <td>
             <div class="h-d-flex">
-                <%= link_to "edit", edit_reaction_path(rt.id), class: "button is-outlined" %>
+                <%= link_to "edit", edit_reaction_path(rt.id), class: "button is-outlined", 'aria-label': "Edit #{rt.name}" %>
             </div>
         </td>
     </tr>
diff --git a/app/views/search/_filters.html.erb b/app/views/search/_filters.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..a282f1303a88503b73c1ecee6b81c9387a5d451f
--- /dev/null
+++ b/app/views/search/_filters.html.erb
@@ -0,0 +1,78 @@
+<% allow_delete ||= false %>
+<% allow_apply = true if allow_apply.nil? %>
+<% @active_filter ||= {} %>
+
+<div class="form-group">
+  <div class="form-group-horizontal">
+    <div class="form-group">
+      <%= label_tag :predefined_filter, 'Predefined Filters', class: "form-element" %>
+      <%= select_tag :predefined_filter, options_for_select([@active_filter[:name]],
+                                                            selected: @active_filter[:name]),
+          include_blank: true, class: "form-element js-filter-select", id: nil,
+          data: { placeholder: "" } %>
+    </div>
+    <button type="button" class="filter-clear button is-medium is-danger">Clear</button>
+    <% if allow_apply %>
+      <%= submit_tag 'Apply', class: 'button is-medium is-outlined', name: nil %>
+    <% end %>
+    <% if user_signed_in? %>
+      <button type="button" class="filter-save button is-medium is-filled">Save</button>
+    <% end %>
+    <% if allow_delete %>
+      <button type="button" class="filter-delete button is-medium is-filled is-danger has-margin-left-2">Delete</button>
+    <% end %>
+  </div>
+  <% if user_signed_in? && defined? @category %>
+    <%= label_tag :save_as_default, 'Is default for this category?' %>
+    <%= check_box_tag :save_as_default, @category.id, false, { class: 'filter-is-default form-checkbox-element' } %>
+  <% end %>
+  <div class="form-group-horizontal">
+    <div class="form-group">
+      <%= label_tag :min_score, 'Min Score (0-1)', class: "form-element" %>
+      <%= number_field_tag :min_score, @active_filter[:min_score],
+          min: 0, max: 1, step: 0.01, class: 'form-element form--filter',
+          data: { name: 'min_score' } %>
+    </div>
+    <div class="form-group">
+      <%= label_tag :max_score, 'Max Score (0-1)', class: "form-element" %>
+      <%= number_field_tag :max_score, @active_filter[:max_score],
+          min: 0, max: 1, step: 0.01, class: 'form-element form--filter',
+          data: { name: 'max_score' } %>
+    </div>
+    <div class="form-group">
+      <%= label_tag :min_answers, 'Min Answers', class: "form-element" %>
+      <%= number_field_tag :min_answers, @active_filter[:min_answers],
+          min: 0, step: 1, class: 'form-element form--filter',
+          data: { name: 'min_answers' } %>
+    </div>
+    <div class="form-group">
+      <%= label_tag :max_answers, 'Max Answers', class: "form-element" %>
+      <%= number_field_tag :max_answers, @active_filter[:max_answers],
+          min: 0, step: 1, class: 'form-element form--filter',
+          data: { name: 'max_answers' } %>
+    </div>
+    <div class="form-group">
+      <%= label_tag :status, 'Status', class: "form-element" %>
+      <%= select_tag :status, options_for_select(['any', 'open', 'closed'], selected: @active_filter[:status] || 'any'),
+        min: 0, step: 1, class: 'form-element form--filter',
+        data: { name: 'status' } %>
+    </div>
+  </div>
+  <div>
+    <a href="/help/scoring">How scores are computed</a>
+  </div>
+  <div class="form-group-horizontal">
+    <div class="form-group">
+      <%= label_tag :include_tags, 'Include Tags', class: "form-element" %>
+      <%= select_tag :include_tags, options_for_select(Tag.where(id: @active_filter[:include_tags]).map { |tag| [tag.name, tag.id] },
+                   selected: @active_filter[:include_tags] || []), multiple: true, class: 'form-element form--filter js-tag-select',
+                   data: { name: 'include_tags', tag_set: @category&.tag_set_id, create: 'false', use_ids: 'true', placeholder: '' } %>
+    </div>
+    <div class="form-group">
+      <%= label_tag :exclude_tags, 'Exclude Tags', class: "form-element" %>
+      <%= select_tag :exclude_tags, options_for_select(Tag.where(id: @active_filter[:exclude_tags]).map { |tag| [tag.name, tag.id] },
+                   selected: @active_filter[:exclude_tags]|| []), multiple: true, class: 'form-element form--filter js-tag-select',
+                   data: { name: 'exclude_tags', tag_set: @category&.tag_set_id, create: 'false', use_ids: 'true', placeholder: '' } %>
+    </div>
+  </div>
+</div>
diff --git a/app/views/search/_widget.html.erb b/app/views/search/_widget.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..0c7231a7ca0c22433467a35ca976c2dd30748aab
--- /dev/null
+++ b/app/views/search/_widget.html.erb
@@ -0,0 +1,39 @@
+<%= form_tag search_path, method: :get, role: 'search' do %>
+  <div class="form-group-horizontal">
+    <div class="form-group">
+      <%= label_tag :search, 'Search term', class: "form-element" %>
+      <%= search_field_tag :search, params[:search], class: 'form-element' %>
+    </div>
+    <div class="actions has-padding-bottom-1">
+      <%= submit_tag 'Search', class: 'button is-medium is-outlined', name: nil %>
+    </div>
+  </div>
+
+  <% if @signed_out_me %>
+  <div class="widget is-red">
+    <div class="widget--body">
+      <p>
+        Search results for <code>user:me</code> can only be displayed when signed in. 
+        Please <%= sign_in_link 'sign in' %> and try again.
+      </p>
+    </div>
+  </div>
+  <% end %>
+
+  <div class="widget is-teal">
+    <div class="widget--header">
+      Advanced Search Options
+    </div>
+    <div class="widget--body">
+      <%= raw(sanitize(render_markdown(SiteSetting['JITAdvancedSearchHelp']), scrubber: scrubber)) %>
+      <p>Further help with searching is available <%= link_to 'in the help center', help_path('search') %>.</p>
+      <p>Quick hints: <code>tag:tagname</code>, <code>user:xxx</code>, <code>"exact phrase"</code>, <code>post_type:xxx</code>, <code>created:&lt;N{d,w,mo,y}</code>, <code>score:&gt;=0.5</code></p>
+    </div>
+  </div>
+
+  <details>
+    <summary>Filters</summary>
+    <%= render 'filters' %>
+  </details>
+
+<% end %>
diff --git a/app/views/search/search.html.erb b/app/views/search/search.html.erb
index 3e91baa9c16c080a33dca177d3d46f14df4e78a5..5e4794cdb2f23e054a314b4dcab1c18f896ee337 100644
--- a/app/views/search/search.html.erb
+++ b/app/views/search/search.html.erb
@@ -2,35 +2,25 @@
 
 <h1>Search</h1>
 
-<%= form_tag search_path, method: :get do %>
-  <div class="form-group-horizontal">
-    <div class="form-group">
-      <%= label_tag :search, 'Search term', class: "form-element" %>
-      <%= text_field_tag :search, params[:search], class: 'form-element' %>
-    </div>
-    <div class="actions has-padding-bottom-1">
-      <%= submit_tag 'Search', class: 'button is-medium is-outlined', name: nil %>
-    </div>
-  </div>
-<% end %>
-
-<div class="widget is-teal">
-  <div class="widget--header">
-    Advanced Search Options
-  </div>
-  <div class="widget--body">
-    <%= raw(sanitize(render_markdown(SiteSetting['JITAdvancedSearchHelp']), scrubber: scrubber)) %>
-    <p>Further help with searching is available <%= link_to 'in the help center', help_path('search') %>.</p>
-  </div>
-</div>
+<%= render 'widget' %>
 
 <% unless @posts.nil? %>
-  <div class="button-list is-gutterless has-float-right">
-    <%= link_to 'Relevance', query_url(sort: 'relevance'),
-                class: "button is-outlined is-muted #{params[:sort] == 'relevance' || params[:sort].nil? ? 'is-active' : ''}" %>
-    <%= link_to 'Score', query_url(sort: 'score'), class: "button is-outlined is-muted #{params[:sort] == 'score' ? 'is-active' : ''}" %>
-    <%= link_to 'Age', query_url(sort: 'age'), class: "button is-outlined is-muted #{params[:sort] == 'age' ? 'is-active' : ''}" %>
+  <% post_count = @posts.count %>
+  <div class="has-color-tertiary-500 flex-row jc-sb ai-c">
+    <span  title="<%= post_count %> posts">
+     <%= short_number_to_human post_count, precision: 1, significant: false %>
+     <%= 'post'.pluralize(post_count) %>
+    </span>
+    <div class="button-list is-gutterless has-margin-2">
+      <%= link_to 'Relevance', request.params.merge(sort: 'relevance'), class: "button is-outlined is-muted #{params[:sort] == 'relevance' || params[:sort].nil? ? 'is-active' : ''}",
+                  role: 'button', 'aria-label': 'Sort by relevance' %>
+      <%= link_to 'Score', request.params.merge(sort: 'score'), class: "button is-outlined is-muted #{params[:sort] == 'score' ? 'is-active' : ''}",
+                  role: 'button', 'aria-label': 'Sort by score' %>
+      <%= link_to 'Age', request.params.merge(sort: 'age'), class: "button is-outlined is-muted #{params[:sort] == 'age' ? 'is-active' : ''}",
+                  role: 'button', 'aria-label': 'Sort by age' %>
+    </div>
   </div>
+
   <div class="has-clear-clear">&nbsp;</div>
 
   <% if @posts.size == 0 %>
diff --git a/app/views/shared/_body_field.html.erb b/app/views/shared/_body_field.html.erb
index 8812a680461f1ee3602163c17f488eae035a9414..2552d739da9f3df44bb9694f50eb66f29b8a93d1 100644
--- a/app/views/shared/_body_field.html.erb
+++ b/app/views/shared/_body_field.html.erb
@@ -1,3 +1,20 @@
+<%#
+  Adds a markdown body form textarea.
+  Variables:
+    post          : [ApplicationRecord, Nil] the entity to which this body field belongs (a post, a tag, a user, ...)
+    field_name    : [Symbol] the name of the body field (for the given entity)
+    field_label   : [String] the label for the field
+    min_length    : [Integer, Nil] optional, the minimum allowed length
+    max_length    : [Integer, Nil] optional, the maximum allowed length
+%>
+
+<%
+  # Defaults
+  cur_length = defined?(cur_length) ? cur_length : nil
+  min_length = defined?(min_length) ? min_length : nil
+  max_length = defined?(max_length) ? max_length : nil
+%>
+
 <div class="form-group">
   <%= f.label field_name, field_label, class: "form-element" %>
   <% if block_given? %>
@@ -22,7 +39,9 @@
     <%= render 'shared/markdown_tools' %>
     <%= f.text_area field_name, **({ class: classes, rows: 15, placeholder: 'Start typing your post...' }).merge(value), 
                                 data: { character_count: ".js-character-count-post-body" } %>
-    <%= render 'posts/mdhint' %>
+    <%= render 'posts/mdhint', cur_length: value[:value]&.length || cur_length, 
+                               min_length: min_length, 
+                               max_length: max_length %>
   </div>
   <%= hidden_field_tag "__html", nil, class: 'js-post-html' %>
 </div>
diff --git a/app/views/shared/_char_count.html.erb b/app/views/shared/_char_count.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..985dec3f89c1d5d122deda6e8c7963def5dfd5cb
--- /dev/null
+++ b/app/views/shared/_char_count.html.erb
@@ -0,0 +1,28 @@
+<%# 
+  Reusable helper view for character count requirements.
+
+  Variables:
+    cur : current number of characters (default 0)
+    max : maximum number of characters allowed (default 255)
+    min : minimum number of characters allowed (default 0)
+    threshold : fraction of max to show the count at (default 0.75)
+    type : character count type (e.g.: post-title)
+%>
+
+<%
+  # defaults & normalization
+  cur ||= defined?(cur) && !cur.nil? ? cur.to_i : 0
+  max ||= defined?(max) && !max.nil? ? max.to_i : 255
+  min ||= defined?(min) && !min.nil? ? min.to_i : 0
+  threshold ||= defined?(threshold) && !threshold.nil? ? threshold.to_f : 0.75
+%>
+
+<span class="has-float-right has-font-size-caption js-character-count-<%= type %>"
+  data-max="<%= max %>"
+  data-min="<%= min %>"
+  data-threshold="<%= threshold %>">
+    <i class="fas fa-ellipsis-h js-character-count__icon"></i>
+    <span class="js-character-count__count">
+      <%= cur %> / <%= cur < min ? min : max %>
+    </span>
+</span>
\ No newline at end of file
diff --git a/app/views/shared/_copy_link.html.erb b/app/views/shared/_copy_link.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..6932d72ab19edda7576b241544e5d7742607ebc0
--- /dev/null
+++ b/app/views/shared/_copy_link.html.erb
@@ -0,0 +1,66 @@
+<%#
+  Adds a copy link button and drop panel.
+  Variables:
+    classes : a CSS string representing class names for the trigger button
+    desc  : short description of what clicking on the button does
+    id    : id of the entity to link to
+    md    : Markdown-formatted link to copy
+    raw   : raw URL of the link to copy
+%>
+
+<%
+  # Defaults
+  classes = defined?(classes) ? classes : []
+%>
+
+<noscript>
+<%= link_to raw, 'aria-label': desc, class: classes, role: "button" do %>
+  <i class="fa fa-link"></i>
+  <span>Copy Link</span>
+<% end %>
+</noscript>
+<%= tag.button 'aria-label': desc,
+               class: classes + ["js-permalink-trigger"],
+               data: { 
+                       drop: "#permalink-#{id}", 
+                       drop_force_dir: "down", 
+                       drop_self_class_toggle: "is-active" 
+                     },
+               hidden: true,
+               type: "button" do %>
+  <i class="fa fa-link"></i>
+  <span>Copy Link</span>
+<% end %>
+<div id="permalink-<%= id %>" class="droppanel has-padding-top-0 has-padding-right-1">
+  <label for="permalink-<%= id %>-raw" class="form-element">Raw</label>
+  <div class="flex-row-always">
+    <input class="form-element" 
+           id="permalink-<%= id %>-raw" 
+           value="<%= raw %>"
+           >
+    <button class="button is-icon-only-button js-permalink-copy" 
+            data-post-id="<%= id %>" 
+            data-link-type="raw" 
+            title="Copy raw URL" 
+            type="button"
+            >
+      <i class="fa fa-copy"></i>
+    </button>
+  </div>
+  <label for="permalink-<%= id %>-raw" class="form-element">Markdown</label>
+  <div class="flex-row-always">
+    <input class="form-element" 
+           id="permalink-<%= id %>-md" 
+           value="<%= md %>"
+           >
+    <button class="button is-icon-only-button js-permalink-copy" 
+            data-post-id="<%= id %>" 
+            data-link-type="md" 
+            title="Copy Markdown Link" 
+            type="button"
+            >
+      <i class="fa fa-copy"></i>
+    </button>
+  </div>
+</div>
+<a class="js-permalink" href="<%= raw %>" hidden aria-hidden></a>
\ No newline at end of file
diff --git a/app/views/shared/_markdown_tools.html.erb b/app/views/shared/_markdown_tools.html.erb
index 1bc846dca2279e4709568cb20ef4c290d32a251c..daa3ea3d39d5df2ba3cab2c0fa91ca1d395944da 100644
--- a/app/views/shared/_markdown_tools.html.erb
+++ b/app/views/shared/_markdown_tools.html.erb
@@ -32,6 +32,11 @@
     <%= md_button action: 'code', label: 'Monospace font', class: 'is-icon-only' do %>
       <i class="fas fa-fw fa-code"></i>
     <% end %>
+    <% if SiteSetting['MathJaxEnabled'] %>
+      <%= md_button action: 'mathjax', label: 'MathJax', class: 'is-icon-only' do %>
+        $
+      <% end %>
+    <% end %>
   </div>
 
   <div class="button-list is-gutterless">
@@ -48,7 +53,7 @@
 
   <div class="button-list is-gutterless">
     <a href="javascript:void(0)" class="button is-outlined is-muted is-icon-only" data-drop="#markdown-tools"
-       data-drop-self-class-toggle="is-active" aria-label="Tools" title="Tools">
+       data-drop-self-class-toggle="is-active" aria-label="Tools" title="Tools" role="button">
       <i class="fas fa-tools"></i>
     </a>
 
@@ -75,7 +80,7 @@
 
   <div class="button-list is-gutterless">
     <a href="javascript:void(0)" class="button is-muted is-outlined is-icon-only js-save-draft" aria-label="Save draft"
-       title="Save draft">
+       title="Save draft" role="button">
       <i class="fas fa-save"></i>
     </a>
   </div>
diff --git a/app/views/shared/_rss_link.html.erb b/app/views/shared/_rss_link.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..c3f144f9b69386a7f00bbc53bc8dc1719ba43773
--- /dev/null
+++ b/app/views/shared/_rss_link.html.erb
@@ -0,0 +1,20 @@
+<%# 
+   'Adds an RSS link.
+    Variables:
+      url     : [String] URL to the feed
+      text    : [String, Nil] text to visibly show
+      tooltip : [String, Nil] text to show in the tooltip
+   '
+ %>
+
+<%
+  # Defaults
+  text = defined?(text) ? text : 'RSS'
+  tooltip = defined?(tooltip) ? tooltip : 'RSS Feed'
+%>
+
+<span title="<%= tooltip %>">
+  <%= link_to url, class: 'has-display-inline-block' do %>
+    <i class="fas fa-rss"></i> RSS
+  <% end %>
+</span>
\ No newline at end of file
diff --git a/app/views/site_settings/index.html.erb b/app/views/site_settings/index.html.erb
index 2d5117582e6bda358dcd3af8255eb1f93a84b498..9562d314c657dad6f03b8539cdc4069dd0e2ebc4 100644
--- a/app/views/site_settings/index.html.erb
+++ b/app/views/site_settings/index.html.erb
@@ -17,15 +17,13 @@
         <% settings.each do |setting| %>
           <tr>
             <td>
-              <h4>
-                <%= setting.name %>
-                <% if setting.community_id.nil? %>
-                  <span class="badge is-tag is-master-tag">global</span>
-                <% else %>
-                  <span class="badge is-tag">site</span>
-                <% end %>
-                <span class="badge is-tag"><%= setting.value_type %></span>
-              </h4>
+              <% if setting.global? %>
+                <span class="badge is-tag is-master-tag">global</span>
+              <% else %>
+                <span class="badge is-tag">site</span>
+              <% end %>
+              <span class="badge is-tag"><%= setting.value_type %></span>
+              <h4><%= setting.name %></h4>
               <div class="form-caption"><%= setting.description %></div>
             </td>
             <td class="site-setting--value js-setting-value" data-type="<%= setting.value_type %>" data-name="<%= setting.name %>"
diff --git a/app/views/subscription_mailer/subscription.html.erb b/app/views/subscription_mailer/subscription.html.erb
index 2bc7213fb4b42b1ee786d7f2ca4a9056237c8287..e1724b5d0484e59c48eb7fad7dc9ff3495e8d5b3 100644
--- a/app/views/subscription_mailer/subscription.html.erb
+++ b/app/views/subscription_mailer/subscription.html.erb
@@ -19,7 +19,7 @@
     <%= question.body.length > 150 ? '...' : '' %>
   </p>
   <p>
-    &mdash; <%= link_to question.user.rtl_safe_username, user_url(question.user, host: @subscription.community.host), dir: 'ltr'%>
+    &mdash; <%= user_link question.user, { host: @subscription.community.host }  %>
     <%= time_ago_in_words(question.created_at) %> ago
   </p>
 <% end %>
@@ -29,4 +29,4 @@
 <br/>
 
 <small><%= link_to 'Change your email settings or unsubscribe here',
-                   subscriptions_url(host: @subscription.community.host) %></small>
\ No newline at end of file
+                   subscriptions_url(host: @subscription.community.host) %></small>
diff --git a/app/views/subscriptions/index.html.erb b/app/views/subscriptions/index.html.erb
index 774017ba6f1ef0e5093ffffaeacda6d85aa043c1..8c8561f2a074e29cb09f6e8a2ff0dbb8460766d5 100644
--- a/app/views/subscriptions/index.html.erb
+++ b/app/views/subscriptions/index.html.erb
@@ -3,10 +3,19 @@
 <h1>Your Subscriptions</h1>
 <p>These are all the email subscriptions you're currently signed up for. You can turn them off or remove them entirely from here.</p>
 
-<% @subscriptions.each do |sub| %>
-  <details data-sub-id="<%= sub.id %>">
-    <summary><%= sub.name %></summary>
-    <p>Subscription to <%= phrase_for sub.type, sub.qualifier %>, emailed every <%= pluralize(sub.frequency, 'day') %>.</p>
+<% @subscriptions.group_by(&:type).flat_map do |type, subs|
+  subs.each_with_index.map do |sub, index|
+    [
+      sub.created_at, 
+      [sub.name.present? ? sub.name : "#{type.capitalize} subscription #{index + 1}",sub]
+    ] 
+  end
+end.sort_by { |a| a }.map { |_, v| v }.each do |name, sub| %>
+  <details data-sub-id="<%= sub %>">
+    <summary><%= name %></summary>
+    <p>Subscription to <%= phrase_for sub.type, sub.qualifier %>, emailed every <%= 
+      pluralize(sub.frequency, 'day') 
+    %>.</p>
     <%= check_box_tag :enabled, 1, sub.enabled, class: 'js-enable-subscription' %>
     <%= label_tag :enabled, 'Enabled?' %> &middot;
     <a href="#" class="js-remove-subscription text-danger">Remove</a>
diff --git a/app/views/suggested_edit/category_index.html.erb b/app/views/suggested_edit/category_index.html.erb
index 462cae1f14043000546cdc9a92dcd6b01fb729b0..f9c6b126c9cfde6a7ad9ece23ec9d896ee2e5c1c 100644
--- a/app/views/suggested_edit/category_index.html.erb
+++ b/app/views/suggested_edit/category_index.html.erb
@@ -1,5 +1,8 @@
 <h1>Suggested Edits</h1>
-<p class="is-lead">This is a list of suggested edits on posts in this category.</p>
+<p class="is-lead">Suggested edits for review.</p>
+
+<% categories = Category.unscoped.where(community: @community).order(sequence: :asc, id: :asc) %>
+<% current_cat = current_category %>
 
 <div class="button-list is-gutterless">
   <%= link_to 'Pending', query_url(show_decided: 0),
@@ -10,11 +13,46 @@
 
 <% if params[:show_decided] != '1' && !current_user&.privilege?('edit_posts') %>
   <div class="notice is-info has-color-teal-800">
-    <i class="fas fa-info-circle"></i> You can't approve or reject suggested edits because you haven't
-    yet earned the <%= link_to 'Edit Posts', ability_path('edit_posts') %> ability.
+    <i class="fas fa-info-circle"></i> You can't approve or reject suggested edits (except on your own posts) 
+    because you haven't yet earned the <%= link_to 'Edit Posts', ability_path('edit_posts') %> ability.
   </div>
 <% end %>
 
+<% if params[:show_decided] != '1' %>
+  <h3>All categories</h3>
+  <table>
+    <tr>
+      <th>category</th>
+      <th>pending edits</th>
+    </tr>
+  <% categories.each do |cat| %>
+      <% next if (cat.min_view_trust_level || -1) > (current_user&.trust_level || 0) %>
+      <% sug_edits = SuggestedEdit.where(post: Post.undeleted.where(category: cat), active: true).count %>
+    <tr>
+      <td>
+      <% if cat == current_cat %>
+        <%= cat.name %>
+      <% else %>
+        <%= link_to suggested_edits_queue_url(cat) do %>
+        <%= cat.name %>
+	<% end %>
+      <% end %>
+      </td>
+      <td>
+      <% if cat == current_cat %>
+        <%= sug_edits %>
+      <% else %>
+        <%= link_to suggested_edits_queue_url(cat) do %>
+        <%= sug_edits %>
+	<% end %>
+      <% end %>
+      </td>
+    </tr>
+    <% end %>
+  </table>
+<% end %>
+
+
 <% if @edits.any? %>
   <div class="widget">
     <% @edits.each do |edit| %>
@@ -35,4 +73,4 @@
   </div>
 <% else %>
 <p>There are no suggested edits in this category.</p>
-<% end %>
\ No newline at end of file
+<% end %>
diff --git a/app/views/suggested_edit/show.html.erb b/app/views/suggested_edit/show.html.erb
index 8a35f3c5a0f239f1892bc3773549a4064f3fb0f5..535a2daa67236abc8e10b32645f8c95c28093ca5 100644
--- a/app/views/suggested_edit/show.html.erb
+++ b/app/views/suggested_edit/show.html.erb
@@ -1,10 +1,17 @@
 <% content_for :title, "Suggested Edit #" + @edit.id.to_s %>
 <% post = @edit.post %>
 <% is_question_or_article = @edit.on_question? || @edit.on_article? %>
-<% may_decide = check_your_post_privilege(post, 'edit_posts') && @edit.user.id != current_user.id %>
+<% may_decide = check_your_post_privilege(post, 'edit_posts') %>
 
 <h1>Review Suggested Edit</h1>
 
+<% if params[:show_decided] != '1' && !may_decide %>
+  <div class="notice is-info has-color-teal-800">
+    <i class="fas fa-info-circle"></i> You can't approve or reject suggested edits because you haven't
+    yet earned the <%= link_to 'Edit Posts', ability_path('edit_posts') %> ability.
+  </div>
+<% end %>
+
 <% if @edit.active? && may_decide %>
   <div class="notice is-info">
     <p>This edit was suggested by another user. Good edits: </p>
@@ -16,10 +23,10 @@
   </div>
 <% end %>
 
-<div class="widget">
+<form class="widget">
   <div class="widget--body h-bg-tertiary-050">
     <div class="h-f-r">
-      <a class="button is-muted is-outlined" href="<%= generic_share_link(post)  %>">Return to post</a>
+      <a class="button is-muted is-outlined" href="<%= generic_share_link(post)  %>">View post</a>
     </div>
     <% if @edit.pending? %>
       <p class="h-m-0"><strong class="h-c-tertiary-700">Pending.</strong><br>This suggested edit is pending review.</p>
@@ -42,7 +49,11 @@
   </div>
   <div class="widget--body">
     <label for="summary" class="form-element">Suggested edit summary:</label>
-    <input id="summary" class="form-element" readonly value="<%= @edit.comment %>">
+    <%= text_field_tag :summary, @edit.comment, class: 'form-element',
+                                                data: { character_count: ".js-character-count-#{@edit.id}" } %>
+    <div class="clearfix">
+      <%= render 'shared/char_count', type: @edit.id, cur: @edit.comment&.length, max: max_edit_comment_length %>
+    </div>
   </div>
   <div class="widget--body">
     <div class="diff">
@@ -63,21 +74,22 @@
     </div>
   </div>
   <div class="widget--body">
-    <p class="h-m-0">Suggested <%= time_ago_in_words(@edit.created_at) %> ago by
-      <%= user_link @edit.user %></p>
+    <p class="h-m-0">Suggested
+      <span title="<%= @edit.created_at.iso8601 %>" ><%= time_ago_in_words(@edit.created_at) %> ago</span>
+      by <%= user_link @edit.user %></p>
   </div>
   <% if @edit.active? && may_decide %>
     <div class="widget--footer">
-      <button class="button is-filled is-medium" data-suggested-edit-approve="<%= @edit.id %>">Approve</button>
+      <button class="button is-filled is-medium js-suggested-edit-approve" data-suggested-edit-approve="<%= @edit.id %>">Approve</button>
       <button class="button is-filled is-medium js-suggested-edit-reject">Reject (with optional reason)</button>
 
       <div class="widget js-suggested-edit-reject-dialog is-hidden">
         <div class="widget--header">Rejecting this suggested edit</div>
         <div class="widget--body">
           <input class="js-rejection-reason form-element" placeholder="Why are you rejecting this edit?">
-          <button class="button is-outlined is-medium " data-suggested-edit-reject="<%= @edit.id %>">Reject</button>
+          <button class="button is-outlined is-medium" data-suggested-edit-reject="<%= @edit.id %>">Reject</button>
         </div>
       </div>
     </div>
   <% end %>
-</div>
+</form>
diff --git a/app/views/tags/_form.html.erb b/app/views/tags/_form.html.erb
index f88e1d6d4ec6d83d59d6ffc398ba75858b4e8de3..43202c83c3ecb28a1f18a05e3bd8d85f821dcb05 100644
--- a/app/views/tags/_form.html.erb
+++ b/app/views/tags/_form.html.erb
@@ -14,15 +14,51 @@
 
 <%= form_for @tag, url: submit_path do |f| %>
   <% if submit_path == create_tag_path %>
+    <% tag_name = @tag.name %>
+    <% key = "saved_post.#{current_user&.id}.#{request.path}.tag_name" %>
+    <% saved_at_key = "saved_post_at.#{current_user&.id}.#{request.path}" %>
+    <% saved_at = DateTime.parse(RequestContext.redis.get(saved_at_key) || '') rescue Date.new(2000, 1, 1) %>
+    <%
+      # Find the most recent between post-create, post-update, and draft-saved, and use the value corresponding to that.
+      tag_name = [
+        [@tag.created_at || Date.new(2000, 1, 1), @tag.name],
+        [@tag.updated_at || Date.new(2000, 1, 1), @tag.name],
+        [saved_at || Date.new(2001, 1, 1), RequestContext.redis.get(key)]
+      ].max_by do |x|
+        x[0]
+      end[1]
+    %>
     <div class="form-group">
       <%= f.label :name, 'Name', class: 'form-element' %>
       <span class="form-caption">
         Name of the tag
       </span>
-      <%= f.text_field :name, class: 'form-element' %>
+      <%= f.text_field :name, value: tag_name, class: 'form-element' %>
     </div>
   <% end %>
 
+  <div class="form-group">
+    <%= f.label :tag_synonyms, 'Tag Synonyms', class: 'form-element' %>
+    <span class="form-caption">
+      Alternative names for this tag
+    </span>
+    <div id="tag-synonyms-wrapper">
+      <% i = -1 %>
+      <%= f.fields_for :tag_synonyms do |tsf| %>
+        <div class="grid tag-synonym" data-id="<%= i+=1 %>" style="<%= 'display: none;' if tsf.object&.name.blank? %>">
+          <div class="grid--cell is-11">
+            <%= tsf.text_field :name, class: 'form-element tag-synonym-name', readonly: tsf.object&.name.present?, disabled: tsf.object&.name.blank? %>
+          </div>
+          <div class="grid--cell is-1">
+            <button type="button" class="button is-outlined remove-tag-synonym has-float-right">X</button>
+          </div>
+          <%= tsf.hidden_field :_destroy, value: tsf.object&.name.blank?, class: 'destroy-tag-synonym' %>
+        </div>
+      <% end %>
+    </div>
+  </div>
+  <button id="add-tag-synonym" type="button" class="button is-outlined">Add synonym</button>
+
   <div class="form-group">
     <%= f.label :parent_id, 'Parent tag', class: 'form-element' %>
     <span class="form-caption">
@@ -31,21 +67,39 @@
     <%= f.select :parent_id, options_for_select(@tag.parent.present? ? [[@tag.parent.name, @tag.parent_id]] : [],
                                                 selected: @tag.parent.present? ? @tag.parent_id : nil),
                  { include_blank: true }, class: "form-element js-tag-select",
-                 data: { tag_set: @category.tag_set_id, use_ids: true, placeholder: "None" } %>
+                 data: { tag_set: @category.tag_set_id, create: false, use_ids: true, placeholder: "None" } %>
   </div>
 
   <div class="form-group">
+    <% excerpt = @tag.excerpt %>
+    <% key = "saved_post.#{current_user&.id}.#{request.path}.excerpt" %>
+    <% saved_at_key = "saved_post_at.#{current_user&.id}.#{request.path}" %>
+    <% saved_at = DateTime.parse(RequestContext.redis.get(saved_at_key) || '') rescue Date.new(2000, 1, 1) %>
+    <%
+      # Find the most recent between post-create, post-update, and draft-saved, and use the value corresponding to that.
+      excerpt = [
+        [@tag.created_at || Date.new(2000, 1, 1), @tag.excerpt],
+        [@tag.updated_at || Date.new(2000, 1, 1), @tag.excerpt],
+        [saved_at || Date.new(2001, 1, 1), RequestContext.redis.get(key)]
+      ].max_by do |x|
+        x[0]
+      end[1]
+    %>
     <%= f.label :excerpt, 'Usage guidance', class: 'form-element' %>
     <span class="form-caption">
       Short usage guidance for this tag. Will be cut off at 120 characters in the tags list, but displayed in full on
       the tag page.
     </span>
-    <%= f.text_area :excerpt, class: 'form-element js-tag-excerpt', rows: 3 %>
-    <span class="has-float-right has-font-size-caption js-character-count"
-          data-target=".js-tag-excerpt" data-max="600">0 / 600</span>
+    <%= f.text_area :excerpt, class: 'form-element js-tag-excerpt',
+                              value: excerpt,
+                              rows: 3,
+                              data: { character_count: '.js-character-count-tag-excerpt' }
+    %>
+    <%= render 'shared/char_count', type: 'tag-excerpt', cur: excerpt&.length || @tag.excerpt&.length, min: 0, max: 600 %>
   </div>
 
-  <%= render 'shared/body_field', f: f, field_name: :wiki_markdown, field_label: 'Wiki', post: @tag do %>
+  <%= render 'shared/body_field', f: f, min_length: 0, max_length: 30_000, cur_length: @tag.wiki_markdown&.length, 
+                                  field_name: :wiki_markdown, field_label: 'Wiki', post: @tag do %>
     Full usage guidance and any other information you want people to know about this tag.
   <% end %>
   <div class="post-preview"></div>
diff --git a/app/views/tags/_list.html.erb b/app/views/tags/_list.html.erb
index cc7e4ea91d0ff381483c94768663366c39b2e4b3..fb46962bab92850f4792228b073d83f42f47c945 100644
--- a/app/views/tags/_list.html.erb
+++ b/app/views/tags/_list.html.erb
@@ -4,7 +4,7 @@
   <% topic_ids = @category&.topic_tag_ids %>
 
   <% ApplicationRecord.with_lax_group_rules do %>
-    <% @tags.each do |tag| %>
+    <% @tags&.each do |tag| %>
       <% required = required_ids&.include?(tag.id) ? 'is-filled' : '' %>
       <% topic = topic_ids&.include?(tag.id) ? 'is-outlined' : '' %>
       <% moderator = moderator_ids&.include?(tag.id) ? 'is-red is-outlined' : '' %>
@@ -12,4 +12,4 @@
       <%= render 'tag', category: @category, tag: tag, classes: classes %>
     <% end %>
   <% end %>
-</div>
\ No newline at end of file
+</div>
diff --git a/app/views/tags/_tag.html.erb b/app/views/tags/_tag.html.erb
index ccb2d56441b9d25af2439c7edcb2fdc86307d82f..c0beafbb01ebefd095ebdd076c76555be7b6aff3 100644
--- a/app/views/tags/_tag.html.erb
+++ b/app/views/tags/_tag.html.erb
@@ -8,6 +8,13 @@
   <% end %>
   <%= link_to tag.name, tag_path(id: category.id, tag_id: tag.id), class: classes, 'data-ckb-item-link' => '' %>
   <span class="has-color-tertiary-900">&times;&nbsp;<%= tag.post_count %></span>
+  <% if tag.tag_synonyms.present? %>
+    <p>
+    <% tag.tag_synonyms.each do |synonym| %>
+      <span class="badge is-tag is-small is-teal"><%= synonym.name %></span>
+    <% end %>
+    </p>
+  <% end %>
   <% if tag.excerpt.present? %>
     <p class="has-font-size-caption has-color-tertiary-900">
       <% splat = split_words_max_length(tag.excerpt, 120) %>
diff --git a/app/views/tags/category.html.erb b/app/views/tags/category.html.erb
index a3119f65f319489ff6a1b513d9b2826e7fb56a03..7f1c553076aef5b62fc29629a6bd60d6dd8b6d7b 100644
--- a/app/views/tags/category.html.erb
+++ b/app/views/tags/category.html.erb
@@ -2,8 +2,16 @@
 
 <h1>Tags used in <%= @category.name %></h1>
 
-<% if current_user&.is_moderator %>
-  <%= link_to 'New', new_tag_path(id: @category.id), class: 'button is-muted is-outlined' %>
+<% ApplicationRecord.with_lax_group_rules do %>
+
+<% if @tags == nil %>
+<p>This category is missing its tag set. Set this in the Category settings (admin).</p>
+<% end %>
+
+<% unless @tags == nil %>
+
+<% if current_user&.is_moderator || check_your_privilege('edit_tags') %>
+<%= link_to 'New', new_tag_path(id: @category.id), class: 'button is-muted is-outlined', 'aria-label': 'Create new tag' %>
 <% end %>
 
 <%= form_tag category_tags_path(@category), method: :get, class: 'form-inline' do %>
@@ -23,15 +31,33 @@
     <h3><%= pluralize(@count, 'tag') %></h3>
     <div class="button-list is-gutterless has-margin-2">
       <%= link_to 'Usage', category_tags_path(@category),
-                  class: "button is-muted is-outlined #{request.query_parameters.size == 0 ? 'is-active' : ''}" %>
+                  class: "button is-muted is-outlined #{request.query_parameters.size == 0 ? 'is-active' : ''}",
+                  'aria-label': 'Show tag usage for this category' %>
       <%= link_to 'Hierarchy', query_url(hierarchical: '1'),
-                  class: "button is-muted is-outlined #{params[:hierarchical].present? ? 'is-active' : ''}" %>
+                  class: "button is-muted is-outlined #{params[:hierarchical].present? ? 'is-active' : ''}",
+                  'aria-label': 'Show tag hierarchies' %>
       <%= link_to 'Missing Excerpt', query_url(no_excerpt: '1'),
-                  class: "button is-muted is-outlined #{params[:no_excerpt].present? ? 'is-active' : ''}" %>
+                  class: "button is-muted is-outlined #{params[:no_excerpt].present? ? 'is-active' : ''}",
+                  'aria-label': 'Show tags without excerpts' %>
     </div>
   </div>
 <% end %>
 
 <%= render 'list' %>
 
+<% if @tags&.size == 0 %>
+  <% if params[:q].present? %>
+    <p class="has-color-tertiary"><em>No results for <strong><%= params[:q] %></strong></em></p>
+  <% else %>
+    <p class="has-color-tertiary"><em>There are no tags in <%= @category.name %></em></p>
+  <% end %>
+<% end %>
+
+<% if @tags&.size > 0 %>
 <%= will_paginate @tags, renderer: BootstrapPagination::Rails %>
+<% end %>
+
+<% end %> <%# unless @tags == nil %>
+
+<% end %> <%# ApplicationRecord.with_lax_group_rules %>
+
diff --git a/app/views/tags/nuke_warning.html.erb b/app/views/tags/nuke_warning.html.erb
index 677c2d10c5276026c09779290a6e31f7cc551776..5813fc82018d0d745e45897046b5bdc1c8f050f4 100644
--- a/app/views/tags/nuke_warning.html.erb
+++ b/app/views/tags/nuke_warning.html.erb
@@ -23,6 +23,7 @@
   <p>This action will be logged.</p>
 </div>
 
-<%= link_to nuke_tag_path(id: @category.id, tag_id: @tag.id), class: 'button is-danger is-filled', method: :delete do %>
+<%= link_to nuke_tag_path(id: @category.id, tag_id: @tag.id), class: 'button is-danger is-filled', method: :delete,
+            role: 'button' do %>
   I understand, delete <%= @tag.name %>
 <% end %>
diff --git a/app/views/tags/show.html.erb b/app/views/tags/show.html.erb
index b2350de223adc09ede337dcc0ed0aa2a457088de..341c5329256dcf221f6647e5caf37e3095864ded 100644
--- a/app/views/tags/show.html.erb
+++ b/app/views/tags/show.html.erb
@@ -4,22 +4,35 @@
   Posts tagged <span class="<%= tag_classes(@tag, @category) %> is-large"><%= @tag.name %></span>
   <% if moderator? %>
     <a href="#" class="has-font-size-caption js-rename-tag h-m-r-2"
-       data-category="<%= @category.id %>" data-tag="<%= @tag.id %>" data-name="<%= @tag.name %>">
+       data-category="<%= @category.id %>" data-tag="<%= @tag.id %>" data-name="<%= @tag.name %>"
+       aria-label="Rename this tag">
       <i class="fas fa-pencil-alt" aria-label="Rename" title="Rename"></i>
     </a>
-    <%= link_to select_tag_merge_path(id: @category.id, tag_id: @tag.id), class: 'has-font-size-caption h-m-r-2' do %>
+    <%= link_to select_tag_merge_path(id: @category.id, tag_id: @tag.id), class: 'has-font-size-caption h-m-r-2',
+                'aria-label': 'Merge this tag with other tags' do %>
       <i class="fas fa-project-diagram" aria-label="Merge" title="Merge"></i>
     <% end %>
   <% end %>
   <% if admin? %>
-    <%= link_to nuke_tag_warning_path(id: @category.id, tag_id: @tag.id), class: 'has-font-size-caption h-m-r-2 is-red' do %>
+    <%= link_to nuke_tag_warning_path(id: @category.id, tag_id: @tag.id), class: 'has-font-size-caption h-m-r-2 is-red',
+                'aria-label': 'Nuke this tag' do %>
       <i class="fas fa-trash" aria-label="Nuke" title="Nuke"></i>
     <% end %>
   <% end %>
 </h1>
+
+<% if @tag.tag_synonyms.any? %>
+  <h3 class="has-margin-0" role="list">
+    Synonyms:
+    <% @tag.tag_synonyms.each do |synonym| %>
+      <span class="badge is-tag is-teal" role="listitem"><%= synonym.name %></span>
+    <% end %>
+  </h3>
+<% end %>
+
 <p class="has-color-tertiary-900 has-font-weight-normal has-margin-0 has-font-family-brand">
   <% if @tag.parent_id.present? %>
-      Subtag of <%= link_to @tag.parent.name, tag_path(id: @category.id, tag_id: @tag.parent_id),
+    Subtag of <%= link_to @tag.parent.name, tag_path(id: @category.id, tag_id: @tag.parent_id),
                             class: tag_classes(@tag.parent, @category) %>
   <% end %>
   <% child_count = @tag.children.count %>
@@ -39,7 +52,8 @@
         <em>
           This tag doesn't have any usage information yet.
           <% if current_user&.privilege?('edit_tags') %>
-            <%= link_to 'Help us create it', edit_tag_path(id: @category.id, tag_id: @tag.id) %>.
+            <%= link_to 'Help us create it', edit_tag_path(id: @category.id, tag_id: @tag.id),
+                        'aria-label': 'Help us create usage information for this tag' %>.
           <% end %>
         </em>
       </p>
@@ -48,12 +62,12 @@
   <div class="widget--body">
     <% if @tag.wiki.present? %>
       <% if @tag.wiki.length < 600 %>
-      <%= raw(sanitize(@tag.wiki, scrubber: scrubber)) %>
-      <% else %>
-      <details>
-        <summary>Tag Wiki</summary>
         <%= raw(sanitize(@tag.wiki, scrubber: scrubber)) %>
-      </details>
+      <% else %>
+        <details>
+          <summary>Tag Wiki</summary>
+          <%= raw(sanitize(@tag.wiki, scrubber: scrubber)) %>
+        </details>
       <% end %>
     <% end %>
     <% unless @tag.wiki.present? %>
@@ -61,7 +75,8 @@
         <em>
           This tag doesn't have a detailed wiki yet.
           <% if current_user&.privilege?('edit_tags') %>
-            <%= link_to 'Help us create it', edit_tag_path(id: @category.id, tag_id: @tag.id) %>.
+            <%= link_to 'Help us create it', edit_tag_path(id: @category.id, tag_id: @tag.id),
+                        'aria-label': 'Help us create a detailed wiki for this tag' %>.
           <% end %>
         </em>
       </p>
@@ -70,24 +85,39 @@
 </div>
 
 <% if current_user&.privilege?('edit_tags') %>
-  <%= link_to 'Edit', edit_tag_path(id: @category.id, tag_id: @tag.id), class: 'button is-muted is-outlined' %>
+  <%= link_to 'Edit', edit_tag_path(id: @category.id, tag_id: @tag.id), class: 'button is-muted is-outlined',
+              'aria-label': "Edit tag #{@tag.name}" %>
 <% end %>
 
 <% post_count = @posts.count %>
 <div class="has-color-tertiary-500 category-meta" title="<%= post_count %>">
-  <div>
-    <%= short_number_to_human post_count, precision: 1, significant: false %>
-    <%= 'post'.pluralize(post_count) %>
-    <%= link_to 'Subscribe',
-                new_subscription_path(type: 'tag', qualifier: @tag.name, return_to: request.path),
-                class: 'button is-outlined' %>
+  <div class="category-meta--start">
+    <span title="<%= post_count %> posts with this tag">
+      <%= short_number_to_human post_count, precision: 1, significant: false %>
+      <%= 'post'.pluralize(post_count) %>
+    </span>
+    <%= render 'shared/rss_link', url: tag_path(id: @category.id, tag_id: @tag.id, format: 'rss'), tooltip: 'RSS feed for this tag' %>
+    <% if user_signed_in? && current_user&.subscriptions.where(type: 'tag', qualifier: @tag.name).exists? %>
+      <span class="has-margin-4" title="Subscribed: manage email subscriptions">
+        <%= link_to 'Subscribed', subscriptions_path(current_user),
+          class: 'button is-outlined', 'aria-label': "Subscribed to tag #{@tag.name}" %>
+      </span>
+    <% else %>
+      <span title="Create email subscription">
+        <%= link_to 'Subscribe',
+          new_subscription_path(type: 'tag', qualifier: @tag.name, return_to: request.path),
+          class: 'button is-outlined', 'aria-label': "Subscribe to tag #{@tag.name}" %>
+      </span>
+    <% end %>
   </div>
 
   <div class="button-list is-gutterless has-margin-2">
     <%= link_to 'Tag Only', query_url(self: 1),
-                class: "button is-muted is-outlined #{params[:self].present? ? 'is-active' : ''}" %>
+                class: "button is-muted is-outlined #{params[:self].present? ? 'is-active' : ''}",
+                role: 'button', 'aria-label': 'View this tag only, without showing its children' %>
     <%= link_to 'Tag + Children', tag_path(id: @category.id, tag_id: @tag.id),
-                class: "button is-muted is-outlined #{params[:self].nil? ? 'is-active' : ''}" %>
+                class: "button is-muted is-outlined #{params[:self].nil? ? 'is-active' : ''}",
+                role: 'button', 'aria-label': 'View this tag and its children'%>
   </div>
 </div>
 
@@ -102,7 +132,5 @@
 </div>
 
 <div class="has-padding-top-4">
-  <%= link_to tag_path(id: @category.id, tag_id: @tag.id, format: 'rss') do %>
-    <i class="fas fa-rss"></i> Tag RSS feed
-  <% end %>
+  <%= render 'shared/rss_link', url: tag_path(id: @category.id, tag_id: @tag.id, format: 'rss'), text: 'Tag RSS feed' %>
 </div>
diff --git a/app/views/tour/question2.html.erb b/app/views/tour/question2.html.erb
index 008550ba3292f1893736aeffa25a293337fff1d6..b2da72932366386ec09f3876d0913603444348d4 100644
--- a/app/views/tour/question2.html.erb
+++ b/app/views/tour/question2.html.erb
@@ -43,7 +43,13 @@
 
     <div class="form-group">
       <label class="form-element" for="post_tags_cache">Tags (at least one):</label>
-      <input name="post[tags_cache][]" type="hidden" value="" /><select multiple="multiple" class="form-element js-tag-select" data-tag-set="-1" name="post[tags_cache][]" id="post_tags_cache"><option value=""></option></select>
+      <input name="post[tags_cache][]" type="hidden" value="" />
+      <select multiple="multiple" 
+              class="form-element js-tag-select" 
+              data-tag-set="<%= @tagset_id %>" 
+              name="post[tags_cache][]" 
+              id="post_tags_cache">
+      </select>
     </div>
 
   </div>
diff --git a/app/views/tour/question3.html.erb b/app/views/tour/question3.html.erb
index eeb62f939ba9332acbc449554992829b71370fd2..0c783c16a14922a364f49a00593de8bf050a64bf 100644
--- a/app/views/tour/question3.html.erb
+++ b/app/views/tour/question3.html.erb
@@ -339,7 +339,7 @@
           <p>Voting also helps good content be more visible. Good answers are promoted to the top, while bad ones sink to the bottom. Please help curate the community by voting responsibly.</p>
           <p>Do so by clicking the <svg class="h-c-tertiary-400" width="2em" height="1.33em" viewbox="0 0 100 50">
             <path d="M50,0 L100,50 L0,50 Z" fill="currentColor" />
-          </svg> icon next to the post.</p>
+          </svg> icon next to the post. After you vote, the tour will continue.</p>
         </div>
         <div class="widget--footer">
           <button class="button"  data-step-from=".step-2" data-step-to=".step-3">Back</button>
diff --git a/app/views/two_factor/disable_link.html.erb b/app/views/two_factor/disable_link.html.erb
index e4f9a5eb800db520f8f7ba1e525ec491fa573eec..e6b98aaecf8fb5328c4cc4a07763f2ae21434890 100644
--- a/app/views/two_factor/disable_link.html.erb
+++ b/app/views/two_factor/disable_link.html.erb
@@ -2,4 +2,4 @@
 
 <p>Please click the button below to confirm you want to disable two-factor authentication on your account.</p>
 
-<p><%= link_to 'Disable 2FA', two_factor_confirm_disable_link_path, method: :post, class: 'button is-filled is-danger' %></p>
\ No newline at end of file
+<p><%= link_to 'Disable 2FA', two_factor_confirm_disable_link_path, method: :post, class: 'button is-filled is-danger', role: 'button' %></p>
\ No newline at end of file
diff --git a/app/views/two_factor/enable_2fa.html.erb b/app/views/two_factor/enable_2fa.html.erb
index 1b82c7df27283c2a25ed45d36bb6398150ccc01e..a92ca2ebe3dfccf8483286708212df6db1d7851f 100644
--- a/app/views/two_factor/enable_2fa.html.erb
+++ b/app/views/two_factor/enable_2fa.html.erb
@@ -4,6 +4,9 @@
   your mobile authenticator. Scan the QR code below with your authenticator app and you should be set to go.
 </p>
 
-<div class="has-text-align-center"><img alt src="<%= @qr_uri %>" width="300" /></div>
+<div class="has-text-align-center">
+  <img alt src="<%= @qr_uri %>" width="300" />
+  <div class="has-color-tertiary-600 has-font-size-caption"><%= @secret %></div>
+</div>
 
-<%= link_to 'Done, next step', two_factor_enable_code_path, class: 'button is-filled' %>
\ No newline at end of file
+<%= link_to 'Done, next step', two_factor_enable_code_path, class: 'button is-filled', role: 'button' %>
diff --git a/app/views/two_factor/enable_code.html.erb b/app/views/two_factor/enable_code.html.erb
index e458ede38737579a7c9be07e91230ccb654c4982..28a05ebdad1963125713dde32e8e18f9f186de18 100644
--- a/app/views/two_factor/enable_code.html.erb
+++ b/app/views/two_factor/enable_code.html.erb
@@ -1,5 +1,16 @@
 <h1>Confirm your code</h1>
-<p>Last step: to confirm that you're all correctly set up, enter the code shown in your mobile authenticator below.</p>
+<p>
+  First, save your two-factor <strong>backup code</strong> somewhere safe. You can use it to log in and reset
+  two-factor authentication if you lose access to your app.
+</p>
+
+<details>
+  <summary>Show code</summary>
+  <label for="backup-code" class="form-element">2FA backup code</label>
+  <input class="form-element" type="text" readonly name="backup-code" id="backup-code" value="<%= current_user.backup_2fa_code %>" />
+</details>
+
+<p>Last step: to confirm that you're all correctly set up, enter the code shown in your authenticator app below.</p>
 
 <%= form_tag two_factor_confirm_enable_path, autocomplete: 'off' do %>
   <div class="field">
diff --git a/app/views/two_factor/tf_status.html.erb b/app/views/two_factor/tf_status.html.erb
index 3c67000a46e862e22effa472d058dbffe0a8d149..c309c379ef94811e57f1517f1da7166427025ff9 100644
--- a/app/views/two_factor/tf_status.html.erb
+++ b/app/views/two_factor/tf_status.html.erb
@@ -21,22 +21,43 @@
 <% if current_user.enabled_2fa %>
   <p>
     <% if current_user.two_factor_method == 'app' %>
-      <%= link_to 'Turn off', two_factor_disable_code_path, class: 'button is-outlined' %>
+      <%= link_to 'Turn off', two_factor_disable_code_path, class: 'button is-outlined', 'aria-label': 'Turn off 2FA using an app' %>
     <% elsif current_user.two_factor_method == 'email' %>
-      <%= link_to 'Turn off', two_factor_send_disable_email_path, method: :post, class: 'button is-outlined' %>
+      <%= link_to 'Turn off', two_factor_send_disable_email_path, method: :post, class: 'button is-outlined', 'aria-label': 'Turn off 2FA using email' %>
     <% end %>
   </p>
 <% else %>
   <div>
     <h3>Enable two-factor authentication</h3>
-    <%= form_tag two_factor_enable_path do %>
-      <div class="form-group">
-        <%= select_tag :method, options_for_select([['Get codes from an app on your phone', 'app'],
-                                                    ['Get a sign-in link emailed to you', 'email']]),
-                       { class: 'form-element' } %>
-      </div>
+    <% if current_user.sso_profile.present? && !SiteSetting['Enable2FAForSsoUsers'] %>
+      <p>
+        Because you sign in with Single Sign-On, you cannot enable two-factor authentication.
+      </p>
+    <% else %>
+      <%= form_tag two_factor_enable_path do %>
+        <div class="form-group">
+          <%= select_tag :method, options_for_select([['Get codes from an app on your phone', 'app'],
+                                                      ['Get a sign-in link emailed to you', 'email']]),
+                         { class: 'form-element' } %>
+        </div>
 
-      <%= submit_tag 'Enable', class: 'button is-filled' %>
+        <%= submit_tag 'Enable', class: 'button is-filled', 'aria-label': 'Enable 2FA' %>
+      <% end %>
     <% end %>
   </div>
-<% end %>
\ No newline at end of file
+<% end %>
+
+<% if current_user.enabled_2fa && current_user.two_factor_method == 'app' %>
+  <h3>Recovery</h3>
+  <p>
+    Your <strong>backup code</strong> for two-factor authentication can be found below. You should save this somewhere
+    safe &mdash; if you lose access to your authenticator app, you can sign in using this code to get back into your
+    account and reconfigure 2FA.
+  </p>
+
+  <%= form_tag two_factor_backup_code_path, class: 'js-backup-code-form' do %>
+    <%= label_tag :code, 'Enter a 2FA code to access your backup code', class: 'form-element' %>
+    <%= text_field_tag :code, nil, class: 'form-element' %>
+    <%= submit_tag 'Show backup code', class: 'button is-filled' %>
+  <% end %>
+<% end %>
diff --git a/app/views/two_factor_mailer/backup_code.html.erb b/app/views/two_factor_mailer/backup_code.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..9b8d9882594111728c98c8969c9aa6b301a784ed
--- /dev/null
+++ b/app/views/two_factor_mailer/backup_code.html.erb
@@ -0,0 +1,21 @@
+<p>Hi <%= @user&.username %>,</p>
+
+<h3>Your two-factor authentication backup code</h3>
+
+<p>
+  You're receiving this email because you have two-factor authentication enabled on your Codidact network account. We've
+  recently added backup codes for every account using 2FA, so that you can still log in if you lose access to your
+  authenticator app.
+</p>
+
+<p>
+  Your backup code can be found in your user account settings, under
+  <a href="<%= two_factor_status_url %>">Two-Factor Authentication settings</a>. Please save it somewhere safe.
+  Please note that logging in using this backup code will disable 2FA on your account so that you can reconfigure it
+  using a new app or device &mdash; this will also reset your backup code.
+</p>
+
+<p>
+  Thanks,<br/>
+  The Codidact Team
+</p>
diff --git a/app/views/two_factor_mailer/backup_code.text.erb b/app/views/two_factor_mailer/backup_code.text.erb
new file mode 100644
index 0000000000000000000000000000000000000000..6ae44eed54a082e70465959b2030a01ab9972d23
--- /dev/null
+++ b/app/views/two_factor_mailer/backup_code.text.erb
@@ -0,0 +1,15 @@
+Hi <%= @user&.username %>,
+
+Re: Your two-factor authentication backup code
+
+
+You're receiving this email because you have two-factor authentication enabled on your Codidact network account. We've
+recently added backup codes for every account using 2FA, so that you can still log in if you lose access to your
+authenticator app.
+
+Your backup code can be found in your user account settings, under Two-Factor Authentication settings. Please save it
+somewhere safe. Please note that logging in using this backup code will disable 2FA on your account so that you can
+reconfigure it using a new app or device &mdash; this will also reset your backup code.
+
+Thanks,
+The Codidact Team
diff --git a/app/views/users/_activity_items.html.erb b/app/views/users/_activity_items.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..f31a1c5522a0860298050467d3cde413fd7b08ea
--- /dev/null
+++ b/app/views/users/_activity_items.html.erb
@@ -0,0 +1,108 @@
+<table class="table is-with-hover is-full-width">
+  <tr>
+    <th>Type</th>
+    <th>On...</th>
+    <th>Excerpt</th>
+    <th>Status</th>
+    <th>Date</th>
+  </tr>
+  <% @items.each do |i| %>
+    <tr class="<%= deleted_item?(i) ? 'deleted-content' : '' %>">
+      <% if i.class == Post %>
+        <% type_name = (i.question? ? 'Question' : (i.article? ? 'Article' : 'Answer')) %>
+        <td>
+          <span class="h-fw-bold">
+            <%= type_name %>
+          </span>
+        </td>
+        <td>&mdash;</td>
+        <td class="h-fs-caption">
+          <% working_title = i.answer? ? "A: #{i.parent.title}" : i.title %>
+          <strong><%= working_title %></strong><br>
+          <%= i.body_plain[0..300] + ((i.body_plain.length > 300) ? "..." : "") %><br>
+          <%= link_to '(more)', generic_share_link(i), 'aria-label': "More information about #{type_name} #{working_title}" %>
+        </td>
+        <td>&mdash;</td>
+      <% elsif i.class == Comment %>
+        <td>
+          Comment
+        </td>
+        <td>
+          <%= link_to "Post #" + i.post.id.to_s, generic_share_link(i.post) %>
+        </td>
+        <td class="h-fs-caption">
+          <%= i.content[0..300] + ((i.content.length > 300) ? "..." : "") %><br>
+          <%= link_to '(more)', comment_link(i), 'aria-label': 'More information about comment' %>
+        </td>
+        <td>&mdash;</td>
+      <% elsif i.class == PostHistory %>
+        <td>
+          Edit
+        </td>
+        <td>
+          <%= link_to "Post #" + i.post.id.to_s, generic_share_link(i.post) %>
+        </td>
+        <td class="h-fs-caption">
+          <% if i.comment %>
+            <em><%= i.post_history_type.name.gsub("_", " ").capitalize %></em>:<br><%= i.comment %>
+          <% else %>
+            <em><%= i.post_history_type.name.gsub("_", " ").capitalize %></em>
+          <% end %>
+        </td>
+        <td>&mdash;</td>
+      <% elsif i.class == SuggestedEdit %>
+        <td>
+          Suggested Edit
+        </td>
+        <td>
+          <%= link_to "Post #" + i.post.id.to_s, generic_share_link(i.post) %>
+        </td>
+        <td class="h-fs-caption">
+          <em>Suggested edit</em>:<br><%= i.comment %><br>
+          <%= link_to '(more)', suggested_edit_url(i.id), 'aria-label': "More information about suggested edit #{i.comment}" %>
+        </td>
+        <td>
+          <%= (i.pending? ? "pending" : (i.approved? ? "helpful" : "declined")) %>
+        </td>
+      <% elsif mod && i.class == ModWarning %>
+        <td>
+          <span class="h-fw-bold h-c-red-700">Warning</span>
+        </td>
+        <td>&mdash;</td>
+        <td class="h-fs-caption">
+          <%= i.body[0..300] + ((i.body.length > 300) ? "..." : "") %>
+        </td>
+        <td>&mdash;</td>
+      <% elsif mod && i.class == Flag %>
+        <td>
+          Flag
+        </td>
+        <td>
+	  <% if i.post_type == 'Post' %>
+          <%= link_to "Post #" + i.post.id.to_s, generic_share_link(i.post)%>
+	  <% elsif i.post_type == 'Comment' %>
+	  <%= link_to "Comment thread: " + i.post.comment_thread.title, comment_link(i.post) %>
+	  <% end %>
+        </td>
+        <td class="h-fs-caption">
+          <%= i.reason[0..300] + ((i.reason.length > 300) ? "..." : "") %><br>
+        </td>
+        <td>
+          <%= i.status || "pending" %>
+        </td>
+      <% else %>
+        <td>
+          <span class="badge is-tag is-muted">Unknown</span>
+        </td>
+        <td>&mdash;</td>
+        <td><%= i.class %></td>
+        <td>&mdash;</td>
+      <% end %>
+      <td>
+        <span title="<%= i.created_at.iso8601 %>"><%= time_ago_in_words(i.created_at) %> ago</span>
+      </td>
+    </tr>
+  <% end %>
+</table>
+
+<%= will_paginate @items, renderer: BootstrapPagination::Rails %>
diff --git a/app/views/users/_network.html.erb b/app/views/users/_network.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..a1f9258aad1312c374953fa553cfddce9c9f206d
--- /dev/null
+++ b/app/views/users/_network.html.erb
@@ -0,0 +1,22 @@
+<table class="table is-with-hover is-full-width">
+  <tr>
+    <th>Profile on Community</th>
+    <th>Posts</th>
+    <th>Reputation</th>
+  </tr>
+
+  <% @communities.each do |c| %>
+    <% if @user.has_profile_on(c) %>
+    <tr>
+      <td><%= user_link @user, { host: c.host}, anchortext: c.name %>
+	<% if @user.is_moderator_on(c) %>
+	(moderator)
+	<% end %>
+      </td>
+      <td><%= @user.post_count_on(c) %> </td>
+      <td><%= @user.reputation_on(c) %> </td>
+    </tr>
+    <% end %>
+  <% end %>
+
+</table>
diff --git a/app/views/users/_tabs.html.erb b/app/views/users/_tabs.html.erb
index 0c4713fef3d009b39f3145f2d5c22e1a34627169..fb2c9d191fe7f60c06067c65e123446ace06750d 100644
--- a/app/views/users/_tabs.html.erb
+++ b/app/views/users/_tabs.html.erb
@@ -18,5 +18,14 @@
     <%= link_to user_preferences_path, class: "tabs--item #{current_page?(user_preferences_path) ? 'is-active' : ''}" do %>
       Preferences
     <% end %>
+    <%= link_to user_filters_path, class: "tabs--item #{current_page?(user_filters_path) ? 'is-active' : ''}" do %>
+      Filters
+    <% end %>
+    <%= link_to notifications_path, class: "tabs--item #{current_page?(notifications_path) ? 'is-active' : ''}" do %>
+      Notifications
+    <% end %>
+  <% end %>
+  <%= link_to network_path(user), class: "tabs--item #{current_page?(network_path(user)) ? 'is-active' : ''}" do %>
+  All Communities
   <% end %>
-</div>
\ No newline at end of file
+</div>
diff --git a/app/views/users/activity.html.erb b/app/views/users/activity.html.erb
index d9afc9b57441325b3440057dafe84442f2e451bc..e86b14425762cbc9f6f85e65fed0136433cdd2c6 100644
--- a/app/views/users/activity.html.erb
+++ b/app/views/users/activity.html.erb
@@ -7,104 +7,24 @@
 <h1>Activity for <%= user_link @user %></h1>
 
 <div class="button-list is-gutterless h-m-v-4">
-  <a class="button is-muted is-outlined" href="?">
+  <a class="button is-muted is-outlined" href="?" role="button">
     Show all events
   </a>
-  <a class="button is-muted is-outlined <%= params[:filter] == 'posts' ? 'is-active' : '' %>" href="?filter=posts">
+  <a class="button is-muted is-outlined <%= params[:filter] == 'posts' ? 'is-active' : '' %>" href="?filter=posts" role="button">
     Posts
     <% if @posts > 0 %><span class="badge is-status"><%= @posts %></span>
     <% end %>
   </a>
-  <a class="button is-muted is-outlined <%= params[:filter] == 'comments' ? 'is-active' : '' %>" href="?filter=comments">
+  <a class="button is-muted is-outlined <%= params[:filter] == 'comments' ? 'is-active' : '' %>" href="?filter=comments" role="button">
     Comments
     <% if @comments > 0 %><span class="badge is-status"><%= @comments %></span>
     <% end %>
   </a>
-  <a class="button is-muted is-outlined <%= params[:filter] == 'edits' ? 'is-active' : '' %>" href="?filter=edits">
+  <a class="button is-muted is-outlined <%= params[:filter] == 'edits' ? 'is-active' : '' %>" href="?filter=edits" role="button">
     Edits
     <% if @all_edits > 0 %><span class="badge is-status"><%= @all_edits %></span>
     <% end %>
   </a>
 </div>
 
-<table class="table is-with-hover is-full-width">
-  <tr>
-    <th>Type</th>
-    <th>On...</th>
-    <th>Excerpt</th>
-    <th>Status</th>
-    <th>Date</th>
-  </tr>
-  <% @items.each do |i| %>
-    <tr class="<%= deleted_item?(i) ? 'deleted-content' : '' %>">
-      <% if i.class == Post %>
-        <td>
-                    <span class="h-fw-bold">
-                        <%= (i.question? ? "Question" : (i.article? ? "Article" : "Answer")) %>
-                    </span>
-        </td>
-        <td>&mdash;</td>
-        <td class="h-fs-caption">
-          <% if !i.answer? %>
-            <strong><%= i.title %></strong><br>
-          <% else %>
-            <strong>A: <%= i.parent.title %></strong><br>
-          <% end %>
-          <%= i.body_plain[0..300] + ((i.body_plain.length > 300) ? "..." : "") %><br>
-          <%= link_to '(more)', generic_share_link(i) %>
-        </td>
-        <td>&mdash;</td>
-      <% elsif i.class == Comment %>
-        <td>
-          Comment
-        </td>
-        <td>
-          <%= link_to "Post #" + i.post.id.to_s, generic_share_link(i.post) %>
-        </td>
-        <td class="h-fs-caption">
-          <%= i.content[0..300] + ((i.content.length > 300) ? "..." : "") %><br>
-          <%= link_to '(more)', comment_link(i) %>
-        </td>
-        <td>&mdash;</td>
-      <% elsif i.class == PostHistory %>
-        <td>
-          Edit
-        </td>
-        <td>
-          <%= link_to "Post #" + i.post.id.to_s, generic_share_link(i.post) %>
-        </td>
-        <td class="h-fs-caption">
-          <% if i.comment %>
-            <em><%= i.post_history_type.name.gsub("_", " ").capitalize %></em>:<br><%= i.comment %>
-          <% else %>
-            <em><%= i.post_history_type.name.gsub("_", " ").capitalize %></em>
-          <% end %>
-        </td>
-        <td>&mdash;</td>
-      <% elsif i.class == SuggestedEdit %>
-        <td>
-          Suggested Edit
-        </td>
-        <td>
-          <%= link_to "Post #" + i.post.id.to_s, generic_share_link(i.post) %>
-        </td>
-        <td class="h-fs-caption">
-          <em>Suggested edit</em>:<br><%= i.comment %><br><%= link_to '(more)', suggested_edit_url(i.id) %>
-        </td>
-        <td>
-          <%= (i.pending? ? "pending" : (i.approved? ? "helpful" : "declined")) %>
-        </td>
-      <% else %>
-        <td>
-          <span class="badge is-tag is-muted">Unknown</span>
-        </td>
-        <td>&mdash;</td>
-        <td><%= i.class %></td>
-        <td>&mdash;</td>
-      <% end %>
-      <td>
-        <span title="<%= i.created_at.iso8601 %>"><%= time_ago_in_words(i.created_at) %> ago</span>
-      </td>
-    </tr>
-  <% end %>
-</table>
\ No newline at end of file
+<%= render 'activity_items', mod: false %>
diff --git a/app/views/users/disconnect_sso.html.erb b/app/views/users/disconnect_sso.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..b55f669fb21dcddb929d369fc6c46a7c61ed79a5
--- /dev/null
+++ b/app/views/users/disconnect_sso.html.erb
@@ -0,0 +1,12 @@
+<h1>Disconnect from Single Sign-On</h1>
+<p>
+  You currently sign in using your institutions Single Sign-On service.
+  It is possible to switch to signing in with a normal email address and password by disconnecting your account from
+  the Single Sign-On service.
+  After disconnecting your account, you will be sent an email to set the password for your account.
+</p>
+
+<%= link_to 'Disconnect my account from SSO',
+            user_confirm_disconnect_sso_path,
+            method: :post,
+            class: 'button is-filled is-danger' %>
\ No newline at end of file
diff --git a/app/views/users/edit_profile.html.erb b/app/views/users/edit_profile.html.erb
index 330bdbbb93af42a6ec4352dd49ad541f17f7c565..654a2ef29e13238c48c4619437e5e426be1ec812 100644
--- a/app/views/users/edit_profile.html.erb
+++ b/app/views/users/edit_profile.html.erb
@@ -34,33 +34,41 @@
   <div class="form-group has-padding-2">
     <%= f.label :username, class: "form-element" %>
     <div class="form-caption">What other people call you.</div>
-    <%= f.text_field :username, class: 'form-element', autocomplete: 'off' %>
+    <%= f.text_field :username, class: 'form-element', autocomplete: 'off', data: { character_count: '.js-character-count-user-name' } %>
+    <%= render 'shared/char_count', type: 'user-name', cur: current_user.username&.length, min: 3, max: 50 %>
   </div>
 
-  <%= render 'shared/body_field', f: f, field_name: :profile_markdown, field_label: 'Profile', post: current_user %>
+  <%= render 'shared/body_field', f: f, field_name: :profile_markdown, field_label: 'Profile', post: current_user,
+            cur_length: current_user.profile_markdown&.length, min_length: 0 %>
 
+  <% unless current_user.community_user.privilege?('unrestricted') %>
+    <p>Note: Links are not shown publicly until you have earned the Participate Everywhere ability.</p>
+  <% end %>
   <div class="post-preview"></div>
 
-  <div class="grid">
-    <div class="grid--cell is-4 is-12-sm form-group">
-      <%= f.label :website, class: "form-element" %>
-      <span class="form-caption">A link to anywhere on the internet for your stuff.</span>
-      <%= f.text_field :website, class: 'form-element', autocomplete: 'off', placeholder: 'https://...' %>
-    </div>
-
-    <div class="grid--cell is-4 is-12-sm form-group">
-      <%= f.label :twitter, class: "form-element" %>
-      <span class="form-caption">Your Twitter username, if you've got one you want to share.</span>
-      <%= f.text_field :twitter, class: 'form-element', autocomplete: 'off', placeholder: '@username' %>
-    </div>
-
-    <div class="grid--cell is-4 is-12-sm form-group">
-      <%= f.label :discord, class: 'form-element' %>
-      <span class="form-caption">Your Discord user tag, in the format <code>username#1234</code>.</span>
-      <%= f.text_field :discord, class: 'form-element', autocomplete: 'off', placeholder: 'username#1234' %>
+  <div>
+    <p>Extra fields -- your web site, GitHub profile, social-media usernames, whatever you want. Only values that begin with "http" are rendered as links.</p>
+    <div class="grid">
+      <%= f.fields_for :user_websites do |w| %>
+      <div class="grid grid--cell is-12 is-12-sm">
+        <div class="grid grid--cell is-3 is-3-sm">
+          <div class="grid--cell is-12"><%= w.text_field :label, class: 'form-element', autocomplete: 'off', placeholder: 'label' %></div>
+        </div>
+        <div class="grid grid--cell is-6 is-9-sm">
+          <div class="grid--cell is-12"><%= w.text_field :url, class: 'form-element', autocomplete: 'off', placeholder: 'https://...' %></div>
+        </div>
+      </div>
+      <% end %>
     </div>
   </div>
 
+  <div class="form-group has-padding-2">
+    <%= f.label :discord, class: 'form-element' %>
+    <span class="form-caption">Your Discord user tag, <code>username</code> or <code>username#1234</code>.</span>
+    <%= f.text_field :discord, class: 'form-element', autocomplete: 'off', placeholder: 'username#1234' %>
+  </div>
+    
+  
   <%= f.submit 'Save', class: 'button is-filled' %>
 <% end %>
 
@@ -72,7 +80,7 @@
       you'll be able to request that any content you created on Stack Exchange that has been copied across here is
       transferred to you.
     </p>
-    <%= link_to stack_oauth_url, class: 'button is-outlined' do %>
+    <%= link_to stack_oauth_url, class: 'button is-outlined', 'aria-label': 'Authenticate Stack Exchange account' do %>
       <i class="fab fa-stack-exchange"></i> Authenticate
     <% end %>
   </div>
@@ -101,4 +109,4 @@
       <%= submit_tag 'Claim Content', name: nil, class: 'button is-outlined' %>
     <% end %>
   </div>
-<% end %>
\ No newline at end of file
+<% end %>
diff --git a/app/views/users/filters.html.erb b/app/views/users/filters.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..95a83c71f84d6f3fb455ef65cc799f114c0fae32
--- /dev/null
+++ b/app/views/users/filters.html.erb
@@ -0,0 +1,11 @@
+<%= render 'tabs', user: current_user %>
+
+<h1>Filters</h1>
+<p>
+  Manage your filters here
+</p>
+
+<%# Just a nominal form so that stuff like validation works %>
+<form>
+  <%= render 'search/filters', allow_apply: false, allow_delete: true %>
+<form>
\ No newline at end of file
diff --git a/app/views/users/full_log.html.erb b/app/views/users/full_log.html.erb
index 3bb5c826a2776ec47269601d0ce0dd562a029a3f..bbd01e8350985e768a049f85c0890b1007b3f22a 100644
--- a/app/views/users/full_log.html.erb
+++ b/app/views/users/full_log.html.erb
@@ -1,4 +1,4 @@
-<% content_for :title, "Moderator Tools: #{rtl_safe_username(@user)}" %>
+<% content_for :title, "Full Activity Log: #{rtl_safe_username(@user)}" %>
 
 <h1>Full activity log for <%= user_link @user %></h1>
 
@@ -9,137 +9,36 @@
 <% end %>
 
 <div class="button-list is-gutterless h-m-v-4">
-    <a class="button is-muted is-outlined" href="?">
+    <a class="button is-muted is-outlined" href="?" role="button">
         Show all events
     </a>
     <% if @interesting > 0 %>
-    <a class="button is-muted is-outlined <%= params[:filter] == 'interesting' ? 'is-active' : '' %>" href="?filter=interesting">
+    <a class="button is-muted is-outlined <%= params[:filter] == 'interesting' ? 'is-active' : '' %>" href="?filter=interesting" role="button">
         Negative
         <span class="badge is-status"><%= @interesting %></span>
     </a>
     <% end %>
-    <a class="button is-muted is-outlined <%= params[:filter] == 'posts' ? 'is-active' : '' %>" href="?filter=posts">
+    <a class="button is-muted is-outlined <%= params[:filter] == 'posts' ? 'is-active' : '' %>" href="?filter=posts" role="button">
         Posts
         <% if @posts > 0 %><span class="badge is-status"><%= @posts %></span><% end %>
     </a>
-    <a class="button is-muted is-outlined <%= params[:filter] == 'comments' ? 'is-active' : '' %>" href="?filter=comments">
+    <a class="button is-muted is-outlined <%= params[:filter] == 'comments' ? 'is-active' : '' %>" href="?filter=comments" role="button">
         Comments
         <% if @comments > 0 %><span class="badge is-status"><%= @comments %></span><% end %>
     </a>
-    <a class="button is-muted is-outlined <%= params[:filter] == 'flags' ? 'is-active' : '' %>" href="?filter=flags">
+    <a class="button is-muted is-outlined <%= params[:filter] == 'flags' ? 'is-active' : '' %>" href="?filter=flags" role="button">
         Flags
         <% if @flags > 0 %><span class="badge is-status"><%= @flags %></span><% end %>
     </a>
-    <a class="button is-muted is-outlined <%= params[:filter] == 'edits' ? 'is-active' : '' %>" href="?filter=edits">
+    <a class="button is-muted is-outlined <%= params[:filter] == 'edits' ? 'is-active' : '' %>" href="?filter=edits" role="button">
         Edits
         <% if @all_edits > 0 %><span class="badge is-status"><%= @all_edits %></span><% end %>
     </a>
-    <a class="button is-muted is-outlined <%= params[:filter] == 'warnings' ? 'is-active' : '' %>" href="?filter=warnings">
+    <a class="button is-muted is-outlined <%= params[:filter] == 'warnings' ? 'is-active' : '' %>" href="?filter=warnings" role="button">
         Warnings
         <% if @mod_warnings_received > 0 %><span class="badge is-status"><%= @mod_warnings_received %></span><% end %>
     </a>
 </div>
 
-<table class="table is-with-hover is-full-width">
-    <tr>
-        <th>Type</th>
-        <th>On...</th>
-        <th>Excerpt</th>
-        <th>Status</th>
-        <th>Date</th>
-    </tr>
-    <% @items.each do |i| %>
-        <tr class="<%= deleted_item?(i) ? 'deleted-content' : '' %>">
-            <% if i.class == Post %>
-                <td>
-                    <span class="h-fw-bold">
-                        <%= (i.question? ? "Question" : (i.article? ? "Article" : "Answer")) %>
-                    </span>
-                </td>
-                <td>&mdash;</td>
-                <td class="h-fs-caption">
-                    <% if !i.answer? %>
-                    <strong><%= i.title %></strong><br>
-                    <% else %>
-                    <strong>A: <%= i.parent.title %></strong><br>
-                    <% end %>
-                    <%= i.body_plain[0..300] + ((i.body_plain.length > 300) ? "..." : "") %><br>
-                    <%= link_to '(more)', generic_share_link(i)%>
-                </td>
-                <td>&mdash;</td>
-            <% elsif i.class == Comment %>
-                <td>
-                    Comment
-                </td>
-                <td>
-                    <%= link_to "Post #" + i.post.id.to_s, generic_share_link(i.post)%>
-                </td>
-                <td class="h-fs-caption">
-                    <%= i.content[0..300] + ((i.content.length > 300) ? "..." : "") %><br>
-                    <%= link_to '(more)', generic_share_link(i.post) + "#comment-" + i.id.to_s %>
-                </td>
-                <td>&mdash;</td>
-            <% elsif i.class == PostHistory %>
-                <td>
-                    Edit
-                </td>
-                <td>
-                    <%= link_to "Post #" + i.post.id.to_s, generic_share_link(i.post)%>
-                </td>
-                <td class="h-fs-caption">
-                    <% if i.comment %>
-                    <em><%= i.post_history_type.name.gsub("_", " ").capitalize %></em>:<br><%= i.comment %>
-                    <% else %>
-                    <em><%= i.post_history_type.name.gsub("_", " ").capitalize %></em>
-                    <% end %>
-                </td>
-                <td>&mdash;</td>
-            <% elsif i.class == SuggestedEdit %>
-                <td>
-                    Suggested Edit
-                </td>
-                <td>
-                    <%= link_to "Post #" + i.post.id.to_s, generic_share_link(i.post)%>
-                </td>
-                <td class="h-fs-caption">
-                    <em>Suggested edit</em>:<br><%= i.comment %><br><%= link_to '(more)', suggested_edit_url(i.id) %>
-                </td>
-                <td>
-                    <%= (i.pending? ? "pending" : (i.approved? ? "helpful" : "declined")) %>
-                </td>
-            <% elsif i.class == ModWarning %>
-                <td>
-                    <span class="h-fw-bold h-c-red-700">Warning</span>
-                </td>
-                <td>&mdash;</td>
-                <td class="h-fs-caption">
-                    <%= i.body[0..300] + ((i.body.length > 300) ? "..." : "") %>
-                </td>
-                <td>&mdash;</td>
-            <% elsif i.class == Flag %>
-                <td>
-                    Flag
-                </td>
-                <td>
-                    <%= link_to "Post #" + i.post.id.to_s, generic_share_link(i.post)%>
-                </td>
-                <td class="h-fs-caption">
-                    <%= i.reason[0..300] + ((i.reason.length > 300) ? "..." : "") %><br>
-                </td>
-                <td>
-                    <%= i.status || "pending" %>
-                </td>
-            <% else %>
-                <td>
-                    <span class="badge is-tag is-muted">Unknown</span>
-                </td>
-                <td>&mdash;</td>
-                <td><%= i.class %></td>
-                <td>&mdash;</td>
-            <% end %>
-            <td>
-                <span title="<%= i.created_at.iso8601 %>"><%= time_ago_in_words(i.created_at) %> ago</span>
-            </td>
-        </tr>
-    <% end %>
-</table>
+<%= render 'activity_items', mod: true %>
+
diff --git a/app/views/users/index.html.erb b/app/views/users/index.html.erb
index aeea59d5c0901da1b4aacb4231462fbfad1d161a..f0ada48e54367dd7415b5a04755395492cecbeed 100644
--- a/app/views/users/index.html.erb
+++ b/app/views/users/index.html.erb
@@ -15,15 +15,21 @@
 <% end %>
 
 <div class="button-list is-gutterless has-margin-bottom-4">
-  <%= link_to 'Reputation', query_url(sort: 'reputation'),
-      class: "button is-muted is-outlined #{params[:sort] == 'reputation' || (params[:sort].nil? && params[:search].nil?) ? 'is-active' : ''}" %>
-  <%= link_to 'Age', query_url(sort: 'age'), class: "button is-muted is-outlined #{params[:sort] == 'age' ? 'is-active' : ''}" %>
+  <%= link_to 'Reputation', request.params.merge(sort: 'reputation'),
+      class: "button is-muted is-outlined #{params[:sort] == 'reputation' || (params[:sort].nil? && params[:search].nil?) ? 'is-active' : ''}",
+      role: 'button', 'aria-label': 'Sort by reputation' %>
+  <%= link_to 'Age', request.params.merge(sort: 'age'), class: "button is-muted is-outlined #{params[:sort] == 'age' ? 'is-active' : ''}",
+              role: 'button', 'aria-label': 'Sort by age' %>
 </div>
 
-<div class="user-list">
-  <% @users.each do |user| %>
-    <%= render 'user', user: user %>
-  <% end %>
-</div>
+<% if @users.count == 0 %>
+  <p><%= I18n.t('search.no_users') %></p>
+<% else %>
+  <div class="user-list">
+    <% @users.each do |user| %>
+      <%= render 'user', user: user %>
+    <% end %>
+  </div>
+<% end %>
 
 <%= will_paginate @users, renderer: BootstrapPagination::Rails %>
diff --git a/app/views/users/mod.html.erb b/app/views/users/mod.html.erb
index 54761db71c0eddebb94bc217aea311818a26d54c..4f4cd80501558eff6ffe057b0e63fd3717b634ea 100644
--- a/app/views/users/mod.html.erb
+++ b/app/views/users/mod.html.erb
@@ -26,15 +26,15 @@
       after initiating an action.</p>
     <div class="delete-actions">
       <%= link_to 'Destroy user', destroy_user_path(@user.id), remote: true,
-                  method: :delete, class: 'js-destroy-user button is-danger is-filled' %>
+                  method: :delete, class: 'js-destroy-user button is-danger is-filled', role: 'button' %>
       <%= link_to 'Delete community profile', soft_delete_user_path(@user.id, type: 'profile'), remote: true,
-                  method: :delete, class: 'js-soft-delete button is-danger is-filled' %>
+                  method: :delete, class: 'js-soft-delete button is-danger is-filled', role: 'button' %>
       <% if current_user.is_global_moderator || current_user.is_global_admin %>
         <%= link_to 'Delete user network-wide', soft_delete_user_path(@user.id, type: 'user'), remote: true,
-                    method: :delete, class: 'js-soft-delete button is-danger is-filled' %>
+                    method: :delete, class: 'js-soft-delete button is-danger is-filled', role: 'button' %>
       <% end %>
       <% if current_user.is_global_admin %>
-        <%= link_to 'Feed to STAT (180 days)', hellban_user_path(@user), method: :post, class: 'button is-danger is-filled' %>
+        <%= link_to 'Feed to STAT (180 days)', hellban_user_path(@user), method: :post, class: 'button is-danger is-filled', role: 'button' %>
       <% end %>
     </div>
   </div>
diff --git a/app/views/users/network.html.erb b/app/views/users/network.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..cb520a24e30eb8752c81e6238552e9e8c760f485
--- /dev/null
+++ b/app/views/users/network.html.erb
@@ -0,0 +1,10 @@
+<%= render 'tabs', user: @user %>
+
+<h1>Profiles for <%= user_link @user %></h1>
+
+<p>
+  Links to profiles on other communities on this network.
+</p>
+
+<%= render 'network' %>
+
diff --git a/app/views/users/posts.html.erb b/app/views/users/posts.html.erb
index ee63d1bc6601df9d0a3d5ce0cd0342b46c10fb3d..83f02b24e35109b1173bbd057b644bc5b5a1528a 100644
--- a/app/views/users/posts.html.erb
+++ b/app/views/users/posts.html.erb
@@ -6,9 +6,19 @@
 
 <h1>Posts by <%= user_link @user %></h1>
 
-<div class="button-list is-gutterless">
-  <%= link_to 'Score', query_url(sort: 'score'), class: 'button is-muted is-outlined ' + (active_search?('score') ? 'is-active' : '') %>
-  <%= link_to 'Age', query_url(sort: 'age'), class: 'button is-muted is-outlined ' + (active_search?('created_at') ? 'is-active' : '') %>
+<% post_count = @posts.count %>
+<div class="has-color-tertiary-500 category-meta">
+  <span title="<%= post_count %> posts">
+    <%= short_number_to_human post_count, precision: 1, significant: false %>
+    <%= 'post'.pluralize(post_count) %> 
+  </span>
+
+  <div class="button-list is-gutterless has-margin-2">
+    <%= link_to 'Score', request.params.merge(sort: 'score'), class: 'button is-muted is-outlined ' + (active_search?('score') ? 'is-active' : ''),
+                role: 'button', 'aria-label': 'Sort by score' %>
+    <%= link_to 'Age', request.params.merge(sort: 'age'), class: 'button is-muted is-outlined ' + (active_search?('created_at') ? 'is-active' : ''),
+                role: 'button', 'aria-label': 'Sort by age' %>
+  </div>
 </div>
 
 <div class="item-list">
@@ -17,4 +27,4 @@
   <% end %>
 </div>
 
-<%= will_paginate @posts, renderer: BootstrapPagination::Rails %>
\ No newline at end of file
+<%= will_paginate @posts, renderer: BootstrapPagination::Rails %>
diff --git a/app/views/users/preferences.html.erb b/app/views/users/preferences.html.erb
index 6d9825d4109777ed44e8ecc214d7a98452418c73..e02038ffd82aa83e47fe7ddacd03a8dcf4f6576f 100644
--- a/app/views/users/preferences.html.erb
+++ b/app/views/users/preferences.html.erb
@@ -2,7 +2,7 @@
 
 <h1>Preferences</h1>
 <p>
-  Manage your user preferences here. Changes are automatically saved once you move away from the input field.
+  Manage your user preferences here. Changes are automatically saved.
 </p>
 
 <p><%= link_to 'Manage Email Subscriptions', subscriptions_path %></p>
diff --git a/app/views/users/qr_login_code.html.erb b/app/views/users/qr_login_code.html.erb
index 9a02d5a896ac8846bac4ca6b0c6c70086cbeb428..3ad98fd75291e939aac04a044a69fa7502d64dee 100644
--- a/app/views/users/qr_login_code.html.erb
+++ b/app/views/users/qr_login_code.html.erb
@@ -4,8 +4,20 @@
   follow the URL to log in there.
 </p>
 
-<div class="has-text-align-center">
-  <svg height="<%= @qr_code.qrcode.module_count * 4 %>" width="<%= @qr_code.qrcode.module_count * 4 %>">
-    <%= raw(@qr_code.as_svg(standalone: false, module_size: 4)) %>
-  </svg>
-</div>
\ No newline at end of file
+<div class="notice is-warning">
+  <p><i class="fas fa-exclamation-triangle"></i> <strong>Caution</strong></p>
+  <p>
+    The QR code below, when scanned, provides immediate access to your <%= t 'platform.network_name' %> account,
+    without asking for your password again. This makes it easier to sign in on your phone, but make sure nobody's
+    looking over your shoulder! Take extra care in public places.
+  </p>
+</div>
+
+<details>
+  <summary>Show QR code</summary>
+  <div class="has-text-align-center">
+    <svg height="<%= @qr_code.qrcode.module_count * 4 %>" width="<%= @qr_code.qrcode.module_count * 4 %>">
+      <%= raw(@qr_code.as_svg(standalone: false, module_size: 4)) %>
+    </svg>
+  </div>
+</details>
diff --git a/app/views/users/sessions/verify_2fa.html.erb b/app/views/users/sessions/verify_2fa.html.erb
index 999ec617acb00f80af01303e7687207a68ee701c..88deb61f2f46c22003944fe2192c1495e9c31921 100644
--- a/app/views/users/sessions/verify_2fa.html.erb
+++ b/app/views/users/sessions/verify_2fa.html.erb
@@ -1,4 +1,5 @@
 <h1>Two-factor authentication</h1>
+
 <p>
   Your account has two-factor authentication enabled. Enter a code from your authenticator app here.
 </p>
@@ -8,7 +9,7 @@
 
   <div class="form-group">
     <%= label_tag 'code', 'Code', class: 'form-element' %>
-    <%= number_field_tag 'code', '', class: 'form-element', autocomplete: 'one-time-code' %>
+    <%= text_field_tag 'code', '', class: 'form-element', autocomplete: 'one-time-code', autofocus: 'true' %>
   </div>
 
   <div class="actions">
diff --git a/app/views/users/show.html.erb b/app/views/users/show.html.erb
index 9545ef8c154cece8bcadd498774ed9da4e0874ca..1ce3f9975a0ad04587cf86e7344b13a4f7091bee 100644
--- a/app/views/users/show.html.erb
+++ b/app/views/users/show.html.erb
@@ -1,3 +1,5 @@
+<%= render 'tabs', user: @user %>
+
 <% content_for :title, "User #{rtl_safe_username(@user)}" %>
 
 <% if moderator? && deleted_user?(@user) %>
@@ -18,23 +20,44 @@
   </div>
 <% end %>
 
-<%= render 'tabs', user: @user %>
-
 <div class="grid <%= deleted_user?(@user) ? 'deleted-content' : '' %>">
   <div class="grid--cell is-9-lg is-12">
-    <div class="h-p-6 h-p-t-2">
+    <div class="h-p-0 h-p-t-0">
       <div class="profile-text">
-      <p>
-        <% if @user.website.present? %>
-          <span class="h-m-r-4">
-            <i class="fas fa-link"></i> <%= link_to @user.website_domain, @user.website, rel: 'nofollow' %>
-          </span>
-        <% end %>
-        <% if @user.twitter.present? %>
-          <span class="h-m-r-4">
-            <i class="fab fa-twitter"></i> <%= link_to @user.twitter, "https://twitter.com/#{@user.twitter}" %>
-          </span>
+
+      <% effective_profile = raw(sanitize(@user.profile&.strip || '', scrubber: scrubber)) %>
+      <% if effective_profile.blank? %>
+        <p class="is-lead">A quiet enigma. We don't know anything about <span dir="ltr"><%= rtl_safe_username(@user) %></span> yet.</p>
+      <% elsif !user_signed_in? && !@user.community_user.privilege?('unrestricted') %>
+        <%= sanitize(effective_profile, attributes: %w()) %>
+      <% else %>
+        <%= effective_profile %>
+      <% end %>
+      </div>
+
+      <% unless !user_signed_in? && !@user.community_user.privilege?('unrestricted') %>
+        <% if @user.valid_websites_for.size.positive? %>
+        <div>
+          <p><strong>Extra fields</strong></p>
+          <table class="table is-with-hover">
+            <% @user.valid_websites_for.each do |w| %>
+            <tr>
+              <td><%= w.label %></td>
+              <td>
+              <% if w.url[0,4] == 'http' %>
+                <%= link_to w.url, w.url, rel: 'nofollow' %>
+              <% else %>
+              <%= w.url %>
+              <% end %>
+              </td>
+            </tr>
+            <% end %>
+          </table>
+        </div>
         <% end %>
+      <% end %>
+     
+      <p>
         <% if @user.discord.present? %>
           <span class="h-m-r-4">
             <i class="fab fa-discord h-m-r-1"></i> <%= @user.discord %>
@@ -42,13 +65,6 @@
         <% end %>
       </p>
 
-      <% if @user.profile.nil? || @user.profile.blank? %>
-        <p class="is-lead">A quiet enigma. We don't know anything about <span dir="ltr"><%= rtl_safe_username(@user) %></span> yet.</p>
-      <% else %>
-        <%= raw(sanitize(@user.profile, scrubber: scrubber)) %>
-      <% end %>
-      </div>
-
       <div class="button-list h-p-2">
         <% if user_signed_in? %>
           <%= link_to new_subscription_path(type: 'user', qualifier: @user.id, return_to: request.path), class: "button is-outlined is-small" do %>
@@ -56,14 +72,18 @@
           <% end %>
         <% end %>
         <% if current_user&.is_moderator %>
-          <a href="<%= mod_user_path(@user) %>" class="button is-danger is-outlined is-small" data-drop="#mod-tools-drop"><i class="fas fa-shield-alt"></i> Moderator Tools</a>
+          <a href="<%= mod_user_path(@user) %>" class="button is-danger is-outlined is-small" data-drop="#mod-tools-drop"><i class="fas fa-shield-alt"></i> Moderator Tools <% if @user.community_user.mod_warnings&.size.positive? %> (<%= pluralize(@user.community_user.mod_warnings.count, 'message') %>) <% end %></a> 
           <div class="droppanel" id="mod-tools-drop">
             <div class="droppanel--header">quick actions</div>
             <div class="droppanel--menu">
               <a href="/users/<%= @user.id %>/mod/activity-log">full activity log</a>
               <a href="/users/<%= @user.id %>/mod/annotations">annotations on user</a>
               <a href="/users/<%= @user.id %>/mod/privileges">privileges</a>
-              <a href="/warning/log/<%= @user.id %>">warnings and suspensions sent to user <% if @user.community_user.suspended? %><em>(includes lifting the suspension)</em><% end %></a>
+              <a href="/warning/log/<%= @user.id %>">warnings and suspensions sent to user
+		<% if @user.community_user.suspended? %><em>(includes lifting the suspension)</em>
+		<% elsif @user.community_user.mod_warnings&.size.positive? %>
+		(latest <%= time_ago_in_words(@user.community_user.latest_warning) %> ago)
+		<% end %></a>
               <a href="/warning/new/<%= @user.id %>">warn or suspend user</a>
             </div>
             <div class="h-m-t-6">
@@ -71,14 +91,21 @@
             </div>
           </div>
         <% end %>
-        <% if current_user&.id == @user.id %>
+        <% if current_user&.same_as?(@user) %>
           <%= link_to qr_login_code_path, class: 'button is-outlined is-small' do %>
             <i class="fas fa-mobile-alt"></i> Mobile Sign In
           <% end %>
         <% end %>
       </div>
 
-      <h2 class="user-profile-heading">Posts</h2>
+      <div class="user-profile-heading-container">
+        <h2 class="user-profile-heading">Posts</h2>
+        <% if @posts.size > 0 %>
+          <%= link_to user_posts_path(@user), class: "button is-muted", 'aria-label': "View all posts by #{rtl_safe_username(@user)}" do %>
+            See all <%= @total_post_count %> &raquo;
+          <% end %>
+        <% end %>
+      </div>
       <% if @posts.size == 0 %>
         <p><span dir="ltr"><%= rtl_safe_username(@user) %></span> hasn't posted anything yet.</p>
       <% else %>
@@ -87,8 +114,8 @@
             <%= render 'posts/type_agnostic', post: a, show_type_tag: true, show_category_tag: true %>
           <% end %>
         </div>
-        <%= link_to user_posts_path(@user), class: "button is-muted" do %>
-          See all &raquo;
+        <%= link_to user_posts_path(@user), class: "button is-muted", 'aria-label': "View all posts by #{rtl_safe_username(@user)}" do %>
+          See all <%= @total_post_count %> &raquo;
         <% end %>
       <% end %>
     </div>
@@ -137,12 +164,21 @@
         <td><%= @user.metric 'E' %></td>
       </tr>
     </table>
+    
+    <% if current_user&.id == @user.id || current_user&.is_moderator %>
+    <table class="table is-full-width">
+      <tr>
+        <td>User since <%= @user.created_at %></td>
+      </tr>
+    </table>
+    <% end %>
 
     <% unless @abilities.empty? %>
       <h2>Earned Abilities</h2>
 
       <div class="widget">
         <% @abilities.each do |a| %>
+        <% if @user.privilege?(a.internal_id) %>
           <div class="widget--body" title="<%= a.summary %>">
             <i class="fa-fw fas fa-<%= a.icon %>"></i>
             <a href="<%= ability_url(a.internal_id, for: @user.id) %>">
@@ -150,13 +186,14 @@
             </a>
           </div>
         <% end %>
+        <% end %>
         <div class="widget--footer">
           <% if current_user&.id == @user.id %>
-           <%= link_to abilities_path, class: 'has-font-weight-bold' do %>
+           <%= link_to abilities_path, class: 'has-font-weight-bold', 'aria-label': 'View your abilities' do %>
              Abilities &raquo;
            <% end %>
           <% else %>
-           <%= link_to abilities_path(for: @user.id), class: 'has-font-weight-bold' do %>
+           <%= link_to abilities_path(for: @user.id), class: 'has-font-weight-bold', 'aria-label': "View abilities of #{rtl_safe_username(@user)}" do %>
              Abilities &raquo;
            <% end %>
           <% end %>
@@ -259,7 +296,8 @@
         <td colspan="2">Count</td>
         <td>
           <% if current_user&.id == @user.id || moderator? %>
-            <%= link_to @user.flags.count, flag_history_path(@user.id), class: 'is-muted' %>
+            <%= link_to @user.flags.count, flag_history_path(@user.id), class: 'is-muted',
+                        'aria-label': "View flag history for #{@user.flags.count} flags" %>
           <% else %>
             <%= @user.flags.count %>
           <% end %>
diff --git a/app/views/users/vote_summary.html.erb b/app/views/users/vote_summary.html.erb
index e6115fa7961e561e8eab285074ef56525164bc60..d1cf7dcaa616baea9c4419a60499a877c5d89166 100644
--- a/app/views/users/vote_summary.html.erb
+++ b/app/views/users/vote_summary.html.erb
@@ -2,12 +2,19 @@
 
 <h1>Vote summary for <%= user_link @user %></h1>
 
-<p class="is-lead">A daily summary of votes you have received for your posts.</p>
+<p class="is-lead">A daily summary of votes received for posts.</p>
 
-<% @votes.each do |day, vote_list| %>
+<% @votes.each do |day, vote_list, net_votes| %>
 <details class="user-vote-summary" <%= 'open' unless day < SiteSetting['VoteSummaryAutoExpandLastNrOfDays'].days.ago%>>
 <summary>
-<h2><%= day.strftime('%b %e, %Y') %></h2>
+<h2><%= day.strftime('%b %e, %Y') %>:
+<% if net_votes > 0 %>
+  <span class="has-color-green-800 has-font-weight-bold">+<%= net_votes %></span>
+<% elsif net_votes < 0 %>
+  <span class="has-color-red-800 has-font-weight-bold"><%= net_votes %></span>
+<% else %>
+  <span class="has-color-tertiary-600">0</span>
+<% end %></h2>
 </summary>
 <div class="item-list">
     <% vote_list.each do |post, vote_data| %>
@@ -33,15 +40,21 @@
         </div>
         <div class="item-list--text-value is-primary grid">
             <div class="grid grid--cell is-12">
-                <div class="grid--cell is-flexible">
-                    <%= link_to generic_share_link(post) do %>
-                        <%= post.post_type.is_top_level ? post.title : post.parent.title %>
-                    <% end %>
-                </div>
-                <div class="grid--cell">
-                    <span class="badge is-tag is-muted is-filled"><%= post.post_type.name %></span>
-                    <a href="<%= category_path(post.category) %>" class="badge is-tag is-muted"><%= post.category.name %></a>
-                </div>
+                <% if post.present? %>
+                    <div class="grid--cell is-flexible">
+                            <%= link_to generic_share_link(post) do %>
+                                <%= post.post_type.is_top_level ? post.title : post.parent.title %>
+                            <% end %>
+                    </div>
+                    <div class="grid--cell">
+                        <span class="badge is-tag is-muted is-filled"><%= post.post_type.name %></span>
+                        <a href="<%= category_path(post.category) %>" class="badge is-tag is-muted"><%= post.category.name %></a>
+                    </div>
+                <% else %>
+                    <div class="grid--cell">
+                        <%= I18n.t('votes.summary.post_missing') %>
+                    </div>
+                <% end %>
             </div>
         </div>
     </div>
diff --git a/config/application.rb b/config/application.rb
index e5c0a400456e88b8e230f8430e880e50a73f7a14..39ebcb6800c92d11265762a22a588e42ad359d85 100644
--- a/config/application.rb
+++ b/config/application.rb
@@ -39,5 +39,10 @@ module Qpixel
       Devise::Mailer.helper :users
       Devise::Mailer.layout 'devise_mailer'
     end
+
+    console do
+      require 'console_extension'
+      include ConsoleExtension
+    end
   end
 end
diff --git a/config/attribute-map.yml b/config/attribute-map.yml
new file mode 100644
index 0000000000000000000000000000000000000000..1f3bc4eea22adc13374e448e6d192943aa8625cc
--- /dev/null
+++ b/config/attribute-map.yml
@@ -0,0 +1,18 @@
+# Add your SAML attribute mapping here.
+#
+# Required:
+#   '<email attribute>': 'saml_init_email'
+#   '<uid attribute>': 'saml_init_identifier'
+#   '<full name attribute>': 'username' (user cannot change it) OR 'saml_init_username_no_update' (only set on first login, user can change)
+#
+# If email is the unique identifier, map the email attribute to 'saml_init_email_and_identifier' instead of the above.
+# In that case you also need to update devise.rb and set `config.saml_default_user_key = :'saml_init_email_and_identifier'`.
+#
+'uid': 'saml_init_identifier'
+'urn:mace:dir:attribute-def:uid': 'saml_init_identifier'
+'mail': 'saml_init_email'
+'urn:mace:dir:attribute-def:mail': 'saml_init_email'
+'email': 'saml_init_email'
+'urn:mace:dir:attribute-def:email': 'saml_init_email'
+'displayName': 'saml_init_username_no_update'
+'urn:mace:dir:attribute-def:displayName': 'saml_init_username_no_update'
diff --git a/config/config/preferences.yml b/config/config/preferences.yml
index 36be0760ac0f11a14ff96504e1c22f091ac8fbb0..5d72b1204b6bebdde2d1b8c2ad78a3a188c745ce 100644
--- a/config/config/preferences.yml
+++ b/config/config/preferences.yml
@@ -10,13 +10,17 @@
 #     - if
 #     - type is choice
 #   default: string default value | ~
-#   community: true | false  # optional, default false (global)
+##  Either community, global or category should be set to true, all default to false
+#   community: true | false
+#   global: true | false
+#   category: true | false
 
 keyboard_tools:
   type: boolean
   description: >
     Enable keyboard shortcuts. Press ? for a list of shortcuts.
   default: 'true'
+  global: true
 
 autosave:
   type: choice
@@ -67,9 +71,16 @@ auto_follow_comment_threads:
   description: >
     Automatically follow any comment thread you participate in.
   default: 'true'
+  global: true
 
 sticky_header:
   type: boolean
   description: >
     Make the top navigation bar sticky.
-  default: false
\ No newline at end of file
+  default: 'false'
+  global: true
+
+default_filter_name:
+  type: ~
+  default: none
+  category: true
\ No newline at end of file
diff --git a/config/credentials.yml.enc b/config/credentials.yml.enc
index 8fe95183b851389459dd8da0a9b2a19c25dd55a7..fc62796fcc8e63c53d74510fe7b843992bc002fd 100644
--- a/config/credentials.yml.enc
+++ b/config/credentials.yml.enc
@@ -1 +1 @@
-7h9/PUnrx1KYHkSB0ye+1GWX4RHzJoc7XbutJDscZv6hISSAO/wjQBTvTGXEJ2yoJ3nWeJw2ePHmEy6Rexw2ycLvglexAJ5PAdO3jdh2d43k1+zUd50Mntz+vuUxCiRJ1lYogKZ0/iyn7i2cf1Xy1ZW05lNlgluzVo8dHvd7gH9A2Xsdd2kouKp2MvU7Kxxd4vnNor7PF53MhfxDOsEgQfmrcBU4fRbQQCTDx7SOXIiTLyvA3EgC7i+ECagetwSKyrvvUuYXyFfsgtXZLzn3w5VtzWBGULVVKy0uMvh3RlPkdzgrDW+sGQtHCi7YwM5CJPz6wah9ypjDnhlQXbcMg9qjcnatMF6YYDg8Jf4NkO0gSHSpzBZweTydb1kWWql5m40bqtrOM6yxkyBeGA2jQRfQ+zSsltOxwGa6ZeDfKD76z+5fFGtl3apIPG91KkFpVZrm8LOJdsKV9P/JFFzXTAMyIRkBHAiacwY/qVFPxzI4exEJUI0R3H2RFgD6tADJOSHAcVDUbkxt56ARUk0DpKq/7/azLdM+1NOwmITO/Nn6d1pjB7/rKjHEjSCu7hAtvXL+Ou9adKDRdxAnM+tOCwc5olQcPzepeiVNuO6lBqhSVE/huu8hQVemBlNIjWhqC1TjSN/dvgUdve6rap/d3Y66T5MEmKFsPzV/iLVkEKZQ7/lHRgZMidOHoemg0FWbShhdDjZWFR8Q2K9d7vhPuZj7DSW0inTgCMayHf/ULR8QGRxCySusCK9rU3BsxKmWHiX+6F0EN0CFQRGEXzIKIi6U7RNh86b8apskPFFgtPUWsCufP1Z1NWnxJU/8hb9+ZhEYbs3z54+ssR9jw+yHvEFg63R6THGwcF/kXhXomRAjNYt/bJV2aAuC4w39YMQakctrgWYcA8PuBXc8tDpBZjA8+yHcxDTOUb0veaZnLNfj0+D54Q6zwO6iV+DdTeOL8JWKh52erhPyh8u6mYRS88CP9R1Ep/LqbtG1om4nPmAsVfuMBbyih9XJxvGwknZEmgUo8w+pGhff2JXCsACnKq66cy6VKWQqiUDTZKo+hSXEc/dRqSTOSTq26ZN5ctbbXE57WNydvQ==--PEsrFqN16qSIVkzT--nvIHvNPjy9ZEHA3g3EuNjg==
\ No newline at end of file
+SA7h7Xm7xF3AEvje/xZiPMzem0wh2CysuvvB+iM6R0Epkg+z+vZG2oxI/9tVzDGELWtwoNDPeqNa0bUbGjo6gt7xyEpmQbivEVKnwcbVK+1bxPHKke+hveVEUhmY/axFIIpty+VTnZVnrJkZ35/rW9egcRr5O2HikuAuBz8KrYLvB2w3h8EdmcIbbfjMvKWlUVvc8D9t3v+k56B1CbTghTmGeG0NLijWAA+U+pDmErPOR9JUAVY/nbF2/T8aQ5Szameuoj88Pk5P6alFDHwlWUSHhuEY5Z7A5LKM1NtzlTGc+CAcTVzKvx6DCg8WUJugy3uS7Bpp6uNj9TvJoZzrb0i5G1nDmIl7tK2GCNclp3JFEklrUJIrF4asRuDQUzdvuPeTEXNyi2cc5d6hSBG+MwEvG90P6n0td5+gv+rr1FCKuzP5ftyxqPsnvZuBKRxNSw0+VTTXhTO73jcwttHYrD9Jwc9tVbCVqegwsEO0dB0sNL6M89rudDMbpPnz7rJXr6lXD4Az3qJ/0N3ubYDwiuGOk7UHDntm4Id4BFDm5qeLeOUg1n2PDhsQUWh2Lfnk7ADvjQMinfXhdAQkYVim9sDtNO/Grn9xSXfCn3vs1xOdTtKLb23+EIxO2STtkHTsJG5XvxC2jmzL5/zQx/vO/zepHom7POdso4Ygoqw9oEttm/KPG85lcX0edXv/l1i5JBMMbzaQgd1PL4uWwrv+3LoSq+vQORijR50OiLdlMZ5IFn+QagJEdeVtUiCbYA3AhUK/FcH3L5QtSGNObEUrUJ4fVuWP6X8E34vd+DoY++D6PtG89SFIGBRmQ23cT111uIw3p2A9HP4N9B/uNKwxAnr4S1he6EkoA5CtDE0NZufy25s0IVf//FDWXS8FDg3ZhtOUeMWB0dew9VeCCWwr8RrLu4jP46IvVqfFeiVWf7gJcenEJ0QuqPlBeukSpgNxckDjwijRz5obzeS9ZAWYgCZg2kg/6CmuIM2VjkWMJ13uKYkFgpheoNsk1jZwH4+0fdm+Zc+fE4KQFcSkkazea/qEanN2/tDo6pRowSlRpi/nWJqHxcu87QMm7cBq5/E9DNjwyKOPmgr7c2NzDrlPYMRgzdcauPz6Y4qfcg9C6lKwJIo18VQhrqwh7wW+nNTov3Shw/yclkVrxnHTQbyH3mOstJsVPpYeYewxOQj/gikvvkx/lUeUnyo2/Ebff2nJCbnLmJ8/7Vwz9c/yvg==--d1HPd+AnhFU1woqD--z2ikkRr6d5WTzC7X5xOVPA==
\ No newline at end of file
diff --git a/config/environments/development.rb b/config/environments/development.rb
index 3586b8d5134d4a77bb0f3c2d3726bd809e9e480a..f6fa52367be15d0883e854f127feb676e468bfb7 100644
--- a/config/environments/development.rb
+++ b/config/environments/development.rb
@@ -34,6 +34,9 @@ Rails.application.configure do
   # Store uploaded files on the local file system (see config/storage.yml for options).
   config.active_storage.service = :local
 
+  # Allow ngrok connections to dev server
+  config.hosts << /[a-z0-9\-.]+\.ngrok-free\.app/
+
   # Don't care if the mailer can't send.
   config.action_mailer.raise_delivery_errors = false
   config.action_mailer.delivery_method = :ses
@@ -77,12 +80,15 @@ Rails.application.configure do
   config.i18n.raise_on_missing_translations = true
 
   config.action_mailer.delivery_method = :letter_opener_web
-  config.action_mailer.default_url_options = { host: 'meta.codidact.com', protocol: 'https' }
+
+  config.action_mailer.default_url_options = { 
+    host: 'meta.codidact.com', protocol: ENV['MAILER_PROTOCOL'] || 'https'
+  }
 
   # Ensure docker ip added to allowed, given that we are in container
   if File.file?('/.dockerenv') == true
     host_ip = `/sbin/ip route|awk '/default/ { print $3 }'`.strip
-    config.web_console.allowed_ips << host_ip
+    config.web_console.permissions = host_ip
 
     # ==> Configuration for :confirmable
     # A period that the user is allowed to access the website even without
diff --git a/config/environments/production.rb b/config/environments/production.rb
index 7a7b6d2d4a42de6a8afcc240e99c9e218c257c81..8884d9834a0ccfed27d538a22cb28d8d3dfbaaa0 100644
--- a/config/environments/production.rb
+++ b/config/environments/production.rb
@@ -93,9 +93,14 @@ Rails.application.configure do
   # config.logger = ActiveSupport::TaggedLogging.new(Syslog::Logger.new 'app-name')
 
   config.action_mailer.delivery_method = :ses
-  config.action_mailer.default_url_options = { host: 'meta.codidact.com', protocol: 'https' }
+  config.action_mailer.default_url_options = { 
+    host: 'meta.codidact.com',
+    protocol: ENV['MAILER_PROTOCOL'] || 'https'
+  }
   config.action_mailer.asset_host = 'https://meta.codidact.com'
 
   # Do not dump schema after migrations.
   config.active_record.dump_schema_after_migration = false
+
+  config.active_job.queue_adapter = :async
 end
diff --git a/config/environments/test.rb b/config/environments/test.rb
index 8bd2cfad8a20dcb3ba63cd1efd3196a7fb39695c..8212caac092de614c6d2fd409ad9ee9bea295245 100644
--- a/config/environments/test.rb
+++ b/config/environments/test.rb
@@ -56,7 +56,10 @@ Rails.application.configure do
   # ActionMailer::Base.deliveries array.
   config.action_mailer.delivery_method = :test
 
-  config.action_mailer.default_url_options = { host: 'test.host' }
+  config.action_mailer.default_url_options = { 
+    host: 'test.host',
+    protocol: ENV['MAILER_PROTOCOL'] || 'https'
+  }
 
   # Print deprecation notices to the stderr.
   config.active_support.deprecation = :stderr
diff --git a/config/initializers/devise.rb b/config/initializers/devise.rb
index 2b2af4c0525ddbe9bc004152b68618bc1f65d1d7..9eadc984e403b79d0afd1838b2eebb349f018983 100644
--- a/config/initializers/devise.rb
+++ b/config/initializers/devise.rb
@@ -270,4 +270,78 @@ Devise.setup do |config|
   # When using OmniAuth, Devise cannot automatically set OmniAuth path,
   # so you need to do it manually. For the users scope, it would be:
   # config.omniauth_path_prefix = '/my_engine/users/auth'
+
+  # Reference https://github.com/apokalipto/devise_saml_authenticatable
+  # ==> Configuration for :saml_authenticatable
+
+  # Add prefix to saml routes
+  config.saml_route_helper_prefix = 'saml'
+
+  # Create user if the user does not exist.
+  config.saml_create_user = true
+
+  # Update the attributes of the user after a successful login. Set this to
+  # false if you don't want to update the users email address from saml when it
+  # is changed.
+  config.saml_update_user = true
+
+  # Set the default user key. The user will be looked up by this key. Make
+  # sure that the (mapped) Authentication Response includes the attribute.
+  #
+  # In the attribute map we register this as saml_init_identifier
+  # (the method to call), so this is also how we have to refer to it here.
+  config.saml_default_user_key = :'saml_init_identifier'
+
+  # Load by sso_profile instead of by email address.
+  config.saml_resource_locator = Proc.new do |model, saml_response, auth_value|
+    # You can use saml_response here to access other attributes than the saml_default_user_key indicated above if need be.
+    SsoProfile.includes(:user).find_by(saml_identifier: auth_value)&.user
+  end
+
+  # Optional. This stores the session index defined by the IDP during login.  If provided it will be used as a salt
+  # for the user's session to facilitate an IDP initiated logout request.
+  config.saml_session_index_key = :session_index
+
+  # You can set this value to use Subject or SAML assertation as info to which email will be compared.
+  # If you don't set it then email will be extracted from SAML assertation attributes.
+  config.saml_use_subject = false
+
+  # You can support multiple IdPs by setting this value to a class that implements a #settings method which takes
+  # an IdP entity id as an argument and returns a hash of idp settings for the corresponding IdP.
+  config.idp_settings_adapter = nil
+
+  # You provide you own method to find the idp_entity_id in a SAML message in the case of multiple IdPs
+  # by setting this to a custom reader class, or use the default.
+  # config.idp_entity_id_reader = DeviseSamlAuthenticatable::DefaultIdpEntityIdReader
+
+  # You can set a handler object that takes the response for a failed SAML request and the strategy,
+  # and implements a #handle method. This method can then redirect the user, return error messages, etc.
+  # config.saml_failed_callback = nil
+
+  # Add your SAML configuration from your IDP here.
+  #
+  # Attributes must be mapped in the attribute-map.yml
+  #
+  # For certificates and keys, you can use
+  #   File.read('path/to/certificate')
+  # instead of providing the certificate/key in a string.
+  #
+  # config.saml_configure do |settings|
+  #   settings.assertion_consumer_service_url     = '<http(s)-site-address-here>/users/saml/auth'
+  #   settings.assertion_consumer_service_binding = 'urn:oasis:names:tc:SAML:2.0:bindings:HTTP-POST'
+  #   settings.name_identifier_format             = 'urn:oasis:names:tc:SAML:2.0:nameid-format:persistent'
+  #   settings.security[:want_assertions_signed]  = true
+  #   settings.security[:metadata_signed]         = true
+  #   settings.security[:authn_requests_signed]   = true
+  #   settings.force_authn                        = !Rails.env.production?
+  #   settings.protocol_binding                   = "urn:oasis:names:tc:SAML:2.0:bindings:HTTP-POST"
+  #   settings.passive                            = false
+  #   settings.issuer                             = '<http(s)-site-address-here>/users/saml/metadata'
+  #   settings.idp_slo_target_url                 = '<single logout service url of IDP>'
+  #   settings.idp_sso_target_url                 = '<single sign on service url of IDP>'
+  #   settings.idp_entity_id                      = '<metadata url of IDP>'
+  #   settings.idp_cert                           = '<certificate of IDP>'
+  #   settings.certificate                        = '<your (self-signed) certificate>'
+  #   settings.private_key                        = '<your key (for your certificate)>'
+  # end
 end
diff --git a/config/initializers/devise_example.rb b/config/initializers/devise_example.rb
index ef47ca5e195da06cd0f44f3f26e7166f4663de3d..f8863061bb2474ff900c563d27cc8490f100d777 100644
--- a/config/initializers/devise_example.rb
+++ b/config/initializers/devise_example.rb
@@ -265,4 +265,75 @@ Devise.setup do |config|
   # When using OmniAuth, Devise cannot automatically set OmniAuth path,
   # so you need to do it manually. For the users scope, it would be:
   # config.omniauth_path_prefix = '/my_engine/users/auth'
+
+  # Reference https://github.com/apokalipto/devise_saml_authenticatable
+  # ==> Configuration for :saml_authenticatable
+
+  # Add prefix to saml routes
+  config.saml_route_helper_prefix = 'saml'
+
+  # Create user if the user does not exist.
+  config.saml_create_user = true
+
+  # Update the attributes of the user after a successful login.
+  config.saml_update_user = true
+
+  # Set the default user key. The user will be looked up by this key. Make
+  # sure that the Authentication Response includes the attribute.
+  #
+  # In the attribute map we register this as saml_init_identifier (the method to call), so this is also how we have to refer to it here.
+  config.saml_default_user_key = :'saml_init_identifier'
+
+  # Load by sso_profile instead of by email address.
+  config.saml_resource_locator = Proc.new do |model, saml_response, auth_value|
+    # You can use saml_response here to access other attributes than the saml_default_user_key indicated above if need be.
+    SsoProfile.includes(:user).find_by(saml_identifier: auth_value)&.user
+  end
+
+  # Optional. This stores the session index defined by the IDP during login.  If provided it will be used as a salt
+  # for the user's session to facilitate an IDP initiated logout request.
+  config.saml_session_index_key = :session_index
+
+  # You can set this value to use Subject or SAML assertation as info to which email will be compared.
+  # If you don't set it then email will be extracted from SAML assertation attributes.
+  config.saml_use_subject = false
+
+  # You can support multiple IdPs by setting this value to a class that implements a #settings method which takes
+  # an IdP entity id as an argument and returns a hash of idp settings for the corresponding IdP.
+  config.idp_settings_adapter = nil
+
+  # You provide you own method to find the idp_entity_id in a SAML message in the case of multiple IdPs
+  # by setting this to a custom reader class, or use the default.
+  # config.idp_entity_id_reader = DeviseSamlAuthenticatable::DefaultIdpEntityIdReader
+
+  # You can set a handler object that takes the response for a failed SAML request and the strategy,
+  # and implements a #handle method. This method can then redirect the user, return error messages, etc.
+  # config.saml_failed_callback = nil
+
+  # Add your SAML configuration from your IDP here.
+  #
+  # Attributes must be mapped in the attribute-map.yml
+  #
+  # For certificates and keys, you can use
+  #   File.read('path/to/certificate')
+  # instead of providing the certificate/key in a string.
+  #
+  # config.saml_configure do |settings|
+  #   settings.assertion_consumer_service_url     = '<http(s)-site-address-here>/users/saml/auth'
+  #   settings.assertion_consumer_service_binding = 'urn:oasis:names:tc:SAML:2.0:bindings:HTTP-POST'
+  #   settings.name_identifier_format             = 'urn:oasis:names:tc:SAML:2.0:nameid-format:persistent'
+  #   settings.security[:want_assertions_signed]  = true
+  #   settings.security[:metadata_signed]         = true
+  #   settings.security[:authn_requests_signed]   = true
+  #   settings.force_authn                        = !Rails.env.production?
+  #   settings.protocol_binding                   = "urn:oasis:names:tc:SAML:2.0:bindings:HTTP-POST"
+  #   settings.passive                            = false
+  #   settings.issuer                             = '<http(s)-site-address-here>/users/saml/metadata'
+  #   settings.idp_slo_target_url                 = '<single logout service url of IDP>'
+  #   settings.idp_sso_target_url                 = '<single sign on service url of IDP>'
+  #   settings.idp_entity_id                      = '<metadata url of IDP>'
+  #   settings.idp_cert                           = '<certificate of IDP>'
+  #   settings.certificate                        = '<your (self-signed) certificate>'
+  #   settings.private_key                        = '<your key (for your certificate)>'
+  # end
 end
diff --git a/config/initializers/zz_cache_setup.rb b/config/initializers/zz_cache_setup.rb
index d7a9fe5f0e2429eae562ceba5bf04494d657f869..ae1e14a404a90107425bf15c0d3ab9800192cbcd 100644
--- a/config/initializers/zz_cache_setup.rb
+++ b/config/initializers/zz_cache_setup.rb
@@ -1,3 +1,5 @@
 Rails.cache.persistent 'current_commit', clear: true do
-  [`git rev-parse HEAD`.strip, `git log -1 --date=iso --pretty=format:%cd`.strip]
+  commit_sha = `git rev-parse HEAD`.strip
+  commit_date = `git log -1 --date=iso-strict --pretty=format:%cd`.strip
+  [commit_sha, commit_date]
 end
diff --git a/config/initializers/zz_codidact_sites.rb b/config/initializers/zz_codidact_sites.rb
index b745b63de8f4d4c3529444aa00df82f851535d3b..ac4511296cb61b36f94edcfbf37d3b31aa73cb67 100644
--- a/config/initializers/zz_codidact_sites.rb
+++ b/config/initializers/zz_codidact_sites.rb
@@ -4,12 +4,18 @@ Rails.cache.persistent 'codidact_sites', clear: true do
   if Rails.env.development? || Rails.env.test?
     []
   else
-    response = Net::HTTP.get_response(URI('https://codidact.com/communities.json'))
+    uri = URI('https://codidact.com/communities.json')
+    req = Net::HTTP::Get.new(uri)
+    req['Authorization'] = "Bearer #{Rails.application.credentials.cf_bot_key}"
+    response = Net::HTTP.start(uri.hostname, uri.port, use_ssl: true) do |http|
+      http.request(req)
+    end
+
     if response.code == '200'
       JSON.parse(response.body)
     else
-      Rails.logger.fatal "Couldn't fetch Codidact sites: response code #{response.code}"
-      exit 255
+      Rails.logger.error "Couldn't fetch Codidact sites: response code #{response.code}"
+      []
     end
   end
 end
diff --git a/config/locales/strings/en.admin.yml b/config/locales/strings/en.admin.yml
index 08115e4dd03aa9bc8892f60de3941082beac0515..ac389d08a8ef79f23ae23d21f6d699836f6c366d 100644
--- a/config/locales/strings/en.admin.yml
+++ b/config/locales/strings/en.admin.yml
@@ -11,6 +11,10 @@ en:
       This tool allows you to send to the moderators' email distribution list, which will reach any current moderator
       who is subscribed to it. Use sparingly and non-spammily. Your email will be sent in both HTML (formatted)
       and plain text (unformatted) forms, so format in a way that will be legible when Markdown is removed.
+    email_all_blurb: >
+      This tool sends email to ALL registered users with a valid email address. Use very very rarely, usually only for
+      organisation-wide changes like privacy policy or TOS changes. Format in a way that will be legible when Markdown
+      is removed.
     error_search_uuid: 'Search for an error UUID'
     privileges_blurb: >
       Here you can define the reputation required to gain each available privilege. Click on a value to edit it.
@@ -18,6 +22,7 @@ en:
       g_site_settings: 'Global Site Settings'
       g_tag_sets: 'Global Tag Sets'
       email_moderators: 'Email Moderators'
+      email_all: 'Email All Users'
       error_reports: 'Error Reports'
       site_settings: 'Site Settings'
       tag_sets: 'Tag Sets'
diff --git a/config/locales/strings/en.edits.yml b/config/locales/strings/en.edits.yml
new file mode 100644
index 0000000000000000000000000000000000000000..a34799dd68756c77c74f442b1024190069f35ce2
--- /dev/null
+++ b/config/locales/strings/en.edits.yml
@@ -0,0 +1,6 @@
+en:
+  edits:
+    # validation error messages
+    max_edit_comment_length:
+      one: Edit comment can't be more than 1 character long.
+      other: Edit comment can't be more than :length characters long.
\ No newline at end of file
diff --git a/config/locales/strings/en.g.yml b/config/locales/strings/en.g.yml
index 53590fa2f32a4013a241e421fccf74693496e52b..1fdf073ccc1e59fb4d4d5e8363d6e3bdcd40b1f8 100644
--- a/config/locales/strings/en.g.yml
+++ b/config/locales/strings/en.g.yml
@@ -22,4 +22,7 @@ en:
     subject: 'subject'
     threshold: 'threshold'
     type: 'type'
-    user: 'user'
\ No newline at end of file
+    user: 'user'
+
+  platform:
+    network_name: 'Codidact network'
diff --git a/config/locales/strings/en.posts.yml b/config/locales/strings/en.posts.yml
index 93fba33ff18e46cb46163bdf941a5e4fc1750ac6..e866d0e8e93519a7cc840f1c99d9f3dc4e9829dd 100644
--- a/config/locales/strings/en.posts.yml
+++ b/config/locales/strings/en.posts.yml
@@ -25,10 +25,12 @@ en:
       Can't delete a deleted post.
     cant_delete_responded: >
       This post cannot be deleted because it has responses.
+    cant_delete_community: >
+      This post cannot be deleted because it is owned by the community (freely editable).
     cant_restore_post: >
       Can't restore this post right now. Try again later.
     cant_restore_undeleted: >
-      Can't restore an undeleted post..
+      Can't restore a post that isn't deleted.
     cant_restore_deleted_by_moderator: >
       You cannot restore this post deleted by a moderator.
     cant_change_category: >
@@ -46,6 +48,10 @@ en:
       Requires at least one of
     edit_comment_label: >
       Edit Comment
+    redact_label: >
+      Redact
+    redact_explanation: >
+      Redact original content by hiding the previous versions from history? Use only for private information such as passwords or personally identifiable information.
     licence_label: >
       License
     unsaved_changes_confirmation: >
@@ -66,3 +72,8 @@ en:
       Responding to:
     no_block_mathjax_title: >
       Title cannot contain block-level MathJax.
+
+    post_closed_guidance: >
+      Your post has been closed and is not accepting new answers. You should read the reason for closure listed below and check for any comments left by others about
+      why your post has been closed and what it needs, then edit your post to improve it to a state where it can be reopened. If you need help, reply to comments to
+      clarify any requests, or join us in chat for more guidance.
diff --git a/config/locales/strings/en.rate_limit.yml b/config/locales/strings/en.rate_limit.yml
index 8510b148e75e1062399f054044a38af85680ff67..e073f705a0f86f5ee8a405e8d475eabc720ce973 100644
--- a/config/locales/strings/en.rate_limit.yml
+++ b/config/locales/strings/en.rate_limit.yml
@@ -1,10 +1,11 @@
-rate_limit:
-  new_user_posts: >
-    You may only post :count :level posts per day. Once you have some well-received posts, your limit will increase.
-  posts: >
-    You may only post :count :level posts per day.
-  new_user_suggested_edits: >
-    You may only suggest :count edits per day. Once you have some well-received contributions, your limit will increase.
-  suggested_edits: >
-    You may only suggest :count edits per day. Once you have enough well-received suggestions, you will earn the ability
-    to edit without review.
+en:
+  rate_limit:
+    new_user_posts: >
+      You may only post :count :level posts per day. Once you have some well-received posts, your limit will increase.
+    posts: >
+      You may only post :count :level posts per day.
+    new_user_suggested_edits: >
+      You may only suggest :count edits per day. Once you have some well-received contributions, your limit will increase.
+    suggested_edits: >
+      You may only suggest :count edits per day. Once you have enough well-received suggestions, you will earn the ability
+      to edit without review.
diff --git a/config/locales/strings/en.search.yml b/config/locales/strings/en.search.yml
new file mode 100644
index 0000000000000000000000000000000000000000..9358256b204f66bb69caaad6ae551ac1ecae53f0
--- /dev/null
+++ b/config/locales/strings/en.search.yml
@@ -0,0 +1,5 @@
+en:
+  search:
+    # used on the users page (not the main search)
+    no_users:
+      No users found.
diff --git a/config/locales/strings/en.votes.yml b/config/locales/strings/en.votes.yml
new file mode 100644
index 0000000000000000000000000000000000000000..485ef2f6f4f199be625f1592e912cdad32d4d95a
--- /dev/null
+++ b/config/locales/strings/en.votes.yml
@@ -0,0 +1,4 @@
+en:
+  votes:
+    summary:
+      post_missing: 'Post not found'
\ No newline at end of file
diff --git a/config/locales/strings/es.posts.yml b/config/locales/strings/es.posts.yml
index ccd7f24203781b386a5631f4f91b1aa36c86c713..d59bb2d56ed962c0763fcac94fd663601bbad28f 100644
--- a/config/locales/strings/es.posts.yml
+++ b/config/locales/strings/es.posts.yml
@@ -46,6 +46,10 @@ es:
       Requiere al menos un/a de
     edit_comment_label: >
       Editar comentario
+    redact_label: >
+      Redactar
+    redact_explanation: >
+      Redactar el contenido original ocultando las versiones anteriores del historial
     licence_label: >
       Licencia
     unsaved_changes_confirmation: >
diff --git a/config/locales/strings/es.rate_limit.yml b/config/locales/strings/es.rate_limit.yml
index a79963a9ec52a2ff601eabe8d6b76159faf2135d..430931763b8608b4409fe040e1d3f658c7375f3c 100644
--- a/config/locales/strings/es.rate_limit.yml
+++ b/config/locales/strings/es.rate_limit.yml
@@ -1,10 +1,11 @@
-rate_limit:
-  new_user_posts: >
-    Usted sólo debería publicar :count :level publicaciones por día. Cuando usted haya publicado varias publicaciones bien recibidas, su límite será incrementado.
-  posts: >
-    Usted sĂłlo deberĂ­a publicar :count :level publicaciones por dĂ­a.
-  new_user_suggested_edits: >
-    Usted sólo debería sugerir :count ediciones por día. Cuando usted haya propuesto varias contribuciones bien recibidas, su límite será incrementado.
-  suggested_edits: >
-    Usted sólo debería sugerir :count ediciones por día. Cuando usted haya propuesto las suficientes sugerencias bien recibidas, usted conseguirá la habilidad
-    de editar sin ser revisado.
+es:
+  rate_limit:
+    new_user_posts: >
+      Usted sólo debería publicar :count :level publicaciones por día. Cuando usted haya publicado varias publicaciones bien recibidas, su límite será incrementado.
+    posts: >
+      Usted sĂłlo deberĂ­a publicar :count :level publicaciones por dĂ­a.
+    new_user_suggested_edits: >
+      Usted sólo debería sugerir :count ediciones por día. Cuando usted haya propuesto varias contribuciones bien recibidas, su límite será incrementado.
+    suggested_edits: >
+      Usted sólo debería sugerir :count ediciones por día. Cuando usted haya propuesto las suficientes sugerencias bien recibidas, usted conseguirá la habilidad
+      de editar sin ser revisado.
diff --git a/config/routes.rb b/config/routes.rb
index 1619878ee6deb23be1a87ae8baa0a443e36fddaf..8b8b6b37bb6ab3a21b7029ff93de215478e17f9d 100644
--- a/config/routes.rb
+++ b/config/routes.rb
@@ -1,8 +1,20 @@
 Rails.application.routes.draw do
-  devise_for :users, controllers: { sessions: 'users/sessions', registrations: 'users/registrations' }
+  # Add normal sign in/sign up, confirmations, registrations, unlocking and password editing routes only if no SSO or mixed sign in.
+  devise_for :users, only: %i[sessions registrations confirmations unlock passwords],
+             controllers: { sessions: 'users/sessions', registrations: 'users/registrations' },
+             constraints: { url: lambda { |_url| SiteSetting['MixedSignIn'] || !SiteSetting['SsoSignIn'] } }
+  # Add SAML routes only when SAML enabled
+  devise_for :users, only: :saml_authenticatable,
+             controllers: { saml_sessions: 'users/saml_sessions' },
+             constraints: { url: lambda { |_url| SiteSetting['SsoSignIn'] } }
+  # Add any other devise routes that may exist that we did not add yet
+  devise_for :users, skip: %i[sessions registrations confirmations unlock passwords saml_authenticatable]
   devise_scope :user do
     get  'users/2fa/login',                to: 'users/sessions#verify_2fa', as: :login_verify_2fa
     post 'users/2fa/login',                to: 'users/sessions#verify_code', as: :login_verify_code
+    get  'users/saml/sign_in_request_from_other/:id', to: 'users/saml_sessions#sign_in_request_from_other', as: :sign_in_request_from_other
+    get  'users/saml/sign_in_return_from_base',       to: 'users/saml_sessions#sign_in_return_from_base', as: :sign_in_return_from_base
+    get  'users/saml/after_sign_in_check',            to: 'users/saml_sessions#after_sign_in_check', as: :after_sign_in_check
   end
 
   root                                     to: 'categories#homepage'
@@ -26,6 +38,8 @@ Rails.application.routes.draw do
 
     get    'mod-email',                    to: 'admin#admin_email', as: :moderator_email
     post   'mod-email',                    to: 'admin#send_admin_email', as: :send_moderator_email
+    get    'email-all',                    to: 'admin#all_email', as: :email_all
+    post   'email-all',                    to: 'admin#send_all_email', as: :send_all_email
 
     get    'audits',                       to: 'admin#audit_log', as: :audit_log
 
@@ -144,8 +158,10 @@ Rails.application.routes.draw do
     get    ':id/:answer',                  to: 'posts#show', as: :answer_post
   end
 
-  get    'policy/:slug',                   to: 'posts#document', as: :policy
-  get    'help/:slug',                     to: 'posts#document', as: :help
+  get    'policy/:slug/history',           to: 'post_history#slug_post', as: :policy_post_history, constraints: { slug: /.*/ }
+  get    'policy/:slug',                   to: 'posts#document', as: :policy, constraints: { slug: /.*/ }
+  get    'help/:slug/history',             to: 'post_history#slug_post', as: :help_post_history, constraints: { slug: /.*/ }
+  get    'help/:slug',                     to: 'posts#document', as: :help, constraints: { slug: /.*/ }
 
   get    'tags',                           to: 'tags#index', as: :tags
 
@@ -159,6 +175,7 @@ Rails.application.routes.draw do
     post 'disable/link-email',             to: 'two_factor#send_disable_email', as: :two_factor_send_disable_email
     get  'disable/link/:token',            to: 'two_factor#disable_link', as: :two_factor_disable_link
     post 'disable/link',                   to: 'two_factor#confirm_disable_link', as: :two_factor_confirm_disable_link
+    post 'backup',                         to: 'two_factor#show_backup_code', as: :two_factor_backup_code
   end
 
   scope  'users' do
@@ -170,17 +187,25 @@ Rails.application.routes.draw do
     get    '/me',                       to: 'users#me', as: :users_me
     get    '/me/preferences',           to: 'users#preferences', as: :user_preferences
     post   '/me/preferences',           to: 'users#set_preference', as: :set_user_preference
+    get    '/me/filters',               to: 'users#filters', as: :user_filters
+    get    '/me/filters/default/',      to: 'users#default_filter', as: :default_filter
+    post   '/me/filters',               to: 'users#set_filter', as: :set_user_filter
+    delete '/me/filters',               to: 'users#delete_filter', as: :delete_user_filter
     get    '/me/notifications',         to: 'notifications#index', as: :notifications
     get    '/edit/profile',             to: 'users#edit_profile', as: :edit_user_profile
     patch  '/edit/profile',             to: 'users#update_profile', as: :update_user_profile
     get    '/me/vote-summary',          to: 'users#my_vote_summary', as: :my_vote_summary
+    get    '/me/network',               to: 'users#my_network', as: :my_network
     get    '/avatar/:letter/:color/:size', to: 'users#specific_avatar', as: :specific_auto_avatar
+    get    '/disconnect-sso',           to: 'users#disconnect_sso', as: :user_disconnect_sso
+    post   '/disconnect-sso',           to: 'users#confirm_disconnect_sso', as: :user_confirm_disconnect_sso
     get    '/:id',                      to: 'users#show', as: :user
     get    '/:id/flags',                to: 'flags#history', as: :flag_history
     get    '/:id/activity',             to: 'users#activity', as: :user_activity
     get    '/:id/mod',                  to: 'users#mod', as: :mod_user
     get    '/:id/posts',                to: 'users#posts', as: :user_posts
     get    '/:id/vote-summary',         to: 'users#vote_summary', as: :vote_summary
+    get    '/:id/network',              to: 'users#network', as: :network
     get    '/:id/mod/privileges',       to: 'users#mod_privileges', as: :user_privileges
     post   '/:id/mod/privileges',       to: 'users#mod_privilege_action', as: :user_privilege_action
     post   '/:id/mod/toggle-role',      to: 'users#role_toggle', as: :toggle_user_role
@@ -237,6 +262,7 @@ Rails.application.routes.draw do
     root                                   to: 'categories#index', as: :categories
     get    'new',                          to: 'categories#new', as: :new_category
     post   'new',                          to: 'categories#create', as: :create_category
+    post   ':id/filters/default',          to: 'categories#default_filter', as: :set_default_filter
     get    ':id',                          to: 'categories#show', as: :category
     get    ':id/edit',                     to: 'categories#edit', as: :edit_category
     post   ':id/edit',                     to: 'categories#update', as: :update_category
@@ -326,6 +352,10 @@ Rails.application.routes.draw do
     end
   end
 
+  scope 'emails' do
+    post   'log', to: 'email_logs#log', as: :create_email_log
+  end
+
   get   '403',                             to: 'errors#forbidden'
   get   '404',                             to: 'errors#not_found'
   get   '409',                             to: 'errors#conflict'
@@ -339,4 +369,8 @@ Rails.application.routes.draw do
   scope 'network' do
     root                                   to: 'fake_community#communities', as: :fc_communities
   end
+
+  # Communities can have custom js or css defined (placed in public/assets/community).
+  # If these are not defined for a community, respond with 204 (ok but empty)
+  get '/assets/community/*path', to: ->(env) { [204, {}, ['']] }
 end
diff --git a/config/schedule.rb b/config/schedule.rb
index a9dd5e5aa61da74bcd9dc33cd10550a14ada587c..1b1c9c343a695e45dbc65fe383df8fc075f7decf 100644
--- a/config/schedule.rb
+++ b/config/schedule.rb
@@ -6,6 +6,22 @@ every 1.day, at: '02:05' do
   runner 'scripts/mail_uncaptured_donations.rb'
 end
 
+every 1.day, at: '02:10' do
+  runner 'scripts/prune_email_logs.rb'
+end
+
+every 1.day, at: '02:15' do
+  runner 'scripts/run_spam_cleanup.rb'
+end
+
+every 1.day, at: '02:20' do
+  runner 'scripts/cleanup_drafts.rb'
+end
+
+every 1.day, at: '02:25' do
+  runner 'scripts/cleanup_votes.rb'
+end
+
 every 6.hours do
   runner 'scripts/recalc_abilities.rb'
 end
diff --git a/config/storage.docker.yml b/config/storage.docker.yml
new file mode 100644
index 0000000000000000000000000000000000000000..88649a3fb37afc606d8d28be062a2d8e600fb2f2
--- /dev/null
+++ b/config/storage.docker.yml
@@ -0,0 +1,14 @@
+test:
+  service: Disk
+  root: <%= Rails.root.join('tmp/storage') %>
+
+local:
+  service: Disk
+  root: <%= Rails.root.join('storage') %>
+
+s3:
+  service: S3
+  access_key_id: ""
+  secret_access_key: ""
+  region: us-east-1
+  bucket: ""
diff --git a/config/storage.sample.yml b/config/storage.sample.yml
index 88649a3fb37afc606d8d28be062a2d8e600fb2f2..8d0ce74389e1b28e39396c26ef6cb4a2bb19bc6f 100644
--- a/config/storage.sample.yml
+++ b/config/storage.sample.yml
@@ -1,3 +1,8 @@
+# To use external files (such as image upload), you need a storage.yml file.
+# If you are using only the local disk (such as in a development environment),
+# you can just copy this file. If you are using an S3 bucket, copy this
+# file and then edit the settings.
+
 test:
   service: Disk
   root: <%= Rails.root.join('tmp/storage') %>
diff --git a/db/migrate/20200813132829_enforce_concurrence_of_is_moderator_and_mod_ability.rb b/db/migrate/20200813132829_enforce_concurrence_of_is_moderator_and_mod_ability.rb
index d84fdcc334d2dec68e775f9712d011e227bf9c9a..606f724a910a6d39a787927fd3e15d75f5e2c678 100644
--- a/db/migrate/20200813132829_enforce_concurrence_of_is_moderator_and_mod_ability.rb
+++ b/db/migrate/20200813132829_enforce_concurrence_of_is_moderator_and_mod_ability.rb
@@ -2,7 +2,7 @@ class EnforceConcurrenceOfIsModeratorAndModAbility < ActiveRecord::Migration[5.2
   def up
     CommunityUser.unscoped.where(is_moderator: true).all.map do |cu|
       RequestContext.community = cu.community
-      cu.grant_privilege  'mod'
+      cu.grant_privilege! 'mod'
     end
   end
 
diff --git a/db/migrate/20220811131155_create_sso_profile.rb b/db/migrate/20220811131155_create_sso_profile.rb
new file mode 100644
index 0000000000000000000000000000000000000000..1ef086211c30db1a2678a9a3c21df6c8ee3896da
--- /dev/null
+++ b/db/migrate/20220811131155_create_sso_profile.rb
@@ -0,0 +1,8 @@
+class CreateSsoProfile < ActiveRecord::Migration[5.2]
+  def change
+    create_table :sso_profiles do |t|
+      t.string :saml_identifier, null: false
+      t.references :user, null: false, foreign_key: true
+    end
+  end
+end
\ No newline at end of file
diff --git a/db/migrate/20220915181608_create_tag_synonyms.rb b/db/migrate/20220915181608_create_tag_synonyms.rb
new file mode 100644
index 0000000000000000000000000000000000000000..2bf629601682b543202522438ab2d6e840b6e37f
--- /dev/null
+++ b/db/migrate/20220915181608_create_tag_synonyms.rb
@@ -0,0 +1,9 @@
+class CreateTagSynonyms < ActiveRecord::Migration[7.0]
+  def change
+    create_table :tag_synonyms do |t|
+      t.belongs_to :tag, foreign_key: true
+      t.string :name, null: false
+      t.timestamps
+    end
+  end
+end
diff --git a/db/migrate/20220916075849_create_filters.rb b/db/migrate/20220916075849_create_filters.rb
new file mode 100644
index 0000000000000000000000000000000000000000..ffb41db6afe1fb84a7aaf2b861ca1d2d77886dd4
--- /dev/null
+++ b/db/migrate/20220916075849_create_filters.rb
@@ -0,0 +1,17 @@
+class CreateFilters < ActiveRecord::Migration[7.0]
+  def change
+    create_table :filters do |t|
+      t.references :user, null: false, foreign_key: true
+      t.string :name, null: false
+      t.float :min_score
+      t.float :max_score
+      t.integer :min_answers
+      t.integer :max_answers
+      t.string :status
+      t.string :include_tags
+      t.string :exclude_tags
+
+      t.timestamps
+    end
+  end
+end
diff --git a/db/migrate/20221002043021_create_category_filter_defaults.rb b/db/migrate/20221002043021_create_category_filter_defaults.rb
new file mode 100644
index 0000000000000000000000000000000000000000..81e5dfa178f680e2211d023d9915b20edd48dbfc
--- /dev/null
+++ b/db/migrate/20221002043021_create_category_filter_defaults.rb
@@ -0,0 +1,9 @@
+class CreateCategoryFilterDefaults < ActiveRecord::Migration[7.0]
+  def change
+    create_table :category_filter_defaults do |t|
+      t.references :user, null: false, foreign_key: true
+      t.references :filter, null: false, foreign_key: true
+      t.references :category, null: false, foreign_key: true
+    end
+  end
+end
diff --git a/db/migrate/20230613205236_add_min_title_length_to_categories.rb b/db/migrate/20230613205236_add_min_title_length_to_categories.rb
new file mode 100644
index 0000000000000000000000000000000000000000..9c5b752feb1e3dc9dbc1569b091fe06e590b3757
--- /dev/null
+++ b/db/migrate/20230613205236_add_min_title_length_to_categories.rb
@@ -0,0 +1,6 @@
+class AddMinTitleLengthToCategories < ActiveRecord::Migration[7.0]
+  def change
+    add_column :categories, :min_title_length, :integer, null: false, default: 15
+    add_column :categories, :min_body_length, :integer, null: false, default: 30
+  end
+end
diff --git a/db/migrate/20230627035349_add_default_filter_to_category.rb b/db/migrate/20230627035349_add_default_filter_to_category.rb
new file mode 100644
index 0000000000000000000000000000000000000000..b6f49a4b499c109f959c2b625cc173399206aecd
--- /dev/null
+++ b/db/migrate/20230627035349_add_default_filter_to_category.rb
@@ -0,0 +1,5 @@
+class AddDefaultFilterToCategory < ActiveRecord::Migration[7.0]
+  def change
+    add_reference :categories, :default_filter, foreign_key: { to_table: :filters }, null: true
+  end
+end
diff --git a/db/migrate/20230726143348_add_hidden_to_post_history.rb b/db/migrate/20230726143348_add_hidden_to_post_history.rb
new file mode 100644
index 0000000000000000000000000000000000000000..c1ba982023a9070cc89eac834cc5dc0de211fb56
--- /dev/null
+++ b/db/migrate/20230726143348_add_hidden_to_post_history.rb
@@ -0,0 +1,5 @@
+class AddHiddenToPostHistory < ActiveRecord::Migration[7.0]
+  def change
+    add_column :post_histories, :hidden, :boolean, null: false, default: false
+  end
+end
diff --git a/db/migrate/20230803191600_add_backup_2fa_code_to_users.rb b/db/migrate/20230803191600_add_backup_2fa_code_to_users.rb
new file mode 100644
index 0000000000000000000000000000000000000000..781ae8712ec4e3a69facc49cb49da622299392d1
--- /dev/null
+++ b/db/migrate/20230803191600_add_backup_2fa_code_to_users.rb
@@ -0,0 +1,5 @@
+class AddBackup2faCodeToUsers < ActiveRecord::Migration[7.0]
+  def change
+    add_column :users, :backup_2fa_code, :string
+  end
+end
diff --git a/db/migrate/20230817213150_rename_site_setting_category_email_subscriptions.rb b/db/migrate/20230817213150_rename_site_setting_category_email_subscriptions.rb
new file mode 100644
index 0000000000000000000000000000000000000000..60382394c51025d72a58a5e7d2f66c3eafb11b57
--- /dev/null
+++ b/db/migrate/20230817213150_rename_site_setting_category_email_subscriptions.rb
@@ -0,0 +1,5 @@
+class RenameSiteSettingCategoryEmailSubscriptions < ActiveRecord::Migration[7.0]
+  def change
+    SiteSetting.where(category: 'EmailSubscriptions').update_all(category: 'Email')
+  end
+end
diff --git a/db/migrate/202309141441_disable_needs_author_attention_flag.rb b/db/migrate/202309141441_disable_needs_author_attention_flag.rb
new file mode 100644
index 0000000000000000000000000000000000000000..917c84dcac0193ec8ff2d09bbf902d622dcc6507
--- /dev/null
+++ b/db/migrate/202309141441_disable_needs_author_attention_flag.rb
@@ -0,0 +1,9 @@
+class DisableNeedsAuthorAttentionFlag < ActiveRecord::Migration[7.0]
+  def up
+    PostFlagType.unscoped.where(name: "needs author's attention").update_all(active: false)
+  end
+
+  def down
+    PostFlagType.unscoped.where(name: "needs author's attention").update_all(active: true)
+  end
+end
diff --git a/db/migrate/20240405113618_create_email_logs.rb b/db/migrate/20240405113618_create_email_logs.rb
new file mode 100644
index 0000000000000000000000000000000000000000..ecea1a21dae273ebc2727181b583983c5eebdaca
--- /dev/null
+++ b/db/migrate/20240405113618_create_email_logs.rb
@@ -0,0 +1,11 @@
+class CreateEmailLogs < ActiveRecord::Migration[7.0]
+  def change
+    create_table :email_logs do |t|
+      t.string :log_type
+      t.string :destination
+      t.text :data
+
+      t.timestamps
+    end
+  end
+end
diff --git a/db/migrate/20241020193053_add_post_count_to_community_users.rb b/db/migrate/20241020193053_add_post_count_to_community_users.rb
new file mode 100644
index 0000000000000000000000000000000000000000..a6c54aa17c11e058c4bab74469aecb5036ee92c0
--- /dev/null
+++ b/db/migrate/20241020193053_add_post_count_to_community_users.rb
@@ -0,0 +1,5 @@
+class AddPostCountToCommunityUsers < ActiveRecord::Migration[7.0]
+  def change
+    add_column :community_users, :post_count, :integer, default: 0, null: false
+  end
+end
diff --git a/db/migrate/20250123141400_create_user_websites.rb b/db/migrate/20250123141400_create_user_websites.rb
new file mode 100644
index 0000000000000000000000000000000000000000..ec77b2d17a97cf4e496cceec13d1c42f1203e52b
--- /dev/null
+++ b/db/migrate/20250123141400_create_user_websites.rb
@@ -0,0 +1,11 @@
+class CreateUserWebsites < ActiveRecord::Migration[7.0]
+  def change
+    create_table :user_websites do |t|
+      t.column :label, :string, limit:80
+      t.string :url
+      t.integer :position
+    end
+    add_reference :user_websites, :user, null: false, foreign_key: true
+    add_index(:user_websites, [:user_id, :url], unique: true)
+  end
+end
diff --git a/db/migrate/20250128030354_create_maintenance_tasks_runs.maintenance_tasks.rb b/db/migrate/20250128030354_create_maintenance_tasks_runs.maintenance_tasks.rb
new file mode 100644
index 0000000000000000000000000000000000000000..c9b215d687074236510e2d3114d15f510f8396cf
--- /dev/null
+++ b/db/migrate/20250128030354_create_maintenance_tasks_runs.maintenance_tasks.rb
@@ -0,0 +1,24 @@
+# frozen_string_literal: true
+
+# This migration comes from maintenance_tasks (originally 20201211151756)
+class CreateMaintenanceTasksRuns < ActiveRecord::Migration[6.0]
+  def change
+    create_table(:maintenance_tasks_runs) do |t|
+      t.string(:task_name, null: false)
+      t.datetime(:started_at)
+      t.datetime(:ended_at)
+      t.float(:time_running, default: 0.0, null: false)
+      t.integer(:tick_count, default: 0, null: false)
+      t.integer(:tick_total)
+      t.string(:job_id)
+      t.bigint(:cursor)
+      t.string(:status, default: :enqueued, null: false)
+      t.string(:error_class)
+      t.string(:error_message)
+      t.text(:backtrace)
+      t.timestamps
+      t.index(:task_name)
+      t.index([:task_name, :created_at], order: { created_at: :desc })
+    end
+  end
+end
diff --git a/db/migrate/20250128030355_change_cursor_to_string.maintenance_tasks.rb b/db/migrate/20250128030355_change_cursor_to_string.maintenance_tasks.rb
new file mode 100644
index 0000000000000000000000000000000000000000..4e414cad3ab30ce9e381cc685333fe910ebdd791
--- /dev/null
+++ b/db/migrate/20250128030355_change_cursor_to_string.maintenance_tasks.rb
@@ -0,0 +1,19 @@
+# frozen_string_literal: true
+
+# This migration comes from maintenance_tasks (originally 20210219212931)
+class ChangeCursorToString < ActiveRecord::Migration[6.0]
+  # This migration will clear all existing data in the cursor column with MySQL.
+  # Ensure no Tasks are paused when this migration is deployed, or they will be resumed from the start.
+  # Running tasks are able to gracefully handle this change, even if interrupted.
+  def up
+    change_table(:maintenance_tasks_runs) do |t|
+      t.change(:cursor, :string)
+    end
+  end
+
+  def down
+    change_table(:maintenance_tasks_runs) do |t|
+      t.change(:cursor, :bigint)
+    end
+  end
+end
diff --git a/db/migrate/20250128030356_remove_index_on_task_name.maintenance_tasks.rb b/db/migrate/20250128030356_remove_index_on_task_name.maintenance_tasks.rb
new file mode 100644
index 0000000000000000000000000000000000000000..1b5cadf8dfaa78a36e1913b4bef9ebe22b8cbcd5
--- /dev/null
+++ b/db/migrate/20250128030356_remove_index_on_task_name.maintenance_tasks.rb
@@ -0,0 +1,16 @@
+# frozen_string_literal: true
+
+# This migration comes from maintenance_tasks (originally 20210225152418)
+class RemoveIndexOnTaskName < ActiveRecord::Migration[6.0]
+  def up
+    change_table(:maintenance_tasks_runs) do |t|
+      t.remove_index(:task_name)
+    end
+  end
+
+  def down
+    change_table(:maintenance_tasks_runs) do |t|
+      t.index(:task_name)
+    end
+  end
+end
diff --git a/db/migrate/20250128030357_add_arguments_to_maintenance_tasks_runs.maintenance_tasks.rb b/db/migrate/20250128030357_add_arguments_to_maintenance_tasks_runs.maintenance_tasks.rb
new file mode 100644
index 0000000000000000000000000000000000000000..8478268eba9ef048aa17cccfde5d32e89a405a39
--- /dev/null
+++ b/db/migrate/20250128030357_add_arguments_to_maintenance_tasks_runs.maintenance_tasks.rb
@@ -0,0 +1,8 @@
+# frozen_string_literal: true
+
+# This migration comes from maintenance_tasks (originally 20210517131953)
+class AddArgumentsToMaintenanceTasksRuns < ActiveRecord::Migration[6.0]
+  def change
+    add_column(:maintenance_tasks_runs, :arguments, :text)
+  end
+end
diff --git a/db/migrate/20250128030358_add_lock_version_to_maintenance_tasks_runs.maintenance_tasks.rb b/db/migrate/20250128030358_add_lock_version_to_maintenance_tasks_runs.maintenance_tasks.rb
new file mode 100644
index 0000000000000000000000000000000000000000..b302565af387a4608bbb2f028cf2b621cadad33b
--- /dev/null
+++ b/db/migrate/20250128030358_add_lock_version_to_maintenance_tasks_runs.maintenance_tasks.rb
@@ -0,0 +1,14 @@
+# frozen_string_literal: true
+
+# This migration comes from maintenance_tasks (originally 20211210152329)
+class AddLockVersionToMaintenanceTasksRuns < ActiveRecord::Migration[6.0]
+  def change
+    add_column(
+      :maintenance_tasks_runs,
+      :lock_version,
+      :integer,
+      default: 0,
+      null: false,
+    )
+  end
+end
diff --git a/db/migrate/20250128030359_change_runs_tick_columns_to_bigints.maintenance_tasks.rb b/db/migrate/20250128030359_change_runs_tick_columns_to_bigints.maintenance_tasks.rb
new file mode 100644
index 0000000000000000000000000000000000000000..064b07705736382731c14914e36124b65d18f1c4
--- /dev/null
+++ b/db/migrate/20250128030359_change_runs_tick_columns_to_bigints.maintenance_tasks.rb
@@ -0,0 +1,18 @@
+# frozen_string_literal: true
+
+# This migration comes from maintenance_tasks (originally 20220706101937)
+class ChangeRunsTickColumnsToBigints < ActiveRecord::Migration[6.0]
+  def up
+    change_table(:maintenance_tasks_runs, bulk: true) do |t|
+      t.change(:tick_count, :bigint)
+      t.change(:tick_total, :bigint)
+    end
+  end
+
+  def down
+    change_table(:maintenance_tasks_runs, bulk: true) do |t|
+      t.change(:tick_count, :integer)
+      t.change(:tick_total, :integer)
+    end
+  end
+end
diff --git a/db/migrate/20250128030360_add_index_on_task_name_and_status_to_runs.maintenance_tasks.rb b/db/migrate/20250128030360_add_index_on_task_name_and_status_to_runs.maintenance_tasks.rb
new file mode 100644
index 0000000000000000000000000000000000000000..74d77e6bef74b226916e54d38e956497aaa6d22a
--- /dev/null
+++ b/db/migrate/20250128030360_add_index_on_task_name_and_status_to_runs.maintenance_tasks.rb
@@ -0,0 +1,20 @@
+# frozen_string_literal: true
+
+# This migration comes from maintenance_tasks (originally 20220713131925)
+class AddIndexOnTaskNameAndStatusToRuns < ActiveRecord::Migration[6.0]
+  def change
+    remove_index(
+      :maintenance_tasks_runs,
+      column: [:task_name, :created_at],
+      order: { created_at: :desc },
+      name: :index_maintenance_tasks_runs_on_task_name_and_created_at,
+    )
+
+    add_index(
+      :maintenance_tasks_runs,
+      [:task_name, :status, :created_at],
+      name: :index_maintenance_tasks_runs,
+      order: { created_at: :desc },
+    )
+  end
+end
diff --git a/db/migrate/20250128030361_add_metadata_to_runs.maintenance_tasks.rb b/db/migrate/20250128030361_add_metadata_to_runs.maintenance_tasks.rb
new file mode 100644
index 0000000000000000000000000000000000000000..054ef0e2a976724eabe7a6b9126eed61d8efe280
--- /dev/null
+++ b/db/migrate/20250128030361_add_metadata_to_runs.maintenance_tasks.rb
@@ -0,0 +1,8 @@
+# frozen_string_literal: true
+
+# This migration comes from maintenance_tasks (originally 20230622035229)
+class AddMetadataToRuns < ActiveRecord::Migration[6.0]
+  def change
+    add_column(:maintenance_tasks_runs, :metadata, :text)
+  end
+end
diff --git a/db/schema.rb b/db/schema.rb
index 9cc021c6d809687d863c2571e039cb81160d2a2f..97e097dbace5e8724a3be08dd2b560f59e5e1e8b 100644
--- a/db/schema.rb
+++ b/db/schema.rb
@@ -10,7 +10,7 @@
 #
 # It's strongly recommended that you check this file into your version control system.
 
-ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
+ActiveRecord::Schema[7.0].define(version: 2025_01_28_030361) do
   create_table "abilities", charset: "utf8mb4", collation: "utf8mb4_0900_ai_ci", force: :cascade do |t|
     t.bigint "community_id"
     t.string "name"
@@ -109,7 +109,11 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
     t.integer "sequence"
     t.boolean "use_for_hot_posts", default: true
     t.boolean "use_for_advertisement", default: true
+    t.integer "min_title_length", default: 15, null: false
+    t.integer "min_body_length", default: 30, null: false
+    t.bigint "default_filter_id"
     t.index ["community_id"], name: "index_categories_on_community_id"
+    t.index ["default_filter_id"], name: "index_categories_on_default_filter_id"
     t.index ["license_id"], name: "index_categories_on_license_id"
     t.index ["sequence"], name: "index_categories_on_sequence"
     t.index ["tag_set_id"], name: "index_categories_on_tag_set_id"
@@ -138,6 +142,15 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
     t.bigint "tag_id"
   end
 
+  create_table "category_filter_defaults", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
+    t.bigint "user_id", null: false
+    t.bigint "filter_id", null: false
+    t.bigint "category_id", null: false
+    t.index ["category_id"], name: "index_category_filter_defaults_on_category_id"
+    t.index ["filter_id"], name: "index_category_filter_defaults_on_filter_id"
+    t.index ["user_id"], name: "index_category_filter_defaults_on_user_id"
+  end
+
   create_table "close_reasons", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
     t.string "name"
     t.text "description", size: :medium
@@ -213,11 +226,20 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
     t.boolean "deleted", default: false, null: false
     t.datetime "deleted_at", precision: nil
     t.bigint "deleted_by_id"
+    t.integer "post_count", default: 0, null: false
     t.index ["community_id"], name: "index_community_users_on_community_id"
     t.index ["deleted_by_id"], name: "index_community_users_on_deleted_by_id"
     t.index ["user_id"], name: "index_community_users_on_user_id"
   end
 
+  create_table "email_logs", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
+    t.string "log_type"
+    t.string "destination"
+    t.text "data"
+    t.datetime "created_at", null: false
+    t.datetime "updated_at", null: false
+  end
+
   create_table "error_logs", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
     t.bigint "community_id"
     t.bigint "user_id"
@@ -234,6 +256,21 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
     t.index ["user_id"], name: "index_error_logs_on_user_id"
   end
 
+  create_table "filters", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
+    t.bigint "user_id", null: false
+    t.string "name", null: false
+    t.float "min_score"
+    t.float "max_score"
+    t.integer "min_answers"
+    t.integer "max_answers"
+    t.string "status"
+    t.string "include_tags"
+    t.string "exclude_tags"
+    t.datetime "created_at", null: false
+    t.datetime "updated_at", null: false
+    t.index ["user_id"], name: "index_filters_on_user_id"
+  end
+
   create_table "flags", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
     t.text "reason"
     t.datetime "created_at", precision: nil, null: false
@@ -272,6 +309,27 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
     t.index ["name"], name: "index_licenses_on_name"
   end
 
+  create_table "maintenance_tasks_runs", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
+    t.string "task_name", null: false
+    t.datetime "started_at", precision: nil
+    t.datetime "ended_at", precision: nil
+    t.float "time_running", default: 0.0, null: false
+    t.bigint "tick_count", default: 0, null: false
+    t.bigint "tick_total"
+    t.string "job_id"
+    t.string "cursor"
+    t.string "status", default: "enqueued", null: false
+    t.string "error_class"
+    t.string "error_message"
+    t.text "backtrace"
+    t.datetime "created_at", null: false
+    t.datetime "updated_at", null: false
+    t.text "arguments"
+    t.integer "lock_version", default: 0, null: false
+    t.text "metadata"
+    t.index ["task_name", "status", "created_at"], name: "index_maintenance_tasks_runs", order: { created_at: :desc }
+  end
+
   create_table "micro_auth_apps", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
     t.string "name"
     t.string "app_id"
@@ -360,6 +418,7 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
     t.bigint "community_id"
     t.string "before_title"
     t.string "after_title"
+    t.boolean "hidden", default: false, null: false
     t.index ["community_id"], name: "index_post_histories_on_community_id"
     t.index ["post_history_type_id"], name: "index_post_histories_on_post_history_type_id"
     t.index ["post_id"], name: "index_post_histories_on_post_type_and_post_id"
@@ -398,8 +457,8 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
     t.boolean "is_top_level", default: false, null: false
     t.boolean "is_freely_editable", default: false, null: false
     t.string "icon_name"
-    t.bigint "answer_type_id"
     t.boolean "has_reactions"
+    t.bigint "answer_type_id"
     t.boolean "has_only_specific_reactions"
     t.index ["answer_type_id"], name: "index_post_types_on_answer_type_id"
     t.index ["name"], name: "index_post_types_on_name"
@@ -535,6 +594,12 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
     t.index ["name"], name: "index_site_settings_on_name"
   end
 
+  create_table "sso_profiles", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
+    t.string "saml_identifier", null: false
+    t.bigint "user_id", null: false
+    t.index ["user_id"], name: "index_sso_profiles_on_user_id"
+  end
+
   create_table "subscriptions", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
     t.string "type", null: false
     t.string "qualifier"
@@ -594,6 +659,14 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
     t.index ["community_id"], name: "index_tag_sets_on_community_id"
   end
 
+  create_table "tag_synonyms", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
+    t.bigint "tag_id"
+    t.string "name", null: false
+    t.datetime "created_at", null: false
+    t.datetime "updated_at", null: false
+    t.index ["tag_id"], name: "index_tag_synonyms_on_tag_id"
+  end
+
   create_table "tags", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
     t.string "name"
     t.datetime "created_at", precision: nil, null: false
@@ -632,6 +705,15 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
     t.index ["community_user_id"], name: "index_user_abilities_on_community_user_id"
   end
 
+  create_table "user_websites", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
+    t.string "label", limit: 80
+    t.string "url"
+    t.integer "position"
+    t.bigint "user_id", null: false
+    t.index ["user_id", "url"], name: "index_user_websites_on_user_id_and_url", unique: true
+    t.index ["user_id"], name: "index_user_websites_on_user_id"
+  end
+
   create_table "users", charset: "utf8mb4", collation: "utf8mb4_unicode_ci", force: :cascade do |t|
     t.string "email"
     t.string "encrypted_password"
@@ -674,6 +756,7 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
     t.boolean "deleted", default: false, null: false
     t.datetime "deleted_at", precision: nil
     t.bigint "deleted_by_id"
+    t.string "backup_2fa_code"
     t.index ["confirmation_token"], name: "index_users_on_confirmation_token", unique: true
     t.index ["deleted_by_id"], name: "index_users_on_deleted_by_id"
     t.index ["email"], name: "index_users_on_email", unique: true
@@ -714,8 +797,8 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
     t.datetime "created_at", precision: nil, null: false
     t.datetime "updated_at", precision: nil, null: false
     t.boolean "read", default: false
-    t.index ["author_id"], name: "index_mod_messages_on_author_id"
-    t.index ["community_user_id"], name: "index_mod_messages_on_community_user_id"
+    t.index ["author_id"], name: "index_warnings_on_author_id"
+    t.index ["community_user_id"], name: "index_warnings_on_community_user_id"
   end
 
   add_foreign_key "abilities", "communities"
@@ -723,8 +806,12 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
   add_foreign_key "active_storage_variant_records", "active_storage_blobs", column: "blob_id"
   add_foreign_key "audit_logs", "communities"
   add_foreign_key "audit_logs", "users"
+  add_foreign_key "categories", "filters", column: "default_filter_id"
   add_foreign_key "categories", "licenses"
   add_foreign_key "categories", "tag_sets"
+  add_foreign_key "category_filter_defaults", "categories"
+  add_foreign_key "category_filter_defaults", "filters"
+  add_foreign_key "category_filter_defaults", "users"
   add_foreign_key "comment_threads", "users", column: "archived_by_id"
   add_foreign_key "comment_threads", "users", column: "deleted_by_id"
   add_foreign_key "comment_threads", "users", column: "locked_by_id"
@@ -735,6 +822,7 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
   add_foreign_key "community_users", "users", column: "deleted_by_id"
   add_foreign_key "error_logs", "communities"
   add_foreign_key "error_logs", "users"
+  add_foreign_key "filters", "users"
   add_foreign_key "flags", "communities"
   add_foreign_key "flags", "users", column: "escalated_by_id"
   add_foreign_key "micro_auth_apps", "users"
@@ -755,17 +843,20 @@ ActiveRecord::Schema[7.0].define(version: 2022_09_13_183826) do
   add_foreign_key "posts", "users", column: "locked_by_id"
   add_foreign_key "privileges", "communities"
   add_foreign_key "site_settings", "communities"
+  add_foreign_key "sso_profiles", "users"
   add_foreign_key "subscriptions", "communities"
   add_foreign_key "subscriptions", "users"
   add_foreign_key "suggested_edits", "communities"
   add_foreign_key "suggested_edits", "posts"
   add_foreign_key "suggested_edits", "users"
   add_foreign_key "suggested_edits", "users", column: "decided_by_id"
+  add_foreign_key "tag_synonyms", "tags"
   add_foreign_key "tags", "communities"
   add_foreign_key "tags", "tags", column: "parent_id"
   add_foreign_key "thread_followers", "posts"
   add_foreign_key "user_abilities", "abilities"
   add_foreign_key "user_abilities", "community_users"
+  add_foreign_key "user_websites", "users"
   add_foreign_key "users", "users", column: "deleted_by_id"
   add_foreign_key "votes", "communities"
   add_foreign_key "warning_templates", "communities"
diff --git a/db/scripts/create_tags_path_view.sql b/db/scripts/create_tags_path_view.sql
index 92708f38b2f27e1ab369542c522404cbac7ae882..6752f67343e12ac3237ff292bc23cf05dfa6d87c 100644
--- a/db/scripts/create_tags_path_view.sql
+++ b/db/scripts/create_tags_path_view.sql
@@ -1,4 +1,4 @@
-create view tags_paths as
+CREATE OR REPLACE VIEW tags_paths AS
 WITH RECURSIVE tag_path (id, created_at, updated_at, community_id, tag_set_id, wiki_markdown,
                          wiki, excerpt, parent_id, name, path) AS
                    (
diff --git a/db/seeds.rb b/db/seeds.rb
index c116c94c36aaaca729481cf0a52ed2b020d8fff9..11cb350a060cb89777dbbcc9442663b7dde79f12 100644
--- a/db/seeds.rb
+++ b/db/seeds.rb
@@ -16,16 +16,81 @@ types = files.map do |f|
 end
 
 # Prioritize the following models (in this order) such that models depending on them get created after
-priority = [PostType, CloseReason, License, TagSet, PostHistoryType]
+priority = [PostType, CloseReason, License, TagSet, PostHistoryType, User, Ability, CommunityUser, Filter]
 sorted = files.zip(types).to_h.sort do |a, b|
   (priority.index(a.second) || 999) <=> (priority.index(b.second) || 999)
 end.to_h
 
+def expand_communities(type, seed)
+  if type.column_names.include?('community_id') && !seed.include?('community_id')
+    # if model includes a community_id, create the seed for every community
+    Community.all.map { |c| seed.deep_symbolize_keys.merge(community_id: c.id) }
+  else
+    # otherwise, no need to worry, just create it
+    [seed]
+  end
+end
+
+def expand_ids(type, seeds)
+  # Transform all _id relations into the actual rails objects to pass validations
+  seeds.map do |seed|
+    columns = type.column_names.select { |name| name.match(/^.*_id$/) }
+    new_seed = seed.deep_symbolize_keys
+    columns.each do |column|
+      begin
+        column_type_name = column.chomp('_id')
+        column_type = column_type_name.classify.constantize
+        new_seed = new_seed.except(column.to_sym)
+                           .merge(column_type_name.to_sym => column_type.unscoped.find(seed[column.to_sym]))
+      rescue StandardError
+        # Either the type does not exist or the value specified as the id is not valid, ignore.
+        next
+      end
+    end
+    new_seed
+  end
+end
+
+def create_objects(type, seed)
+  seeds = expand_communities(type, seed)
+  seeds = expand_ids(type, seeds)
+
+  # Actually create the objects and count successes
+  objs = type.create seeds
+
+  skipped = objs.select { |o| o.errors.any? }.size
+  created = objs.select { |o| !o.errors.any? }.size
+
+  [created, skipped]
+end
+
+def ensure_system_user_abilities
+  system_users = CommunityUser.unscoped.where(user_id: -1)
+
+  system_users.each do |su|
+    abilities = Ability.unscoped
+      .where(internal_id: ['everyone', 'mod', 'unrestricted'])
+      .where(community_id: su.community_id)
+
+    user_abilities = UserAbility.unscoped.where(community_user_id: su.id)
+
+    abilities.each do |ab|
+      unless user_abilities.any? { |ua| ua.ability_id == ab.id }
+        UserAbility.create community_user_id: su.id, ability: ab
+      end
+    rescue => e
+      puts "#{type}: failed to add \"#{ab.name}\" to system user \"#{su.id}\" on \"#{su.community.name}\""
+      puts e
+    end
+  end
+end
+
 sorted.each do |f, type|
   begin
     processed = ERB.new(File.read(f)).result(binding)
     data = YAML.load(processed)
     created = 0
+    errored = 0
     skipped = 0
     updated = 0
     data.each do |seed|
@@ -39,57 +104,60 @@ sorted.each do |f, type|
         puts "Running full Posts update..."
 
         seed['body'] = ApplicationController.helpers.render_markdown(seed['body_markdown'])
+
+        system_usr = User.find(-1)
+
         Community.all.each do |c|
           RequestContext.community = c
           post = Post.find_by doc_slug: seed['doc_slug']
-          if post.present? && PostHistory.where(post: post).count <= 1
+          if post.present? && PostHistory.where(post: post)
+                                         .where.not(post_history_type:
+                                                      PostHistoryType.find_by(name: 'initial_revision'))
+                                         .count.zero?
+
             # post exists, still original version: update post
             post.update(seed.merge('community_id' => c.id))
+
+            no_initial = PostHistory.where(post: post)
+                       .where(post_history_type: PostHistoryType.find_by(name: 'initial_revision'))
+                       .count.zero?
+
+            if no_initial
+              puts "[#{c.name}:#{seed['doc_slug']}] missing initial revision, creating..."
+              PostHistory.initial_revision(post, system_usr)
+            end
+
             updated += 1
           elsif post.nil?
             # post doesn't exist: create post
-            Post.create seed.merge('community_id' => c.id)
-            created += 1
+            status = Post.create seed.merge('community_id' => c.id, 'user' => system_usr)
+
+            if status.errors.size
+              status.errors.full_messages.each do |msg|
+                puts "[#{c.name}:#{seed['doc_slug']}] invalid: #{msg}"
+              end
+
+              errored += 1
+            else
+              created += 1
+            end
           else
             # post exists, versions diverged: skip
             skipped += 1
           end
         end
       else
-        seeds = if type.column_names.include?('community_id') && !seed.include?('community_id')
-                 # if model includes a community_id, create the seed for every community
-                 Community.all.map { |c| seed.deep_symbolize_keys.merge(community_id: c.id) }
-               else
-                 # otherwise, no need to worry, just create it
-                 [seed]
-                end
-
-        # Transform all _id relations into the actual rails objects to pass validations
-        seeds = seeds.map do |seed|
-          columns = type.column_names.select { |name| name.match(/^.*_id$/) }
-          new_seed = seed.deep_symbolize_keys
-          columns.each do |column|
-            begin
-              column_type_name = column.chomp('_id')
-              column_type = column_type_name.classify.constantize
-              new_seed = new_seed.except(column.to_sym)
-                                 .merge(column_type_name.to_sym => column_type.unscoped.find(seed[column.to_sym]))
-            rescue StandardError
-              # Either the type does not exist or the value specified as the id is not valid, ignore.
-              next
-            end
-          end
-          new_seed
-        end
+        new_created, new_skipped = create_objects(type, seed)
+        created += new_created
+        skipped += new_skipped
 
-        # Actually create the objects and count successes
-        objs = type.create seeds
-        skipped += objs.select { |o| o.errors.any? }.size
-        created += objs.select { |o| !o.errors.any? }.size
+        if type == CommunityUser
+          ensure_system_user_abilities
+        end
       end
     end
     unless Rails.env.test?
-      puts "#{type}: Created #{created}, #{updated > 0 ? "updated #{updated}, " : ''}skipped #{skipped}"
+      puts "#{type}: errored #{errored}, created #{created}, #{updated > 0 ? "updated #{updated}, " : ''}skipped #{skipped}"
     end
   rescue StandardError => e
     puts "Got error #{e}. Continuing..."
diff --git a/db/seeds/abilities/edit_posts.html b/db/seeds/abilities/edit_posts.html
index ecb1741a4501e88f641001e85cd84a11b2a2d072..ff0fdff8387c91cd7f1e003cfdbe103597ec4dcb 100644
--- a/db/seeds/abilities/edit_posts.html
+++ b/db/seeds/abilities/edit_posts.html
@@ -1,9 +1,9 @@
 <h2>What does this ability allow me to do?</h2>
 <p>This ability allows you to unilaterally edit posts, as well as review suggested edits from other users.</p>
 
-<p>Edits should be used to improve posts, including edits improving spelling, grammar, and formatting, as well as fixes for missing <a href="/help/alt-text">alt text</a>, broken images and links, and other improvements. Edits should, however, respect the goal of the original poster.</p>
+<p>Edits should be used to improve posts, including edits improving spelling, grammar, and formatting, as well as fixes for missing <a href="/help/alt-text">alt text</a>, broken images and links, and other improvements. Edits should, however, preserve the meaning of the post.</p>
 
 <img src="/assets/ability-help/edit-link.png" alt="Edit post button">
 
 <h2>How do I earn this ability?</h2>
-<p>To earn this ability, you need to have at least a 95% approval rate for your suggested edits, with a hard minimum of 30 approved suggested edits (these numbers may vary from site to site).</p>
\ No newline at end of file
+<p>To earn this ability, you need to have at least a 95% approval rate for your suggested edits, with a hard minimum of 30 approved suggested edits (these numbers may vary from community to community).</p>
diff --git a/db/seeds/abilities/everyone.html b/db/seeds/abilities/everyone.html
index be547ccca9097de73471566354d1c1392f5a318e..8f112ef81f4a2d45a247a26001cfb88c90521fb1 100644
--- a/db/seeds/abilities/everyone.html
+++ b/db/seeds/abilities/everyone.html
@@ -1,9 +1,9 @@
 <h2>What does this ability allow me to do?</h2>
 
-<p>This ability allows you to posts 3 top-level posts (questions and articles) a day, and to post 20 answers a day.</p>
+<p>This ability allows you to post 3 top-level posts (questions and articles) per day, and to post 20 answers per day.</p>
 
-<p>This ability also allows you to raise 15 flags on posts a day.</p>
+<p>This ability also allows you to raise 15 flags on posts per day.</p>
 
 <h2>How do I earn this ability?</h2>
 
-<p>This ability is automatically granted to you when you create an account.</p>
\ No newline at end of file
+<p>This ability is automatically granted to you when you create an account.</p>
diff --git a/db/seeds/abilities/unrestricted.html b/db/seeds/abilities/unrestricted.html
index cce29e0e1857833cded28985820d66501eef6259..d10050344609d642d1a247812a47a9ce675d9522 100644
--- a/db/seeds/abilities/unrestricted.html
+++ b/db/seeds/abilities/unrestricted.html
@@ -2,4 +2,6 @@
 <p>This ability allows you to post questions, articles, and answers without limit, as well as being able to vote and leave comments on any post. You can see guidance on <a href="/help/how-to-ask">asking questions</a> and <a href="/help/voting">how to vote</a> in the Help Center.</p>
 
 <h2>How do I earn this ability?</h2>
-<p>To earn this ability, you need to have roughly 75% of your posts be positively received, with a minimum of 5 positively-recieved posts (these numbers may vary from site to site).</p>
+<p>To earn this ability, you need to have roughly 75% of your posts be positively received, with a minimum of 5 positively-received posts (these numbers may vary from site to site).</p>
+<p>If a site is in "new site" mode, all users start with this ability.</p>
+
diff --git a/db/seeds/community_users.yml b/db/seeds/community_users.yml
new file mode 100644
index 0000000000000000000000000000000000000000..4935d657d83ca0ce5cb36543677564f680448099
--- /dev/null
+++ b/db/seeds/community_users.yml
@@ -0,0 +1,7 @@
+- deleted: false
+  is_admin: true
+  is_moderator: true
+  is_suspended: false
+  post_count: 0
+  reputation: 1
+  user_id: -1
\ No newline at end of file
diff --git a/db/seeds/filters.yml b/db/seeds/filters.yml
new file mode 100644
index 0000000000000000000000000000000000000000..d23f91b9e7e885354501d47d127552f92052d140
--- /dev/null
+++ b/db/seeds/filters.yml
@@ -0,0 +1,14 @@
+- name: None
+  user_id: -1
+
+- name: Positive
+  user_id: -1
+  min_score: 0.5
+  max_score: 1
+  status: 'open'
+
+- name: Unanswered
+  user_id: -1
+  min_answers: 0
+  max_answers: 0
+  status: 'open'
\ No newline at end of file
diff --git a/db/seeds/post_flag_types.yml b/db/seeds/post_flag_types.yml
index 618f76e4e798e632c340f484175613c7ebd88066..5c3204aca5feb02b1d4b5fb67b080ba5d4bd001a 100644
--- a/db/seeds/post_flag_types.yml
+++ b/db/seeds/post_flag_types.yml
@@ -16,7 +16,7 @@
   description: >
     This question is off-topic or cannot be reasonably answered in its current form and needs revision by its author.
   confidential: false
-  active: true
+  active: false
   post_type_id: <%= Question.post_type_id %>
 
 - name: is a duplicate
@@ -24,4 +24,11 @@
     This question has been asked before and has already been answered. It should be marked as a duplicate.
   confidential: false
   active: true
-  post_type_id: <%= Question.post_type_id %>
\ No newline at end of file
+  post_type_id: <%= Question.post_type_id %>
+
+- name: generated by AI
+  description: >
+    This post appears to contain AI-generated content (such as ChatGPT) without [attribution](/help/referencing).
+  confidential: false
+  active: true
+  post_type_id: null
diff --git a/db/seeds/post_history_types.yml b/db/seeds/post_history_types.yml
index 9ed649d142ed676b7c4b4d9d60a78a11aa2b16ec..d12b6e28c0140913425be4812c1ef8a014c1fe87 100644
--- a/db/seeds/post_history_types.yml
+++ b/db/seeds/post_history_types.yml
@@ -9,4 +9,5 @@
 - name: attribution_notice_changed
 - name: imported_from_external_source
 - name: nominated_for_promotion
-- name: promotion_removed
\ No newline at end of file
+- name: promotion_removed
+- name: history_hidden
\ No newline at end of file
diff --git a/db/seeds/posts.yml b/db/seeds/posts.yml
index 8930150271ce32580b99c36aa24a47df30ff5fc1..f345061269d080262fd22ad481254e09bf8cf6de 100644
--- a/db/seeds/posts.yml
+++ b/db/seeds/posts.yml
@@ -38,6 +38,14 @@
   help_category: About the Network
   community_id: ~
 
+- post_type_id: <%= PolicyDoc.post_type_id %>
+  title: Accessibility Statement
+  body_markdown: $FILE posts/accessibility-policy.html
+  body: $FILE posts/accessibility-policy.html
+  doc_slug: accessibility-policy
+  help_category: About the Network
+  community_id: ~
+
 - post_type_id: <%= HelpDoc.post_type_id %>
   title: FAQ
   body_markdown: $FILE posts/local_faq.html
@@ -101,6 +109,13 @@
   doc_slug: licenses
   help_category: Guidance
 
+- post_type_id: <%= HelpDoc.post_type_id %>
+  title: Accessibility
+  body_markdown: $FILE posts/accessibility.html
+  body: $FILE posts/accessibility.html
+  doc_slug: accessibility
+  help_category: Guidance
+
 - post_type_id: <%= HelpDoc.post_type_id %>
   title: Search options
   body_markdown: $FILE posts/search.html
@@ -142,3 +157,11 @@
   body: $FILE posts/deleted.html
   doc_slug: deleted
   help_category: Site Information
+
+- post_type_id: <%= HelpDoc.post_type_id %>
+  title: Advanced formatting help
+  body_markdown: $FILE posts/advanced-formatting.html
+  body: $FILE posts/advanced-formatting.html
+  doc_slug: advanced-formatting
+  help_category: Site Information
+
diff --git a/db/seeds/posts/accessibility-policy.html b/db/seeds/posts/accessibility-policy.html
new file mode 100644
index 0000000000000000000000000000000000000000..b3f9eebf3d9de8ece4bd0996899fdc201de4b456
--- /dev/null
+++ b/db/seeds/posts/accessibility-policy.html
@@ -0,0 +1,41 @@
+| URL slug | Category |
+|:--------:|:--------:|
+| accessibility-policy | About the Network |
+
+<h1>$EDIT-Organization-Name Accessibility Statement</h1>
+
+<!--- Edit as applicable for your organization -->
+
+Sharing information and being part of a community are the two fundamental goals of the Codidact Project. Everyone deserves access to information and to be a part of a community, and with that in mind,  the Codidact Foundation is committed to accessibility in all areas of the Codidact Project.
+
+<h2>Who is responsible for accessibility?</h2>
+
+Accessibility is the responsibility of everyone, from the people writing posts to the people working on the website. However, the chief person responsible for accessibility in the $EDIT-Organization-Name is $Edit-Accessibility-Officer. Questions about $EDIT-Organization-Name's accessibility can be directed to $EDIT-email.
+
+<h2>I'm encountering an accessibility problem with the QPixel software, what do I do?</h2>
+
+Please report any accessibility problems you have on <a href="https://meta.codidact.com">Codidact Meta</a>, using the [bug] and [accessibility] tags. That makes sure that the Codidact developers are aware of the problem, and they'll do their best to fix it as soon as reasonably possible with their available resources.
+
+<h2>Technical details: QPixel Accessibility Policy</h2>
+
+This Accessibility Policy outlines the technical details of the commitment to accessibility for sites running the QPixel software and sets expectations for what is in scope of that commitment.
+
+<h3>The $EDIT-Organization-Name Network</h3>
+
+<!--- edit with your specific details --->
+
+The platform that the Network runs on, known as QPixel, should meet both the Web Content Accessibility Guidelines (WCAG) 2.2 and the Authoring Tool Accessibility Guidelines (ATAG) 2.0 standards at the AA conformance level.
+
+Since the platform is open-source and volunteer operated, setting strict timetables isn't realistic. However, new changes to the platform should be made with these standards in mind, including using Accessible Rich Internet Applications (WAI-ARIA) features.
+
+<h3>User-contributed content</h3>
+
+The $EDIT-Organization-Name encourages all people contributing posts, such as questions, answers, and articles, to make their posts accessible. This is done via system reminders, such as a warning when posting images without alt text, and via other users editing those posts to be more accessible.
+
+An in-depth guide to creating accessible posts is available in the help center.
+
+<h3>$EDIT other sites</h3>
+
+<!-- edit with any other sites controlled by your organization -->
+
+This document was last updated on [date].
diff --git a/db/seeds/posts/accessibility.html b/db/seeds/posts/accessibility.html
new file mode 100644
index 0000000000000000000000000000000000000000..d8d1b556eeaec7ad57a729834292da980b25797d
--- /dev/null
+++ b/db/seeds/posts/accessibility.html
@@ -0,0 +1,255 @@
+<p>This short guide goes over several aspects of accessibility to keep in mind when writing or editing posts on the Codidact network. There are many different disabilities out there, and these tips cover accessibility for many of them, such as people using screen readers, keyboard users, colorblind users, and people with cognitive disabilities.</p>
+<hr>
+<h2>Table of Contents</h2>
+<ul>
+<li>
+<a href="#Formatting">Formatting</a>
+<ul>
+<li>
+<a href="#Headings">Headings</a>
+<ul>
+<li><a href="#Headings-Examples">Examples</a></li>
+</ul>
+</li>
+<li><a href="#Emphasis">Emphasized text</a></li>
+<li><a href="#Code">Code markup</a></li>
+<li><a href="#Text-size">Text size</a></li>
+<li><a href="#Tables">Tables</a></li>
+</ul>
+</li>
+<li>
+<a href="#Images">Images</a>
+<ul>
+<li><a href="#Alt-text">Alternative (alt) text</a></li>
+<li><a href="#Decorative-images">Decorative images</a></li>
+<li>
+<a href="#Local-images">Stable resource (local images)</a>
+<ul>
+<li><a href="#Images-examples">Examples</a></li>
+</ul>
+</li>
+<li><a href="#Color">Color</a></li>
+<li><a href="#Contrast">Contrast</a></li>
+<li><a href="#Animations">Animations</a></li>
+<li>
+<a href="#Text-images">Images of text</a>
+<ul>
+<li><a href="#Quotes">Quoting</a></li>
+</ul>
+</li>
+</ul>
+</li>
+<li>
+<a href="#Links">Links</a>
+<ul>
+<li><a href="#Link-text">Link text</a></li>
+<li><a href="#Web-Archive">Stable resource (Web Archive)</a></li>
+</ul>
+</li>
+<li>
+<a href="#Understandable">Making your posts understandable</a>
+<ul>
+<li><a href="#Vocabulary">Vocabulary</a></li>
+<li><a href="#Jargon">Define acronyms and specialized terms</a></li>
+<li><a href="#Paragraphs">Paragraph and sentence breaks</a></li>
+</ul>
+</li>
+</ul>
+<hr>
+<h2 id="Formatting">Formatting</h2>
+<h3 id="Headings">Headings</h3>
+<p>Headings are used to separate sections of information. Each heading should give you a decent indication of what you can expect in that section. (As an example, the heading "headings" before this section is a good hint that this section is about headings.)</p>
+<ul>
+<li>
+<p>Follow an intuitive header order.</p>
+<p>Don't jump from an <code>&lt;h2&gt;</code> to an <code>&lt;h5&gt;</code>; follow a consistent, intuitive order for headings, where top-level sections have a higher-level header and sub-sections have a lower-level header. You shouldn't skip levels of headings (for instance, moving from an <code>&lt;h2&gt;</code> to an <code>&lt;h4&gt;</code> without an <code>&lt;h3&gt;</code> in between)<br>
+However, when you are ending a sub-section and moving back to a higher-level section, you can move from a low-level header (such as <code>&lt;h4&gt;</code>) back up to a higher-level header (such as <code>&lt;h2&gt;</code>) without including a mid-level header.</p>
+</li>
+<li>
+<p>There should generally only be one top-level header (which can be formatted using <code>&lt;h1&gt;</code> or <code>#</code>) per-page.</p>
+<p>Posts on the Codidact network have a top-level heading as the question or article title.<br>
+Whether or not using another top-level header is appropriate depends on the type of post you're writing, as well as on how the question and answers are structured. Posts covering a large number of topics or that are split up into multiple, entirely distinct sections sometimes might need to include more than one top-level heading, but consider carefully if it's actually necessary.</p>
+</li>
+<li>
+<p>Only use headings for actual headings, and don't use headings for non-header content.</p>
+<p>If something acts as a heading, it should be properly marked up as a heading, using either the appropriate Markdown or HTML tags. The opposite holds true as well; if something is not acting as a heading, it should not be formatted as a heading. If you want to emphasize something, don't use headings to do that.</p>
+</li>
+</ul>
+<p>Screen readers and similar can jump from heading to heading (including describing what level of heading), so keep that in mind when choosing headings for your posts.</p>
+<hr>
+<h4 id="Headings-Examples">Examples</h4>
+<p>A good heading order follows the logical layout of the page, without skipping over levels, consistently uses the same level headings for sections at the same level, and only uses headings for actual heading content.</p>
+<pre>// Example of a good heading order
+&lt;h1&gt;Question title (automatically provided)&lt;h1&gt;
+&lt;h2&gt;Answer section 1&lt;/h2&gt;
+&lt;h3&gt;Answer sub-section 1a&lt;/h3&gt;
+&lt;h3&gt;Answer sub-section 1b&lt;/h3&gt;
+&lt;h2&gt;Answer section 2&lt;/h2&gt;
+&lt;h3&gt;Answer sub-section 2a&lt;/h3&gt;
+&lt;h3&gt;Answer sub-section 2b&lt;/h3&gt;
+&lt;h4&gt;Answer sub-sub-section 2b.1&lt;/h4&gt;
+&lt;h4&gt;Answer sub-sub-section 2b.2&lt;/h4&gt;
+&lt;h2&gt;Answer section 3&lt;/h2&gt;
+</pre>
+<p>The heading order above shows an answer that doesn't use a top-level heading (<code>&lt;h1&gt;</code>), since there's already one on the page - the question title. It uses <code>&lt;h2&gt;</code> to differentiate its high-level sections. It then uses <code>&lt;h3&gt;</code> for the sub-sections, and when it's necessary to have a sub-sub-section, it uses <code>&lt;h4&gt;</code>.</p>
+<p>A bad heading order would skip levels, use headings for non-heading content, or unnecessarily over-use top-level headings.</p>
+<pre>// Example of a bad heading order
+&lt;h1&gt;Question title (automatically provided)&lt;/h1&gt;
+&lt;h1&gt;Answer summary (actual content)&lt;/h1&gt;
+&lt;h3&gt;Answer section 1&lt;/h3&gt;
+&lt;h5&gt;Answer sub-section 1a&lt;/h5&gt;
+&lt;h4&gt;Answer sub-section 1b&lt;/h4&gt;
+&lt;h2&gt;Answer section 2&lt;/h2&gt;
+&lt;h4&gt;Notes&lt;/h4&gt;
+&lt;h1&gt;Thank you for reading!&lt;/h1&gt;
+</pre>
+<p>This bad example uses <code>&lt;h1&gt;</code> twice in the answer, despite there already being a top-level heading as the question title. It uses an <code>&lt;h1&gt;</code> heading for the answer summary, using a heading for non-heading content. It then uses wildly inconsistent heading levels for its different sections and sub-sections, and skips levels of headings (such as moving directly from <code>&lt;h3&gt;</code> to <code>&lt;h5&gt;</code>).</p>
+<hr>
+<h3 id="Emphasis">Emphasized text</h3>
+<p>Emphasized text, such as bold or italics, is good for calling attention to key words or sentences. However, if it's overused, it ends up defeating its own purpose and making a post more difficult to read instead of clearer.</p>
+<ul>
+<li>
+<p>Bold text should be used sparingly.</p>
+<p>Only use it to highlight words or sentences that actually need special attention called to them.</p>
+</li>
+<li>
+<p>Italics should only be used when it is appropriate to use italics.</p>
+<p>Italics are used for emphasizing stress on a certain word, italicizing the names of works, indicating words in foreign languages, and other standard uses of italics. Don't over-use italics on text that doesn't need to be italicized; remember that italics can make it harder to read the text for certain people, including some people with dyslexia.</p>
+</li>
+<li>
+<p>Avoid bolding or italicizing entire paragraphs.</p>
+</li>
+</ul>
+<h3 id="Code">Code markup</h3>
+<ul>
+<li>
+<p>Code markup should be used for code, including variable names and other code elements that may be found in non-code lines.</p>
+<p>This allows for code highlighting to work, and makes it clear when a code element is being referred to.</p>
+</li>
+<li>
+<p>Code markup should not be used for any non-code elements.</p>
+<p>This includes using it for emphasis, for tables, or other non-code usage. Instead, use the <a href="https://meta.codidact.com/help/formatting">dedicated formatting</a> for those elements. Misusing code markup can cause issues for assistive technology such as screen readers.</p>
+</li>
+</ul>
+<h3 id="Text-size">Text size</h3>
+<ul>
+<li>
+<p>Don't stack subscripts or superscripts to make your text tiny.</p>
+<p>Using superscript or subscript once is enough, and only use it when necessary. Screen readers may not differentiate between sub- or superscript and regular text, so keep that in mind.</p>
+</li>
+</ul>
+<p>Codidact has integrated footnotes available, so you should avoid using sub- and superscript for footnotes; use the dedicated Markdown instead.</p>
+<h3 id="Tables">Tables</h3>
+<ul>
+<li>
+<p>If possible, avoid putting ambiguous data into tables - i.e., having data that you can't tell which column of the table it would be associated with without checking, such as having two columns containing plain numbers.</p>
+</li>
+<li>
+<p>Avoid blank header rows in tables, and don't use table formatting for data that doesn't actually belong in a table.</p>
+</li>
+</ul>
+<h2 id="Images">Images</h2>
+<h3 id="Alt-text">Alternative (alt) text</h3>
+<ul>
+<li>
+<p>Whenever you include an image in a post, you should include alternative text (commonly called "alt text") that serves the same purpose as the image.</p>
+<p>This replaces the default text of "Image_alt_text". This is used by screen readers, search engines, and when images can't be displayed (such as images being blocked in certain countries or by school/business networks).</p>
+</li>
+<li>
+<p>The alt text should be short, succinct, and serve the exact same purpose as the image - it shouldn't contain more or less information than the image itself.</p>
+</li>
+</ul>
+<p>As a general way of making sure your alt text is appropriate, consider if the information present in the post would change at all if the image was replaced entirely with the alt text. If the information would stay the same, you're good to go.</p>
+<h3 id="Decorative-images">Decorative images</h3>
+<ul>
+<li>A decorative image, which serves no purpose other than visual, should have its alt text be entirely blank.</li>
+</ul>
+<p>Note that this is <em>blank</em>, not <em>missing</em>. From a coding perspective, this means setting its <code>alt</code> attribute to <code>=""</code>, not leaving out the <code>alt</code> attribute.</p>
+<p>In general, you should avoid including images that don't serve any specific purpose or that are just decorative in your post.<br>
+If you find yourself including a decorative image, make sure that it's not formatted as a link, leaving only the embedded image, and to set the alt text to be blank.</p>
+<h3 id="Local-images">Stable resource (local images)</h3>
+<p>In order to avoid the possibility of dead images, and ensure that that post remains stable for as long as possible, you should avoid using an external image hosting service. Instead, use the built-in image uploader for the Codidact Network. This means that the image is stored on our own servers instead of depending on someone else.</p>
+<hr>
+<h4 id="Images-examples">Examples</h4>
+<p>Let's take the following snippet of a post for our example:</p>
+<blockquote>
+<p>When you go to edit a post, you now have the option to check the "redact" button:</p>
+<p><img alt="Checkbox to select &quot;redact&quot;: Redact original content by hiding the previous versions from history?" src="https://meta.codidact.com/uploads/qwvy25mvolpdjabhknuujzjncbpm"><br>
+<sub>(Source: "<a href="https://meta.codidact.com/posts/289192/289193#answer-289193">What should I do when I come across PII in a post?</a>" by Mithical on Codidact Meta, licensed under CC BY-SA-NC 4.0)</sub></p>
+</blockquote>
+<p>In this example, the image is being used to illustrate the new button and what it does. That information needs to be presented in the alt text as well, which the current alt text does:</p>
+  <pre>![Checkbox to select "redact": Redact original content by hiding the previous versions from history?](https://meta.codidact.com/uploads/qwvy25mvolpdjabhknuujzjncbpm)
+</pre>
+<p>This is short and to the point. It tells anybody who can't see the image what information is shown with the checkbox, which is why the screenshot was included.</p>
+<p>A bad example would be leaving out the alt text, having overly long alt text, or relying on an external image hosting service:</p>
+  <pre>![](https://i.imgur.com/XtjJxjF.jpg) 
+
+![A screenshot of two different sections that you see when editing a post. One says "Edit Comment" with a blank text box, where you can input an edit comment that will show up in the revision history for that post. The other says "Redact", and it has text with an unchecked checkbox. It has the text "Redact original content by hiding the previous revisions from history?" Below those are two buttons. One is blue and says "Save Post in Q&amp;A". The other is gray and says "Cancel".](https://meta.codidact.com/uploads/qwvy25mvolpdjabhknuujzjncbpm)
+</pre>
+<p>In the first bad example, there is both no alt text and it relies on an external image service. There is no information presented to anyone who can't see the image, and there's the risk that the image will go dead even for people who can see it.<br>
+In the second bad example, the alt text is too long. It has information that's not present in the image itself, such as information about the edit comment appearing in the revision history, and describes information that's not relevant to the purpose of the screenshot - which is simply to show what you're presented with when you go to redact a post.</p>
+<hr>
+<h3 id="Color">Don't rely solely on color</h3>
+<p>When your image uses colors to indicate a difference between things - such as on a chart or graph - you should also use a different method of differentiating, such as an icon or different shape. Also avoid using colors that are known to be a problem for colorblind users (such as red/green).</p>
+<h3 id="Contrast">Contrast</h3>
+<p>Avoid colors that are too close to each other, especially for text on a background color. As a simple way of testing, take a glance at the image in sunlight - can you still make it out?</p>
+<h3 id="Animations">Animations</h3>
+<p>We don't currently support any way to disable or pause animations in posts, so avoid using animations where possible. In particular, make sure to avoid flashing content (especially anything flashing more than three times a second - don't do that!). Flashing content can cause seizures, and looping animations can be distracting for everyone, but especially for people with some cognitive disabilities.</p>
+<h3 id="Text-images">Images of text</h3>
+<ul>
+<li>
+<p>Avoid images of text.</p>
+<p>Images of text can't have the text selected, be read by screen readers, indexed by search engines, have the text adjust in a responsive design, or have the font changed. This includes images of code; instead, put the actual code in your post and format it using the dedicated code formatting Markdown.</p>
+</li>
+</ul>
+<h4 id="Quotes">Quoting</h4>
+<ul>
+<li>
+<p>If you are quoting from somewhere, don't provide an image of the text; use text, formatted as a blockquote (which can be done by putting a <code>&gt;</code> at the beginning of a paragraph), and cite your source.</p>
+<p>This applies to both online and offline resources, such as Wikipedia or a physical book.</p>
+</li>
+<li>
+<p>Do not use code formatting for quotes.</p>
+</li>
+</ul>
+<h2 id="Links">Links</h2>
+<h3 id="Link-text">Link text</h3>
+<ul>
+<li>
+<p>Avoid link text such as "Here" or "Read more".</p>
+<p>The link should explain its purpose through the text itself. Remember that screen readers and similar tools can jump to specific links, but if they're named something like "this", navigating to the correct link is much harder.<br>
+Don't go too far in the other direction, though; there's no need to make an entire sentence a link as long as the link text is descriptive and distinct.</p>
+</li>
+<li>
+<p>In general, link text should be unique - don't use the same link text twice in one post if those links go to different places.</p>
+</li>
+</ul>
+<h3 id="Web-Archive">Stable resource (Web Archive)</h3>
+<p>While not required by any standard that I'm aware of, I'd encourage you to take steps to make sure that any resource you link to remains stable by archiving it in the Web Archive when you link to it. (This is similar to what Wikipedia does; sources used in articles are almost always archived so that a backup exists.)</p>
+<h2 id="Understandable">Making your posts understandable</h2>
+<h3 id="Vocabulary">Vocabulary</h3>
+<ul>
+<li>
+<p>In general, try to keep your vocabulary simple.</p>
+<p>This doesn't mean avoiding all technical terms, or not using the correct terms for things, but don't use jargon or fancy words when it's not necessary. This makes it easier for people who don't speak English as their first language, or people with cognitive disabilities, to understand your post.</p>
+</li>
+</ul>
+<p>This should not come at the expense of precision or accuracy, though; keep your audience in mind. If you are writing for a highly specialized or technical audience, you shouldn't necessary shy away from using the relevant terminology; but if you're writing for everyone, using relatively simple terms is often a good idea.</p>
+<h3 id="Jargon">Define acronyms and specialized terms</h3>
+<ul>
+<li>
+<p>The first time you use an acronym, you should fully spell out what you're referring to.</p>
+<p>A common example found on Codidact (CD) is just that - the acronym CD. However, that acronym can also refer to the terminal command <code>cd</code>, or the physical medium of Compact Discs, as well as other meanings. Once you've defined what CD stands for in your specific post, you can continue to use the acronym.</p>
+</li>
+</ul>
+<p>This applies to specialized terms, jargon, and words in other languages as well. The first time you use a specialized term, define what it means or translate it.</p>
+<h3 id="Paragraphs">Paragraph and sentence breaks</h3>
+<ul>
+<li>
+<p>Avoid walls of text.</p>
+<p>Make sure to break up your posts into sections, paragraphs, and sentences. When something runs on for too long, or is too dense, it can be very hard for people to get through, especially people who don't speak the language well or people with certain cognitive disabilities.</p>
+</li>
+</ul>
+<hr>
+<p>And those are the top tips for making your post accessible! Remember that accessibility is an ongoing process, so don't feel too bad if not all of your posts meet these guidelines. They can always be edited later, and the important thing is to keep accessibility in mind as you go forwards writing and editing posts.</p>
+
diff --git a/db/seeds/posts/advanced-formatting.html b/db/seeds/posts/advanced-formatting.html
new file mode 100644
index 0000000000000000000000000000000000000000..6373aa4261847b07482e905b8bcc2d43c1d34df0
--- /dev/null
+++ b/db/seeds/posts/advanced-formatting.html
@@ -0,0 +1,86 @@
+<p
+    >Posts on Codidact, including questions, answers, articles, tag wikis, help pages, and user
+    profiles, support the use of the following HTML tags:</p
+>
+<ul>
+    <li><code>a</code></li>
+    <li><code>p</code></li>
+    <li><code>span</code></li>
+    <li><code>b</code></li>
+    <li><code>i</code></li>
+    <li><code>em</code></li>
+    <li><code>s</code></li>
+    <li><code>strong</code></li>
+    <li><code>hr</code></li>
+    <li
+        ><code>h1</code>, <code>h2</code>, <code>h3</code>, <code>h4</code>, <code>h5</code>,
+        <code>h6</code></li
+    >
+    <li><code>blockquote</code></li>
+    <li><code>img</code></li>
+    <li><code>strike</code></li>
+    <li><code>del</code>, <code>ins</code></li>
+    <li><code>code</code></li>
+    <li><code>pre</code></li>
+    <li><code>br</code></li>
+    <li><code>ul</code>, <code>ol</code>, <code>li</code></li>
+    <li><code>sup</code>, <code>sub</code></li>
+    <li><code>section</code></li>
+    <li><code>details</code>, <code>summary</code></li>
+    <li
+        ><code>table</code>, <code>thead</code>, <code>tbody</code>, <code>tr</code>,
+        <code>th</code>, <code>td</code></li
+    >
+</ul>
+<p>These tags may have the following attributes:</p>
+<ul>
+    <li><code>id</code></li>
+    <li><code>class</code></li>
+    <li><code>href</code></li>
+    <li><code>title</code></li>
+    <li><code>src</code></li>
+    <li><code>height</code></li>
+    <li><code>width</code></li>
+    <li><code>alt</code></li>
+    <li><code>dir</code></li>
+    <li><code>lang</code></li>
+    <li><code>start</code></li>
+    <li><code>rowspan</code></li>
+    <li><code>colspan</code></li>
+</ul>
+<p>Shorter text forms, such as comments, support the following HTML tags:</p>
+<ul>
+    <li><code>a</code></li>
+    <li><code>b</code></li>
+    <li><code>i</code></li>
+    <li><code>em</code></li>
+    <li><code>strong</code></li>
+    <li><code>strike</code></li>
+    <li><code>del</code></li>
+    <li><code>code</code></li>
+</ul>
+<p>These tags may have the following attributes:</p>
+<ul>
+    <li><code>href</code></li>
+    <li><code>title</code></li>
+</ul>
+<hr />
+<p
+    >HTML tags that do not appear on this list will be stripped out and not displayed if they are
+    included in posts or comments. The source code for supported tags in posts can be found in
+    <a href="https://github.com/codidact/qpixel/blob/develop/app/helpers/posts_helper.rb#L46"
+        ><code>posts_helper.rb</code></a
+    >, and for comments in
+    <a href="https://github.com/codidact/qpixel/blob/develop/app/helpers/comments_helper.rb#L81"
+        ><code>comments_helper.rb</code></a
+    >.</p
+>
+<hr />
+<p
+    ><sub
+        >Much of the content of this page came from
+        <a href="https://meta.codidact.com/posts/277420/277424#answer-277424"
+            >this answer by luap42</a
+        >.</sub
+    ></p
+>
diff --git a/db/seeds/posts/category-permissions.html b/db/seeds/posts/category-permissions.html
index 922970152264536ff445f2e1bcacfe6df3dec6d9..8bc18a9b410b82f46dfe21a4739f77446d076225 100644
--- a/db/seeds/posts/category-permissions.html
+++ b/db/seeds/posts/category-permissions.html
@@ -6,7 +6,7 @@
 <p>Please note that trust levels are hierarchical, so that anyone who has one trust level has also all lower trust levels by default.</p>
 
 <h3>1. Everyone</h3>
-<p>As the name says, **absolutely everyone** has this trust level. This includes anonymous (unregistered) and new users.</p>
+<p>As the name says, <strong>absolutely everyone</strong> has this trust level. This includes anonymous (unregistered) and new users.</p>
 <p>Most write actions will require a registered user account, though, even if the trust level "everyone" has been selected.</p>
 
 <h3>2. Anyone with a user account</h3>
@@ -19,7 +19,7 @@
 <p>For the sake of this trust level, every user with either the <a href="/abilities/edit_posts">Edit Posts</a> or the <a href="/abilities/flag_close">Vote on Hold</a> is considered "veteran".</p>
 
 <h3>5. Moderators only</h3>
-<p>Moderators, Adminis and the global variants thereof are the users that have this trust level.</p>
+<p>Moderators, Admins, and the global variants thereof are the users that have this trust level.</p>
 
 <h3>6. Staff only</h3>
-<p>This trust level is only reached by users with the staff mark.</p>
\ No newline at end of file
+<p>This trust level is only reached by users with the staff mark.</p>
diff --git a/db/seeds/posts/formatting.html b/db/seeds/posts/formatting.html
index e9ad8a39dd21e426762a0cb35418db632e059f14..4e4e966b4dd5580cd8cb4a1a1d42d561f7368e2a 100644
--- a/db/seeds/posts/formatting.html
+++ b/db/seeds/posts/formatting.html
@@ -64,3 +64,26 @@
 </blockquote>
 <h3 id="-footnotes-"><strong>Footnotes</strong></h3>
 <p>  To include a footnote in your post, you can use the syntax <code>[^1]</code>. In your main text, include <code>Text[^1] and more text</code>, and at the bottom (where you want to include your footnote), place a line resembling <code>[^1]: footnote text</code>.</p>
+
+<h3 id="-details-"><strong>Hidden Sections</strong></h3>
+<p>CommonMark does not support collapsible sections (sometimes called "spoiler blocks"), but you can use the HTML <code>details</code> and <code>summary</code> tags, like this:</p>
+<pre>&lt;details&gt;
+    &lt;summary&gt;Spoiler! Click here to reveal&lt;/summary&gt;
+    Secret details
+&lt;/details&gt;</pre>
+<p>Which renders like this:</p>
+<details>
+<summary>Spoiler! Click here to reveal</summary>
+Secret details
+</details>
+<p>If the details text uses any Markdown, you must add a blank line between the summary and the text:</p>
+<pre>&lt;details&gt;
+    &lt;summary&gt;Spoiler! Click here to reveal&lt;/summary&gt;
+
+    *Secret* details 
+&lt;/details&gt;</pre>
+<p>Renders:</p>
+<details>
+<summary>Spoiler! Click here to reveal</summary>
+<i>Secret</i> details
+</details>
diff --git a/db/seeds/posts/global_faq.html b/db/seeds/posts/global_faq.html
index 3830f2a4f5075f9dea92dbb9d04ebfa710b1486f..6fb50d0684ad2ac8691331ab911206ca9f1e3c2d 100644
--- a/db/seeds/posts/global_faq.html
+++ b/db/seeds/posts/global_faq.html
@@ -1,4 +1,5 @@
-<!-- This section appears under a "Global FAQ" heading in the help.  Insert something about your network here. We've provided some other sections as starting points.  The section about the Codidact software and project is meant to go at the end, after the stuff about *your* network and communities. -->
+<!-- This section appears under a "Global FAQ" heading in the help.  Insert something about your network here. We've provided some other sections as starting points.  Please ADD A SUPPORT CONTACT METHOD in the support section! -->
+<!-- The section about the Codidact software and project is meant to go at the end, after the stuff about *your* network and communities. -->
 
 <h2 id="how-do-i-ask-a-question-">How do I ask a question?</h2>
 <p>You will need to register an account. Make sure you are logged into the appropriate topic-related site for your question and click the &quot;Ask Question&quot; button at the top of the page. A template should appear with helpful guidelines on how to ask a quality question. Check out the local FAQ page for important information about each community.</p>
@@ -7,7 +8,9 @@
 <h2 id="what-license-do-my-posts-fall-under-">What license do my posts fall under?</h2>
 <p>Most posts fall under <a href="https://creativecommons.org/licenses/by-sa/4.0/">CC BY-SA 4.0</a>. Some communities or contributors may use a license that more closely aligns with the content posted or their ideological position. Check a community&#39;s local FAQ page and Terms of Service for site-specific details. </p>
 
+<!-- uncomment and populate: -->
+<!-- <h2 id="support">Where can I get support?</h2> -->
 
 <h2 id="what-is-codidact-">What is Codidact?</h2>
-<p>This community is built using the <a href="https://github.com/codidact/qpixel">open-source QPixel software</a> provided by <a href="https://codidact.org">The Codidact Foundation</a>. The software is free, open-source, and customizable.</p>
+<p>This community is built with the <a href="https://github.com/codidact/qpixel">QPixel engine</a> provided by <a href="https://codidact.org">The Codidact Foundation</a>. The software is free, open-source, and highly customizable.</p>
   
diff --git a/db/seeds/posts/scoring.html b/db/seeds/posts/scoring.html
index 20aeb6e4c3b892904277e2a65a1f80e3c7b06767..ed1795d5c358c88f31ea0754bee1430fea0fdf21 100644
--- a/db/seeds/posts/scoring.html
+++ b/db/seeds/posts/scoring.html
@@ -4,7 +4,7 @@
 <p>The actual score of a particular post is calculated as</p>
 <img src="/assets/scoring_formula.png" alt="(upvote_count + z^2/2)/(upvote_count + downvote_count + z^2)">
 <p>This uses a modified form of a <a href="https://en.wikipedia.org/wiki/Binomial_proportion_confidence_interval#Wilson_score_interval">Wilson Score</a>, assuming a binomial distribution. By default the weighting constant <em>z</em> is two, but each community can change this to another value as they see fit. Several charts and graphs showing the default weighting distribution are shown below. </p>
-<img src="/assets/scoring_table.png" alt="table of scores with the values for all upvote and downvote combinations from one to ten">
+<img src="/assets/scoring_table.svg" alt="table of scores with the values for all upvote and downvote combinations from one to ten">
 <img src="/assets/scoring_distribution.png" alt="lines in a cartesian coordinate plane sloping from upper left to lower right and converging as z increases">
 <p>Diagram showing the distribution in scores where upvotes less downvotes = 3 for different values of z</p>
 <img src="/assets/scoring_3d.png" alt="a 3D diagram showing the score distribution for upvotes and downvotes">
diff --git a/db/seeds/posts/search.html b/db/seeds/posts/search.html
index 8215a42c619f43524a4931e39cbd309cd3f6390c..e21de8a20c0b7f5efe0afdd67b9f1a39c4cbed30 100644
--- a/db/seeds/posts/search.html
+++ b/db/seeds/posts/search.html
@@ -17,20 +17,39 @@
 </ul>
 <p><strong>Filtering by score and age</strong></p>
 <p>It&#39;s possible to filter your search to only include results that have been posted within a certain timeframe, or match certain score requirements.</p>
+<p>You can use <code>&gt;</code>, <code>&gt;=</code>, <code>&lt;</code>, and <code>&lt;=</code> with these options to search for ranges. By default, a value without an operator looks for an exact match. For example, <code>upvotes:4</code> searches for exactly 4; <code>upvotes:&gt;=4</code> searches for at least 4.</p>
 <ul>
     <li><p>filtering by post score</p>
-        <p>Codidact uses Wilson scoring to help in sorting posts. (To learn more about how this works, see <a href="/help/scoring">/help/scoring</a> for a detailed explanation.) Every post has a score between 0.0 and 1.0. To use this in search, you can use <code>score:0.5</code> to filter your search to only include posts with a score of at least 0.5.</p>
+        <p>Codidact uses Wilson scoring to help in sorting posts. (To learn more about how this works, see <a href="/help/scoring">/help/scoring</a> for a detailed explanation.) Every post has a score between 0.0 and 1.0. To use this in search, you can use <code>score:&gt;=0.5</code> to filter your search to only include posts with a score of at least 0.5.</p>
     </li>
     <li><p>filtering by votes</p>
-        <p>If you want to filter by the raw votes that a post has, you can use <code>votes:5</code> to find posts where the net votes (upvotes minus downvotes) of a post equals 5 or higher.</p>
+        <p>If you want to filter by the raw votes that a post has, you can use <code>votes:5</code> to find posts where the net votes (upvotes minus downvotes) of a post equals 5 or <code>votes:&gt;=5</code> for votes of 5 or more.</p>
     </li>
     <li><p>filtering by upvotes and downvotes</p>
-        <p>If you search for <code>upvotes:4</code>, Codidact will find posts that have received at least 4 upvotes, irrespective of how many downvotes the post has. Likewise, if you search for <code>downvotes:4</code>, Codidact will find posts that have received at least 4 downvotes without taking upvotes into consideration. You can also use a less than (<code>&lt;</code>) symbol to filter for posts that have received no more than a certain number of votes (for instance, <code>downvotes:&lt;4</code> will find posts that have received less than four downvotes total).</p>
+      <p>If you search for <code>upvotes:4</code>, Codidact will find posts that have received exactly 4 upvotes, irrespective of how many downvotes the post has. Likewise, if you search for <code>downvotes:4</code>, Codidact will find posts that have received exactly 4 downvotes without taking upvotes into consideration. <code>downvotes:&lt;4</code> will find posts that have received fewer than four downvotes total).</p>
+    </li>
+    <li><p>filtering by number of answers</p>
+      <p>If you want to find posts with <code>n</code> answers, use <code>answers:n</code>. This is particularly helpful to find unanswered questions: <code>answers:0</code>. <code>answers:&lt;5</code> shows posts with fewer than five answers.</p>
     </li>
     <li><p>filtering by creation date</p>
         <p>If you want to only find posts that have been written within a certain timeframe, you can use the <code>created:</code> search operator. <code>created:&lt;1w</code> will find all posts created less than a week ago, where <code>created:&gt;1w</code> will find only posts older than a week. You can use <code>m</code> for minute, <code>h</code> for hour, <code>d</code> for day, <code>w</code> for week, <code>mo</code> for month, and <code>y</code> for year. </p>
     </li>
 </ul>
+<p><strong>Filtering by tag, user, category, or post type</strong></p>
+<ul>
+  <li><p>filtering by tag</p>
+    <p>To filter for all posts with the tag <code>snake</code>, use the <code>tag:snake</code> operator. To exclude all posts with the tag <code>oil</code>, use <code>-tag:oil</code>.</p>
+  </li>
+  <li><p>filtering by user</p>
+    <p>If you want to search for posts written by a particular user you will need to know their unique user number for the community. This can be found by looking at their profile URL. You can then use <code>user:xxxx</code> where <code>xxxx</code> is the unique user number you are interested in.</p>
+  </li>
+  <li><p>filtering by category</p>
+    <p>To filter by category, you will need to know the unique numeric ID for that category. This can be found by looking at the URL shown when you click to view all posts in a particular category. Use the formatting <code>category:xxxx</code> to apply this filter.</p>
+  </li>
+  <li><p>filtering by post type</p>
+    <p>If you want to restrict your search to a particular post type, type <code>post_type:</code> into the search bar, and a dropdown menu should show the available post types to search. Note that not all communities may use all post types.</p>
+  </li>
+</ul>
 <p><strong>Advanced</strong></p>
 <ul>
     <li><p>wildcard</p>
diff --git a/db/seeds/site_settings.yml b/db/seeds/site_settings.yml
index cb91ffc3be4b49cf972ef57b10defbe1b2337491..48a1ca475eb6e492aca5e4c9ed43ab01a9aea032 100644
--- a/db/seeds/site_settings.yml
+++ b/db/seeds/site_settings.yml
@@ -150,14 +150,21 @@
   description: >
     The maximum characters a single tag name may contain. Default is 35 for compatibility with Stack Exchange; going lower
     may introduce validation issues with content imported from SE.
-    
+
 - name: MaxTitleLength
   value: 150
   value_type: integer
   category: SiteDetails
   description: >
     The maximum characters a post title may contain. Default is 150 for compatibility with Stack Exchange; going lower
-    may introduce validation issues with content imported from SE.    
+    may introduce validation issues with content imported from SE.
+
+- name: MaxEditCommentLength
+  value: 255
+  value_type: integer
+  category: SiteDetails
+  description: >
+    The maximum number of characters an edit comment may contain. Defaults to 255.
 
 - name: MaxUploadSize
   value: 2MB
@@ -178,10 +185,122 @@
 - name: InterestingSubscriptionScoreThreshold
   value: 1
   value_type: integer
-  category: EmailSubscriptions
+  category: Email
   description: >
     The minimum score a question must have to qualify for selection for the Interesting email subscription.
 
+- name: SubscriptionSenderName
+  value: Codidact Subscriptions
+  value_type: string
+  category: Email
+  description: >
+    The name of the sender of subscription emails.
+
+- name: SubscriptionSenderEmail
+  value: subscriptions@codidact.com
+  value_type: string
+  category: Email
+  description: >
+    The address to send subscription emails from (can be a fake address).
+    Make sure your server is allowed to send email from this address, or your mails will not be received.
+
+- name: ModeratorDistributionListSenderName
+  value: Codidact Admins
+  value_type: string
+  category: Email
+  description: >
+    The name of the sender of the moderator distribution list.
+
+- name: ModeratorDistributionListSenderEmail
+  value: moderators-noreply@codidact.com
+  value_type: string
+  category: Email
+  description: >
+    The address to send moderator distribution list emails from (can be a fake address).
+    Make sure your server is allowed to send email from this address, or your mails will not be received.
+
+- name: NoReplySenderName
+  value: Codidact
+  value_type: string
+  category: Email
+  description: >
+    The name of the sender of no-reply emails.
+
+- name: NoReplySenderEmail
+  value: noreply@codidact.com
+  value_type: string
+  category: Email
+  description: >
+    The address to send no-reply emails from (can be a fake address).
+    Example uses of this address are 2FA emails, flag notifications, account emails, and more.
+    Make sure your server is allowed to send email from this address, or your mails will not be received.
+
+- name: DonationSupportReceiverName
+  value: Codidact Support
+  value_type: string
+  category: Email
+  description: >
+    The name of the donation support email address.
+
+- name: DonationSupportReceiverEmail
+  value: support@codidact.com
+  value_type: string
+  category: Email
+  description: >
+    The (real) address to receive donation support emails on. Used for users who want to reply to donation emails.
+
+- name: DonationSenderName
+  value: Codidact Donations
+  value_type: string
+  category: Email
+  description: >
+    The name of the sender of donation related emails.
+
+- name: DonationSenderEmail
+  value: donations-support@codidact.com
+  value_type: string
+  category: Email
+  description: >
+    The address to send donation related emails from (can be a fake address).
+    Make sure your server is allowed to send email from this address, or your mails will not be received.
+
+- name: SupportReceiverName
+  value: Codidact Support
+  value_type: string
+  category: Email
+  description: >
+    The name of the donation support email address.
+
+- name: SupportReceiverEmail
+  value: info@codidact.org
+  value_type: string
+  category: Email
+  description: >
+    The (real) address to receive support emails on.
+
+- name: AllUsersSenderName
+  value: Codidact Team
+  value_type: string
+  category: Email
+  description: >
+    The name of the sender of emails sent to all users of the network via the admin tools.
+
+- name: AllUsersSenderEmail
+  value: allusers-noreply@codidact.org
+  value_type: string
+  category: Email
+  description: >
+    The address to use as sender for emails sent to all users of the network via the admin tools (can be a fake address).
+    Make sure your server is allowed to send email from this address, or your mails will not be received.
+
+- name: AllUsersReplyToEmail
+  value: info@codidact.org
+  value_type: string
+  category: Email
+  description: >
+    The (real) address to use as reply-to for emails sent to all users of the network via the admin tools.
+
+
 - name: LotteryAgeDeprecationSpeed
   value: 0.002
   value_type: float
@@ -515,3 +634,86 @@
   description: >
     The content of a post is shown in short in lists (e.g. category post overview or in search).
     This setting controls how many characters of a post are shown.
+
+- name: NetworkPitch
+  value: $FILE site_settings/widgets_network_pitch.html
+  value_type: text
+  community_id: ~
+  category: Widgets
+  description: >
+    Sidebar text to promote the rest of the network.
+  
+- name: SsoSignIn
+  value: false
+  value_type: boolean
+  community_id: ~
+  category: SignInAndSignUp
+  description: >
+    Whether to enable SSO Sign in. If enabled, this replaces normal Sign In and Sign Up unless if Mixed Sign In is enabled.
+    NOTE: This requires a SAML provider to be configured and set up.
+
+- name: MixedSignIn
+  value: false
+  value_type: boolean
+  community_id: ~
+  category: SignInAndSignUp
+  description: >
+    This setting only has an effect when SSO Sign In is enabled. Enables mixed sign in: both signing in and signing up as well as SSO sign in are enabled.
+    If disabled, only one sign in method is enabled.
+
+- name: Enable2FAForSsoUsers
+  value: false
+  value_type: boolean
+  community_id: ~
+  category: SignInAndSignUp
+  description: >
+    This setting only has an effect when SSO Sign In is enabled. Enables 2FA options (and enforces 2FA for global mods and admins if configured) also for SSO users.
+    When the authentication is outsourced to a Single Sign-On provider (which may already require 2FA), it often does not make sense to have an additional 2FA check on top of that.
+
+- name: AllowSsoDisconnect
+  value: false
+  value_type: boolean
+  community_id: ~
+  category: SignInAndSignUp
+  description: >
+    This setting only has an effect when SSO Sign In and Mixed Sign In are enabled. Allows users to disconnect their account from SSO and switch over to normal login.
+
+- name: LoadStripeEverywhere
+  value: false
+  value_type: boolean
+  category: Integrations
+  description: >
+    Load Stripe JS API on all pages instead of just donation pages. May improve security and fraud detection.
+
+- name: RestrictedAccess
+  value: false
+  value_type: boolean
+  category: SiteDetails
+  description: >
+    Whether the content of this community should be visible only to users who are signed in.
+
+- name: RestrictedAccessFrontPageText
+  value: >
+    <h1>Welcome to our community!</h1>
+    <p>Please sign in to continue</p>
+  value_type: text
+  category: SiteDetails
+  description: >
+    This setting only has an effect when RestrictedAccess is enabled.
+    This is the text that will be displayed on the front page for users who are not signed in. Markdown allowed.
+
+- name: RestrictedAccessHelpPagesPublic
+  value: true
+  value_type: boolean
+  category: SiteDetails
+  description: >
+    This setting only has an effect when RestrictedAccess is enabled.
+    Whether the help pages are publicly accessible.
+
+- name: RestrictedAccessPolicyPagesPublic
+  value: true
+  value_type: boolean
+  category: SiteDetails
+  description: >
+    This setting only has an effect when RestrictedAccess is enabled.
+    Whether the policy pages are publicly accessible (Terms of Service, Privacy Policy, etc.).
diff --git a/db/seeds/site_settings/widgets_network_pitch.html b/db/seeds/site_settings/widgets_network_pitch.html
new file mode 100644
index 0000000000000000000000000000000000000000..4038bfe1f18a81ef92f0954c557d8213757aa810
--- /dev/null
+++ b/db/seeds/site_settings/widgets_network_pitch.html
@@ -0,0 +1 @@
+<p>This community is part of a <a href="/dashboard">network of communities</a> &#8212; take a look!</p>
diff --git a/db/seeds/tag_sets.yml b/db/seeds/tag_sets.yml
index d675cb51a7fb2a9d26dcb8ce0c52b245f92dfeea..8320d8bf57f8d734cd354e874e842f54b87d4d91 100644
--- a/db/seeds/tag_sets.yml
+++ b/db/seeds/tag_sets.yml
@@ -1,5 +1,2 @@
 - name: Main
 - name: Meta
-- name: Tour
-  id: -1
-  community_id: 1
\ No newline at end of file
diff --git a/db/seeds/warning_templates/off-topic.md b/db/seeds/warning_templates/off-topic.md
index f682b8589a4b01431fcc6359cece45d858d11d99..f4f205c5fd9e62e11bc7ba781e697c73bc1c6765 100644
--- a/db/seeds/warning_templates/off-topic.md
+++ b/db/seeds/warning_templates/off-topic.md
@@ -3,7 +3,9 @@ We've noticed that you've written a number of posts about topics that are beyond
 You can find out about what's on-topic and what's off-topic on $SiteName in the [help center](/help/faq).
 
 This is just a gentle reminder that we expect posts on this site to stay focused on the topic on-hand.
-We have a [Network of communities](https://codidact.com/) that you are free to use; you may find one of our other communities more suitable to some of your posts. If you would like to talk about the possibility of creating a site for a subject not currently covered, please write a post on [meta](https://meta.codidact.com/categories/10) with your request.
+We have a [Network of communities](/dashboard) that you are free to use; you may find one of our other communities more suitable to some of your posts.
+If you would like to explore creating a site for a subject not currently covered, see [the Proposals process on codidact.com](https://proposals.codidact.com/help/proposals).
+
 Additionally, we have a $ChatLink available for more free-form discussion.
 
 While we appreciate your continued contributions within the scope of this site, we do ask that you make sure that the topic of your posts remain in scope.
\ No newline at end of file
diff --git a/docker-compose.yml b/docker-compose.yml
index 201795be7e0718c573126e20aca666cd5d7a89e2..556c4b8185ff6d90dae9e5b213d748d7919772b2 100644
--- a/docker-compose.yml
+++ b/docker-compose.yml
@@ -1,6 +1,6 @@
-version: "3.8"
 services:
   db:
+    restart: on-failure:3
     build:
       context: "."
       dockerfile: docker/Dockerfile.db
@@ -8,20 +8,31 @@ services:
       - ./docker/mysql:/var/lib/mysql
     env_file:
       - ${ENV_FILE_LOCATION}
-    command: mysqld --default-authentication-plugin=mysql_native_password --skip-mysqlx
+    command: mysqld --mysql-native-password=on --skip-mysqlx
     cap_add:
       - SYS_NICE
+    healthcheck:
+      test: mysqladmin ping -h 127.0.0.1 -u $$MYSQL_USER --password=$$MYSQL_PASSWORD
+      start_period: 5s
+      interval: 5s
+      timeout: 5s
+      retries: 12
+
 
   uwsgi:
-    restart: always
+    restart: on-failure:3
     build:
       context: "."
-      dockerfile: docker/Dockerfile
+      dockerfile: ${CLIENT_DOCKERFILE}
     depends_on:
-      - db
+      db:
+        condition: service_healthy
+      redis:
+        condition: service_healthy
     environment:
       - COMMUNITY_NAME=${COMMUNITY_NAME}
       - RAILS_ENV=${RAILS_ENV}
+      - MAILER_PROTOCOL=${MAILER_PROTOCOL}
       - CONFIRMABLE_ALLOWED_ACCESS_DAYS=${CONFIRMABLE_ALLOWED_ACCESS_DAYS}
       - LOCAL_DEV_PORT=${LOCAL_DEV_PORT}
     env_file:
@@ -37,7 +48,7 @@ services:
       - db
 
   redis:
-    restart: always
+    restart: on-failure:3
     image: redis:latest
-    depends_on:
-      - db
+    healthcheck:
+      test: ["CMD", "redis-cli","ping"]
diff --git a/docker/Dockerfile b/docker/Dockerfile
index 06fe360ac4de170284d79d3f730e7766f005866c..8944ffb63c993aec69f476667253f2233dc1e501 100644
--- a/docker/Dockerfile
+++ b/docker/Dockerfile
@@ -1,41 +1,53 @@
-FROM ruby:2.7.6
+FROM ruby:3.1.2-bullseye AS ruby
+FROM node:12.18.3-slim AS node
 
-# docker build -f docker/Dockerfile -t qpixel_uwsgi .
+FROM ruby AS build
 
+# Set all encoding to UTF-8
 ENV RUBYOPT="-KU -E utf-8:utf-8"
+
+# Install additional dependencies not present in the base image
 RUN apt-get update && \
-    apt-get install -y gcc && \
-    apt-get install -y make && \
-    apt-get install -y \
-        default-libmysqlclient-dev \
-        autoconf \ 
-        bison \
-        build-essential \
-        libssl-dev \
-        libyaml-dev \
-        libreadline-dev \
-        zlib1g-dev \
-        libncurses5-dev \
-        libffi-dev \
-        libgdbm-dev && \
-   apt-get install -y default-mysql-server
-
-# Install nodejs and imagemagick
-WORKDIR /opt
-RUN wget https://nodejs.org/dist/v12.18.3/node-v12.18.3-linux-x64.tar.xz && \
-    tar xf node-v12.18.3-linux-x64.tar.xz && \
-    wget https://imagemagick.org/archive/binaries/magick && \
-    chmod +x magick && \
-    mv magick /usr/local/bin/magick
+    apt-get install -y bison \
+    build-essential \
+    libxslt-dev \
+    default-mysql-server
+
+# Add core code to container
+WORKDIR /code
+COPY . /code
+
+RUN gem install bundler:2.4.13
+RUN bundle install
 
+# cherry pick only what we really need to run Node.js
+COPY --from=node /usr/local/bin/node /usr/local/bin
+COPY --from=node /usr/local/bin/nodejs /usr/local/bin
+COPY --from=node /usr/local/bin/npm /usr/local/bin
+COPY --from=node /usr/local/bin/npx /usr/local/bin
+COPY --from=node /usr/local/bin/yarn /usr/local/bin
+COPY --from=node /usr/local/bin/yarnpkg /usr/local/bin
+COPY --from=node /usr/local/include/node /usr/local/include
+COPY --from=node /usr/local/lib/node_modules /usr/local/lib
+COPY --from=node /usr/local/share/doc/node /usr/local/share/doc
+COPY --from=node /usr/local/share/man/man1/node.1 /usr/local/share/man/man1
+COPY --from=node /usr/local/share/systemtap/tapset/node.stp /usr/local/share/systemtap/tapset
+COPY --from=node /opt/yarn-v1.22.4 /opt/yarn-v1.22.4
+
+FROM build
+
+# setup a dedicated user for Node.js
+RUN groupadd --gid 1000 node 
+RUN useradd --uid 1000 \
+            --gid node \
+            --shell /bin/bash \
+            --create-home node
+
+# setup Node.js environment
 ENV NODEJS_HOME=/opt/node-v12.18.3-linux-x64/bin
 ENV PATH=$NODEJS_HOME:$PATH
 
-# Add core code to container
 WORKDIR /code
-COPY . /code
-RUN gem install bundler && \
-    bundle install
 
 EXPOSE 80 443 3000
 ENTRYPOINT ["/bin/bash"]
diff --git a/docker/Dockerfile.arm b/docker/Dockerfile.arm
new file mode 100644
index 0000000000000000000000000000000000000000..b6790b416c55660b8b580f2ad73286b29d8b9821
--- /dev/null
+++ b/docker/Dockerfile.arm
@@ -0,0 +1,55 @@
+FROM ruby:3.1.2-bullseye AS ruby
+FROM node:12.18.3-slim AS node
+
+FROM ruby AS build
+
+# Set all encoding to UTF-8
+ENV RUBYOPT="-KU -E utf-8:utf-8"
+
+# Install additional dependencies not present in the base image
+RUN apt-get update && \
+    apt-get install -y bison \
+    build-essential \
+    libxslt-dev \
+    default-mysql-server
+
+# Add core code to container
+WORKDIR /code
+COPY . /code
+
+RUN gem install bundler:2.4.13
+RUN bundle install
+
+# cherry pick only what we really need to run Node.js
+COPY --from=node /usr/local/bin/node /usr/local/bin
+COPY --from=node /usr/local/bin/nodejs /usr/local/bin
+COPY --from=node /usr/local/bin/npm /usr/local/bin
+COPY --from=node /usr/local/bin/npx /usr/local/bin
+COPY --from=node /usr/local/bin/yarn /usr/local/bin
+COPY --from=node /usr/local/bin/yarnpkg /usr/local/bin
+COPY --from=node /usr/local/include/node /usr/local/include
+COPY --from=node /usr/local/lib/node_modules /usr/local/lib
+COPY --from=node /usr/local/share/doc/node /usr/local/share/doc
+COPY --from=node /usr/local/share/man/man1/node.1 /usr/local/share/man/man1
+COPY --from=node /usr/local/share/systemtap/tapset/node.stp /usr/local/share/systemtap/tapset
+COPY --from=node /opt/yarn-v1.22.4 /opt/yarn-v1.22.4
+
+FROM build
+
+# setup a dedicated user for Node.js
+RUN groupadd --gid 1000 node 
+RUN useradd --uid 1000 \
+            --gid node \
+            --shell /bin/bash \
+            --create-home node
+
+# setup Node.js environment
+ENV NODEJS_HOME=/usr/local/bin/node
+ENV PATH=$NODEJS_HOME:$PATH
+
+WORKDIR /code
+
+EXPOSE 80 443 3000
+ENTRYPOINT ["/bin/bash"]
+CMD ["/code/docker/entrypoint.sh"]
+
diff --git a/docker/Dockerfile.db b/docker/Dockerfile.db
index bf43b19dff3f7b454384f220c18a1f4ff75b5358..41875ec298a1bc8df1a15fe84c3e1178c577b904 100644
--- a/docker/Dockerfile.db
+++ b/docker/Dockerfile.db
@@ -1,4 +1,4 @@
-FROM mysql
+FROM mysql:8.4.2
 
 # docker build -t qpixel_db -f docker/Dockerfile.db .
 
diff --git a/docker/Dockerfile.dev b/docker/Dockerfile.dev
new file mode 100644
index 0000000000000000000000000000000000000000..994452878a0b89f93b1a86e917a4da1fc9e577a7
--- /dev/null
+++ b/docker/Dockerfile.dev
@@ -0,0 +1,60 @@
+FROM ruby:3.1.2-bullseye AS ruby
+FROM node:12.18.3-slim AS node
+
+FROM ruby AS build
+
+# Set all encoding to UTF-8
+ENV RUBYOPT="-KU -E utf-8:utf-8"
+
+# Install additional dependencies not present in the base image
+RUN apt-get update && \
+    apt-get install -y bison \
+    build-essential \
+    libxslt-dev \
+    default-mysql-server
+
+# Add core code to container
+WORKDIR /code
+COPY . /code
+
+RUN gem install bundler:2.4.13
+RUN bundle install
+
+# cherry pick only what we really need to run Node.js
+COPY --from=node /usr/local/bin/node /usr/local/bin
+COPY --from=node /usr/local/bin/nodejs /usr/local/bin
+COPY --from=node /usr/local/bin/npm /usr/local/bin
+COPY --from=node /usr/local/bin/npx /usr/local/bin
+COPY --from=node /usr/local/bin/yarn /usr/local/bin
+COPY --from=node /usr/local/bin/yarnpkg /usr/local/bin
+COPY --from=node /usr/local/include/node /usr/local/include
+COPY --from=node /usr/local/lib/node_modules /usr/local/lib
+COPY --from=node /usr/local/share/doc/node /usr/local/share/doc
+COPY --from=node /usr/local/share/man/man1/node.1 /usr/local/share/man/man1
+COPY --from=node /usr/local/share/systemtap/tapset/node.stp /usr/local/share/systemtap/tapset
+COPY --from=node /opt/yarn-v1.22.4 /opt/yarn-v1.22.4
+
+FROM build
+
+# setup a dedicated user for Node.js
+RUN groupadd --gid 1000 node 
+RUN useradd --uid 1000 \
+            --gid node \
+            --shell /bin/bash \
+            --create-home node
+
+# setup Node.js environment
+ENV NODEJS_HOME=/usr/local/bin/node
+ENV PATH=$NODEJS_HOME:$PATH
+
+WORKDIR /code
+
+EXPOSE 80 443 3000
+
+RUN ls -1 /
+RUN [ ! -f "/db-created" ] && echo 1 || echo 2
+
+RUN /code/docker/entrypoint.sh dev
+
+# ensures continued running of the container
+CMD sleep 7d
diff --git a/docker/README.md b/docker/README.md
index f7e1a33ba13d1b7a55d9a54e912b42eca67482f5..18e1af18c7293fffcf5a571c4b33d73397a27b92 100644
--- a/docker/README.md
+++ b/docker/README.md
@@ -1,13 +1,22 @@
 # Docker Installation
 
-A [docker-compose.yml](../docker-compose.yml) file is provided for deployment with Docker compose, if you choose.
+A [docker-compose.yml](../docker-compose.yml) file is provided for deployment with Docker Compose. If you want to use a custom one, simply override the `COMPOSE_FILE` predefined environment variable in the `.env` file in project root ([local-setup.sh](./local-setup.sh) script creates one automatically from [compose-env](./compose-env). By default, `COMPOSE_FILE` is set to [docker-compose.yml](../docker-compose.yml)).
 
-To use docker compose, you need to install the docker-compose-plugin. For a system like debian or ubuntu, you can use the following command.
+To use Docker Compose, you need to install the docker-compose-plugin. 
+You can check if it's already installed via the following command:
+
+```bash
+sudo docker compose version
+```
+
+If the version is 2.x or higher, you are all set. If not, you should install the plugin. On a Debian-based Linux distro, you can run the following:
 
 ```bash
 sudo apt-get install docker-compose-plugin
 ```
 
+For Mac OS, you can install Docker Desktop by downloading it from the Docker website. After starting the application, the docker compose command becomes available in your terminal.
+
 Depending on your setup, you may need to prefix every docker command with sudo.
 
 ## 1. Setup and Secrets
@@ -24,7 +33,14 @@ chmod +x docker/local-setup.sh
 docker/local-setup.sh
 ```
 
-Editing the `./.env` file will modify the corresponding variables used in the docker-compose.yml file but **NOT** the environment variables in the container. Editing the `./docker/env` file will change environment variables only in the running container.
+Editing the `./.env` file will modify the corresponding variables used in the docker-compose.yml file but **NOT** the environment variables in the container. 
+Editing the `./docker/env` file will change environment variables only in the running container.
+
+### Custom build config
+
+Our Docker setup supports custom build configurations for the uwsgi contianer via the `CLIENT_DOCKERFILE` environment variable (see [compose-env](/docker/compose-env)). The default is `docker/Dockerfile`, which points to a preconfigured [production-like setup](/docker/Dockerfile). For developers who need more control over their setup, we also provide a [configuration](/docker/Dockerfile.dev) that is tailored for local development.
+
+To use a custom build config, change the `CLIENT_DOCKERFILE` variable in the .env file that is automatically created by [local-setup.sh](/docker/local-setup.sh) in the project root.
 
 ## 2. Database File
 Ensure `config/database.yml` has the username and password as defined in [docker/env](docker/env) file. The `config/database.yml` should already be gitignored.
@@ -49,30 +65,30 @@ docker compose build db
 docker compose build redis
 ```
 
+NOTE: If you get an error like "Cannot connect to the Docker daemon at ...", you need to ensure you start docker. Depending on your system, this can be done with `sudo service docker start` (Ubuntu) or by opening the Docker Desktop application and waiting for it to start (Mac OS).
+
 ## 4. Start Containers
 
-Then start your containers! 
+Then start your containers:
 
 ```bash
-docker compose up # append -d if you want to detach the processes, although it can be useful to see output into the terminal
-Creating qpixel_redis_1 ... done
-Creating qpixel_db_1    ... done
-Creating qpixel_uwsgi_1 ... done
+docker compose up # append -d (--detach) if you don't want to see output in the terminal
 ```
 
-The uwsgi container has a sleep command for 15 seconds to give the database a chance to start,
-so don't expect to see output right away. After about 20 seconds, check to make sure the server is running (and verify port 3000, note that you can change this mapping in the `.env` file)
+After about 20 seconds, check to make sure the server is running (and verify port 3000, note that you can change this mapping in the `.env` file)
 
 ```
-uwsgi_1  | => Booting Puma
-uwsgi_1  | => Rails 5.2.4.3 application starting in development 
-uwsgi_1  | => Run `rails server -h` for more startup options
-uwsgi_1  | Puma starting in single mode...
-uwsgi_1  | * Version 3.12.6 (ruby 2.6.5-p114), codename: Llamas in Pajamas
-uwsgi_1  | * Min threads: 0, max threads: 16
-uwsgi_1  | * Environment: development
-uwsgi_1  | * Listening on tcp://localhost:3000
-uwsgi_1  | Use Ctrl-C to stop
+qpixel_uwsgi_1  | => Booting Puma
+qpixel_uwsgi_1  | => Rails 7.0.4 application starting in development 
+qpixel_uwsgi_1  | => Run `rails server -h` for more startup options
+qpixel_uwsgi_1  | Puma starting in single mode...
+qpixel_uwsgi_1  | * Puma version: 5.6.5 (ruby 2.7.6-p219) ("Birdie's Version")
+qpixel_uwsgi_1  | * Min threads: 5
+qpixel_uwsgi_1  | * Max threads: 5
+qpixel_uwsgi_1  | * Environment: development
+qpixel_uwsgi_1  | *         PID: 49
+qpixel_uwsgi_1  | * Listening on http://0.0.0.0:3000
+qpixel_uwsgi_1  | Use Ctrl-C to stop
 ```
 
 You should then be able to open your browser to [http://localhost:3000](http://localhost:3000)
@@ -82,6 +98,22 @@ and see the interface.
 
 You can then click "Sign in" to login with what you defined for `$COMMUNITY_ADMIN_EMAIL` and `$COMMUNITY_ADMIN_PASSWORD`. Importantly, your password must be 6 characters or longer, otherwise the user won't be created.
 
+### Custom build configs
+
+If you are using a custom build config that doesn't automatically start Rails or our [config for local development](/docker/Dockerfile.dev), you will also have to manually start the server.
+
+After the containers have started, connect to the uwsgi container (if you are using Docker Desktop, you can connect directly from the application):
+
+```bash
+docker exec -it qpixel-uwsgi-1 bash
+```
+
+And run the following command to start Rails (starting the server is intentionally disabled to allow for live debugging):
+
+```bash
+rails server -b 0.0.0.0
+```
+
 ## 5. Login
 
 Once you are logged in, you should see your icon in the top right:
diff --git a/docker/compose-env b/docker/compose-env
index 162e3169b4c5745f535533d9b190a9640fe25928..5374e1ad40b5189acfef1335247ebde51cd082d5 100644
--- a/docker/compose-env
+++ b/docker/compose-env
@@ -1,5 +1,8 @@
+COMPOSE_FILE=./docker-compose.yml
 LOCAL_DEV_PORT=3000
 COMMUNITY_NAME=Dev Community
+MAILER_PROTOCOL=https
 RAILS_ENV=development
 CONFIRMABLE_ALLOWED_ACCESS_DAYS=2
 ENV_FILE_LOCATION=./docker/env
+CLIENT_DOCKERFILE=docker/Dockerfile
diff --git a/docker/entrypoint.sh b/docker/entrypoint.sh
old mode 100644
new mode 100755
index 145b478c6637feee46303b694e7817bf209b98bf..b6fbc730eacf987e9ded4257b4e0d912a06dc51a
--- a/docker/entrypoint.sh
+++ b/docker/entrypoint.sh
@@ -1,8 +1,5 @@
 #!/bin/bash
 
-# Give database chance to finish creation
-sleep 15
-
 # If not created yet
 if [ ! -f "/db-created" ]; then
     rails db:create
@@ -18,5 +15,7 @@ fi
 # If this isn't done again, there is a 500 error on the first page about posts
 rails db:seed
 
-# defaults to port 3000
-rails server -b 0.0.0.0
+# we don't start the server immediately in dev mode
+if [[ "$1" != 'dev' ]]; then
+    rails server -b 0.0.0.0
+fi
diff --git a/docker/local-setup.sh b/docker/local-setup.sh
index 0b2674eebe3e9b5b8f94ef6e534518244f7dd4f0..9426b7cbc282ca73a2d0a50e6e5bde0191494076 100755
--- a/docker/local-setup.sh
+++ b/docker/local-setup.sh
@@ -3,3 +3,5 @@
 cp ./docker/dummy.env ./docker/env
 cp ./docker/compose-env .env
 cp config/database.docker.yml config/database.yml
+cp config/storage.docker.yml config/storage.yml
+cp ./.sample.irbrc ./.irbrc
diff --git a/docker/mysql-init.sql b/docker/mysql-init.sql
index 2996693ece05af97f3bb5ead90cef2fcb418e88c..510059bb9d7d8ed66b594cb6935b7755c914f17a 100644
--- a/docker/mysql-init.sql
+++ b/docker/mysql-init.sql
@@ -1,8 +1,6 @@
 
 /* database qpixel and user are already created *
    if you change your environment file, you need to update database names here */
-CREATE DATABASE qpixel_dev;
-CREATE DATABASE qpixel_test;
 GRANT ALL ON qpixel_dev.* TO qpixel;
 GRANT ALL ON qpixel_test.* TO qpixel;
 GRANT ALL ON qpixel.* TO qpixel;
diff --git a/img/edit-category.png b/img/edit-category.png
new file mode 100644
index 0000000000000000000000000000000000000000..fefd5c760eb4207a09bac93fa48aa89d2279b05c
Binary files /dev/null and b/img/edit-category.png differ
diff --git a/img/meta-tags.png b/img/meta-tags.png
new file mode 100644
index 0000000000000000000000000000000000000000..0de519f729babc281ece1c808da991b9aebf26ac
Binary files /dev/null and b/img/meta-tags.png differ
diff --git a/img/qpixel-dev.png b/img/qpixel-dev.png
new file mode 100644
index 0000000000000000000000000000000000000000..242818e36e250d9cd0b323a614fadeeb83aaa2cc
Binary files /dev/null and b/img/qpixel-dev.png differ
diff --git a/img/required-tags.png b/img/required-tags.png
new file mode 100644
index 0000000000000000000000000000000000000000..4e8fa4bd71d7945d5622ba50604d28a1e43600d5
Binary files /dev/null and b/img/required-tags.png differ
diff --git a/lib/console_extension.rb b/lib/console_extension.rb
new file mode 100644
index 0000000000000000000000000000000000000000..ce7cd5c0bc60b1f6db0713cf589a7a5aa92f5798
--- /dev/null
+++ b/lib/console_extension.rb
@@ -0,0 +1,106 @@
+module ConsoleExtension
+  extend ActiveSupport::Concern
+
+  included do
+    console do
+      Rails.logger.silence do
+        console_init_community
+      end
+    end
+  end
+
+  def console_init_community
+    community_count = Community.count
+    if community_count.zero?
+      puts "\e[31m(!) You have not yet created any communities.\e[0m"
+      puts 'Create a community by entering:'
+      puts ''
+      puts "  Community.create(name: 'my community name', host: 'my.site.com')"
+      puts '  Rails.cache.clear'
+      puts ''
+
+      if Rails.env.development?
+        begin
+          port = Rails::Server::Options.new.parse!(ARGV)[:Port] || 3000
+        rescue
+          port = 3000
+        end
+        puts "Since you are running in development mode, you probably want to set host to localhost:#{port}"
+        puts ''
+      elsif Rails.env.production?
+        puts 'Since you are running in production mode, set host to your fully qualified domain name without http(s).'
+        puts 'For example, if you host your site at https://meta.codidact.org, set host to meta.codidact.org'
+        puts ''
+      end
+      puts 'For more information, see the set up instructions.'
+    elsif community_count == 1
+      community = Community.first
+      RequestContext.community = community
+      puts "\e[32m(!) Found one community, set current community to #{community.name} @ #{community.host}\e[0m"
+    else
+      community = Community.find_by(host: 'localhost:3000') if Rails.env.development?
+      community ||= Community.first
+      RequestContext.community = community
+      puts "\e[32m(!) Found multiple communities, set current community to #{community.name} @ #{community.host}\e[0m"
+      puts ''
+      puts 'You can change your current community by entering:'
+      puts ''
+      puts '  RequestContext.community = Community.find_by(...)'
+      puts ''
+      puts "You can use `host: 'my.host'` or `name: 'community name'` in place of the dots"
+    end
+    puts ''
+  rescue
+    puts "\e[31m(!) Unable to load communities. Is your database configuration correct?\e[0m"
+  end
+end
+
+# Create module that can be included in the .irbrc:
+#
+#   Qpixel.irb! if defined?(Qpixel)
+module Qpixel
+  def self.irb!
+    IRB::Irb.class_eval do
+      private
+
+      def self.rails_environment
+        case Rails.env
+        when 'development'
+          "\e[32mdev\e[0m"
+        when 'production'
+          "\e[31mprod\e[0m"
+        when 'test'
+          "\e[32mtest\e[0m"
+        when 'staging'
+          "\e[32mstag\e[0m"
+        else
+          "\e[31m#{Rails.env}\e[0m"
+        end
+      end
+
+      def self.qpixel_prompt
+        c = RequestContext.community
+        "[#{rails_environment}] [\e[34m#{c&.name || '-'} @ #{c&.host || '-'}\e[0m]"
+      end
+    end
+
+    IRB::Irb.class_eval do
+      # Define an alternative string dup method which will redetermine the prompt part if community changes
+      qpixel_block = proc do |s|
+        def s.dup
+          IRB::Irb.qpixel_prompt + self
+        end
+      end
+
+      IRB.conf[:PROMPT][:QPIXEL] = {
+        PROMPT_I: ':%03n> '.tap(&qpixel_block),
+        PROMPT_N: ':%03n> '.tap(&qpixel_block),
+        PROMPT_S: ':%03n%l '.tap(&qpixel_block),
+        PROMPT_C: ':%03n* '.tap(&qpixel_block),
+        RETURN: IRB.conf[:PROMPT][:DEFAULT][:RETURN]
+      }
+
+      IRB.conf[:PROMPT_MODE] = :QPIXEL
+    end
+  end
+end
diff --git a/lib/namespaced_env_cache.rb b/lib/namespaced_env_cache.rb
index af5130e70f6fae819ba0450ea592005b341ac455..093c9a90aa27e91538770756657eecc8985565ff 100644
--- a/lib/namespaced_env_cache.rb
+++ b/lib/namespaced_env_cache.rb
@@ -5,10 +5,16 @@ module QPixel
       @getters = {}
     end
 
+    def include_community(opts)
+      include = opts.delete(:include_community)
+      include.nil? ? true : include
+    end
+
     # These methods need the cache key name updating before we pass it to the underlying cache.
     [:decrement, :delete, :exist?, :fetch, :increment, :read, :write, :delete_matched].each do |method|
       define_method method do |name, *args, **opts, &block|
-        @underlying.send(method, construct_ns_key(name, include_community: opts.delete(:include_community) || true),
+        include_community = include_community(opts)
+        @underlying.send(method, construct_ns_key(name, include_community: include_community),
                          *args, **opts, &block)
       end
     end
@@ -16,7 +22,8 @@ module QPixel
     # These methods need a hash of cache keys updating before we pass it to the underlying cache.
     [:write_multi].each do |method|
       define_method method do |hash, *args, **opts, &block|
-        hash = hash.map { |k, v| [construct_ns_key(k), v] }.to_h
+        include_community = include_community(opts)
+        hash = hash.map { |k, v| [construct_ns_key(k, include_community: include_community), v] }.to_h
         @underlying.send(method, hash, *args, **opts, &block)
       end
     end
@@ -29,14 +36,16 @@ module QPixel
     end
 
     def read_multi(*keys, **opts)
-      keys = keys.map { |k| [construct_ns_key(k), k] }.to_h
-      results = @underlying.read_multi *keys.keys, **opts
+      include_community = include_community(opts)
+      keys = keys.map { |k| [construct_ns_key(k, include_community: include_community), k] }.to_h
+      results = @underlying.read_multi(*keys.keys, **opts)
       results.map { |k, v| [keys[k], v] }.to_h
     end
 
     def fetch_multi(*keys, **opts, &block)
-      keys = keys.map { |k| construct_ns_key(k) }
-      @underlying.fetch_multi *keys, **opts, &block
+      include_community = include_community(opts)
+      keys = keys.map { |k| construct_ns_key(k, include_community: include_community) }
+      @underlying.fetch_multi(*keys, **opts, &block)
     end
 
     def persistent(name, **opts, &block)
diff --git a/public/assets/scoring_table.png b/public/assets/scoring_table.png
index 6f818f6ce18a0e8c947ff4c3b771e2adc44e479c..42c001f31b6b0856589a55f2ba6149ea03cc0cce 100644
Binary files a/public/assets/scoring_table.png and b/public/assets/scoring_table.png differ
diff --git a/public/assets/scoring_table.svg b/public/assets/scoring_table.svg
new file mode 100644
index 0000000000000000000000000000000000000000..48835ef0fcaa05613f3c52794e52e40281d9ab49
--- /dev/null
+++ b/public/assets/scoring_table.svg
@@ -0,0 +1 @@
+<svg width="2128" height="629" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" xml:space="preserve" overflow="hidden"><defs><clipPath id="clip0"><rect x="536" y="586" width="2128" height="629"/></clipPath><image width="2128" height="629" xlink:href="" preserveAspectRatio="none" id="img1"></image></defs><g clip-path="url(#clip0)" transform="translate(-536 -586)"><use width="100%" height="100%" xlink:href="#img1" fill="none" transform="translate(536 586)"></use></g></svg>
\ No newline at end of file
diff --git a/scripts/cleanup_drafts.rb b/scripts/cleanup_drafts.rb
new file mode 100644
index 0000000000000000000000000000000000000000..27ef4eb63c210d5abc598b61c8552041b4aeece1
--- /dev/null
+++ b/scripts/cleanup_drafts.rb
@@ -0,0 +1,5 @@
+redis = RequestContext.redis
+
+redis.scan_each(:match => "saved_post.*.*.tags") do |key| 
+  redis.srem?(key, '')
+end
\ No newline at end of file
diff --git a/scripts/cleanup_votes.rb b/scripts/cleanup_votes.rb
new file mode 100644
index 0000000000000000000000000000000000000000..67707824ee676c7f630b43d2faf5f872e328c11e
--- /dev/null
+++ b/scripts/cleanup_votes.rb
@@ -0,0 +1 @@
+CleanupVotesJob.perform_later
\ No newline at end of file
diff --git a/scripts/create_backup_2fa_codes.rb b/scripts/create_backup_2fa_codes.rb
new file mode 100644
index 0000000000000000000000000000000000000000..ddeac6793a2a1c7320705f02b9c168966d82401b
--- /dev/null
+++ b/scripts/create_backup_2fa_codes.rb
@@ -0,0 +1,4 @@
+User.where(enabled_2fa: true).each do |user|
+  user.update(backup_2fa_code: SecureRandom.alphanumeric(24))
+  TwoFactorMailer.with(user: user, host: 'meta.codidact.com').backup_code.deliver_now
+end
diff --git a/scripts/prune_email_logs.rb b/scripts/prune_email_logs.rb
new file mode 100644
index 0000000000000000000000000000000000000000..0336c9e0cdcaea503f488cd9deeb6f5cf4182c1c
--- /dev/null
+++ b/scripts/prune_email_logs.rb
@@ -0,0 +1 @@
+EmailLog.where('created_at < DATE_SUB(CURRENT_TIMESTAMP, INTERVAL 1 MONTH)').destroy_all
diff --git a/scripts/recalc_abilities.rb b/scripts/recalc_abilities.rb
index eeda78688153578d971cf87228c942c7ab37d06f..17c99d4fd56e15a52cded50f4d7ac8f36d95067d 100644
--- a/scripts/recalc_abilities.rb
+++ b/scripts/recalc_abilities.rb
@@ -36,11 +36,11 @@ all.each do |q|
       puts "Scope: CommunityUser : #{cu.id}"
     end
 
-    cu.recalc_privileges
+    cu.recalc_abilities
 
     # Grant mod ability if mod status is given
-    if (cu.is_moderator || cu.is_admin || u.is_global_moderator || u.is_global_admin) && !cu.privilege?('mod')
-      cu.grant_privilege('mod')
+    if (cu.is_moderator || cu.is_admin || u.is_global_moderator || u.is_global_admin) && !cu.ability?('mod')
+      cu.grant_ability!('mod')
     end
 
     resolved << q.id
diff --git a/scripts/recalc_abilities_upon_first_migration.rb b/scripts/recalc_abilities_upon_first_migration.rb
index 520144e650cf80d395ae4b9c73e97524381f1d19..32b3c48e4116647c528fab813741afe93f910948 100644
--- a/scripts/recalc_abilities_upon_first_migration.rb
+++ b/scripts/recalc_abilities_upon_first_migration.rb
@@ -9,7 +9,7 @@ User.unscoped.all.map do |u|
     cu.recalc_privileges
 
     if (cu.is_moderator || cu.is_admin || u.is_global_moderator || u.is_global_admin) && !cu.privilege?('mod')
-      cu.grant_privilege('mod')
+      cu.grant_privilege!('mod')
     end
   rescue
     puts "    !!! Error recalcing for CommunityUser.Id=#{cu.id}"
diff --git a/scripts/run_spam_cleanup.rb b/scripts/run_spam_cleanup.rb
new file mode 100644
index 0000000000000000000000000000000000000000..2b2a6853de882864e4c0f63fd37fb7f1dc006631
--- /dev/null
+++ b/scripts/run_spam_cleanup.rb
@@ -0,0 +1 @@
+CleanUpSpammyUsersJob.perform_later
diff --git a/scripts/wipe_everything.rb b/scripts/wipe_everything.rb
index 98da394d96683964ada80e25092b1f2be9f7b4ed..b5311f9a1d021e8dc4f31fef7ee14b8da93ad26f 100644
--- a/scripts/wipe_everything.rb
+++ b/scripts/wipe_everything.rb
@@ -3,17 +3,21 @@ unless Rails.env.development?
   exit 255
 end
 
-def exec(sql)
+def exec_sql(sql)
   ActiveRecord::Base.connection.execute sql
 end
 
 conn = ActiveRecord::Base.connection
 leave_tables = ['ar_internal_metadata', 'schema_migrations']
+
+exec_sql 'SET FOREIGN_KEY_CHECKS = 0'
 (conn.tables - leave_tables).each do |t|
-  exec "DELETE FROM `#{t}`"
-  exec "ALTER TABLE `#{t}` AUTO_INCREMENT=1"
+  exec_sql "DELETE FROM `#{t}`"
+  exec_sql "ALTER TABLE `#{t}` AUTO_INCREMENT=1"
 end
+exec_sql 'SET FOREIGN_KEY_CHECKS = 1'
 
+Community.create(name: 'Dev Community', host: 'localhost:3000')
 Rails.cache.clear
 
-`bundle exec rails db:seed`
\ No newline at end of file
+`rails db:seed`
diff --git a/test/application_system_test_case.rb b/test/application_system_test_case.rb
new file mode 100644
index 0000000000000000000000000000000000000000..da6f04b0020ef37e3048ae2c01ecef66a3930c89
--- /dev/null
+++ b/test/application_system_test_case.rb
@@ -0,0 +1,103 @@
+require 'test_helper'
+
+# This class serves as the base for all system test cases.
+#
+# The DRIVER environment variable is used to determine the browser that is used. Possible options are:
+# - headless_chrome
+# - chrome
+# - headless_firefox (default)
+# - firefox
+class ApplicationSystemTestCase < ActionDispatch::SystemTestCase
+  DRIVER = if ENV['DRIVER']
+             ENV['DRIVER'].to_sym
+           else
+             :headless_firefox
+           end
+
+  driven_by :selenium, using: DRIVER, screen_size: [1920, 1080]
+
+  setup do
+    Community.first.update(host: root_url.gsub(/https?:\/\//, '').gsub('/', ''))
+  end
+
+  # Logs in as the specified user
+  #
+  # @param user_or_fixture [User, Symbol] either a user or a symbol referring to a user from the fixtures
+  # @param password [String] the password to sign in with
+  def log_in(user_or_fixture, password = 'test123')
+    @user = user(user_or_fixture)
+    visit new_user_session_url
+    fill_in 'Email', with: @user.email
+    fill_in 'Password', with: password
+
+    click_button 'Sign in'
+  end
+
+  # Attempts to log out using the buttons in the top menu bar.
+  def log_out
+    within :css, '.header' do
+      find(:css, 'i.far.fa-caret-square-down').find(:xpath, '..').click
+    end
+
+    find_link('Sign Out').click
+  end
+
+  # Pretends the user has clicked the confirmation link in the email they received.
+  #
+  # @param user_or_fixture [User, Symbol] the user or a symbol referring to the user fixture to use
+  def confirm_email(user_or_fixture)
+    u = user(user_or_fixture)
+    visit user_confirmation_url(
+      params: { confirmation_token: u.confirmation_token }
+    )
+  end
+
+  # Translates the given parameter to a proper user.
+  #
+  # @param user_or_fixture [User, Symbol] either a user or a symbol referring to a fixture
+  def user(user_or_fixture)
+    if user_or_fixture.is_a? User
+      user_or_fixture
+    else
+      users(user_or_fixture)
+    end
+  end
+
+  # In the post form, this method will select the given tag.
+  #
+  # @param tag_name [String] the name of the tag
+  # @param create_new [Boolean] whether creating a new tag is allowed (default false)
+  def post_form_select_tag(tag_name, create_new = false)
+    # First enter the tag name into the select2 search field for the tag
+    within find_field('Tags (at least one):').find(:xpath, '..') do
+      find('.select2-search__field').fill_in(with: tag_name)
+    end
+
+    # Get the first item listed that is not the "Searching..." item
+    first_option = find('#select2-post_tags_cache-results li:first-child') { |el| el.text != 'Searching…' }
+
+    if first_option.first('span').text == tag_name
+      # If the text matches the tag name, first check whether we are creating a new tag.
+      # If so, confirm that we are allowed to. If all is good, actually click on the item.
+      if create_new || !first_option.text.include?('Create new tag')
+        first_option.click
+      else
+        raise "Expected to find tag with the name #{tag_name}, " \
+              'but could not select it from options without creating a new tag.'
+      end
+    elsif create_new
+      # The first item returned is not the tag we were looking for (another tag partial match + not existing)
+      # If we are allowed to create a tag, select the last option from the list, which is always the tag creation.
+      last_option = find('#select2-post_tags_cache-results li:last-child')
+      if last_option.first('span').text == tag_name
+        last_option.click
+      else
+        raise "Tried to select tag #{tag_name} for creation, but it does not seem to be a presented option."
+      end
+    else
+      # The first item returned is not the tag we were looking for, and we are not allowed to create a tag.
+      raise "Expected to find tag with the name #{tag_name}, " \
+            'but could not select it from options without creating a new tag.'
+    end
+  end
+end
diff --git a/test/controllers/donations_controller_test.rb b/test/controllers/donations_controller_test.rb
index 81105d8ed4857e7fba99b3e1fb7de15041c18603..c2ed31624b7c91162f9568b0b8aa13da63ffd121 100644
--- a/test/controllers/donations_controller_test.rb
+++ b/test/controllers/donations_controller_test.rb
@@ -9,6 +9,7 @@ class DonationsControllerTest < ActionController::TestCase
   end
 
   test 'should create PaymentIntent' do
+    skip unless Stripe.api_key
     post :intent, params: { currency: 'EUR', amount: '24.99', desc: 'Created from Rails test' }
     assert_response 200
     assert_not_nil assigns(:intent)&.id
diff --git a/test/controllers/email_logs_controller_test.rb b/test/controllers/email_logs_controller_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..9addb0be163bd4e1988f3edc4af259e7cd354f98
--- /dev/null
+++ b/test/controllers/email_logs_controller_test.rb
@@ -0,0 +1,7 @@
+require 'test_helper'
+
+class EmailLogsControllerTest < ActionDispatch::IntegrationTest
+  # test "the truth" do
+  #   assert true
+  # end
+end
diff --git a/test/controllers/posts/drafts_test.rb b/test/controllers/posts/drafts_test.rb
index 21f629cb696fce1467ac59a7313cdeb165fae44e..bc7c9603b792105b9a5d8aaa355806bd177d2805 100644
--- a/test/controllers/posts/drafts_test.rb
+++ b/test/controllers/posts/drafts_test.rb
@@ -5,13 +5,31 @@ class PostsControllerTest < ActionController::TestCase
 
   test 'can save draft' do
     sign_in users(:standard_user)
-    post :save_draft, params: { path: 'test', post: 'test' }
+    post :save_draft, params: {
+      body: 'test_body',
+      comment: 'test_comment',
+      excerpt: 'test_excerpt',
+      license: '4',
+      path: 'test_path',
+      tag_name: 'test_tag_name',
+      tags: ['tag1', 'tag2'],
+      title: 'test_title'
+    }
     assert_response 200
     assert_nothing_raised do
       JSON.parse(response.body)
     end
-    assert_equal "saved_post.#{users(:standard_user).id}.test", JSON.parse(response.body)['key']
-    assert_equal 'test', RequestContext.redis.get(JSON.parse(response.body)['key'])
+
+    base_key = JSON.parse(response.body)['key']
+
+    assert_equal "saved_post.#{users(:standard_user).id}.test_path", base_key
+    assert_equal 'test_body', RequestContext.redis.get(base_key)
+    assert_equal 'test_comment', RequestContext.redis.get("#{base_key}.comment")
+    assert_equal 'test_excerpt', RequestContext.redis.get("#{base_key}.excerpt")
+    assert_equal 'test_tag_name', RequestContext.redis.get("#{base_key}.tag_name")
+    assert_equal '4', RequestContext.redis.get("#{base_key}.license")
+    assert_empty ['tag1', 'tag2'].difference(RequestContext.redis.smembers("#{base_key}.tags"))
+    assert_equal 'test_title', RequestContext.redis.get("#{base_key}.title")
   end
 
   test 'can delete draft' do
diff --git a/test/controllers/posts/help_test.rb b/test/controllers/posts/help_test.rb
index 48ba059abe5be0614c0ba8fb0d523f72db5b8802..db3e59f4d10b93e0feacff8b03dd44a0da140c43 100644
--- a/test/controllers/posts/help_test.rb
+++ b/test/controllers/posts/help_test.rb
@@ -25,20 +25,20 @@ class PostsControllerTest < ActionController::TestCase
   test 'moderator help requires authentication' do
     get :document, params: { slug: posts(:mod_help_article).doc_slug }
     assert_response 404
-    assert_not_nil assigns(:post)
+    assert_nil assigns(:post)
   end
 
   test 'regular user cannot get mod help' do
     sign_in users(:standard_user)
     get :document, params: { slug: posts(:mod_help_article).doc_slug }
     assert_response 404
-    assert_not_nil assigns(:post)
+    assert_nil assigns(:post)
   end
 
   test 'cannot get disabled help article' do
     sign_in users(:moderator)
     get :document, params: { slug: posts(:disabled_help_article).doc_slug }
     assert_response 404
-    assert_not_nil assigns(:post)
+    assert_nil assigns(:post)
   end
 end
diff --git a/test/controllers/posts/lock_test.rb b/test/controllers/posts/lock_test.rb
index 95a30470b1ce526b0803a3c767aa4269a9c6e5a7..cfb7b2ddc0b71518fb698ac74960de5af281df3f 100644
--- a/test/controllers/posts/lock_test.rb
+++ b/test/controllers/posts/lock_test.rb
@@ -79,4 +79,15 @@ class PostsControllerTest < ActionController::TestCase
     end
     assert_equal 'success', JSON.parse(response.body)['status']
   end
+
+  test 'Locks on posts expire' do
+    sign_in users(:moderator)
+    post :lock, params: { id: posts(:question_one).id, length: 1, format: :json }
+    assert_response 200
+
+    # Change the locked_until to have already passed
+    assigns(:post).update(locked_until: 1.second.ago)
+
+    assert_not assigns(:post).locked?
+  end
 end
diff --git a/test/controllers/search_controller_test.rb b/test/controllers/search_controller_test.rb
index 00792a96e546902667c800efb9a5033fd8f16635..9beba194c8c7a4073a1d6f411adfdf3fd75bf7ef 100644
--- a/test/controllers/search_controller_test.rb
+++ b/test/controllers/search_controller_test.rb
@@ -3,10 +3,10 @@ require 'test_helper'
 class SearchControllerTest < ActionController::TestCase
   include Devise::Test::ControllerHelpers
 
-  test 'get without a search term should result in nil' do
+  test 'get without a search term should result in all posts' do
     get :search
     assert_response 200
-    assert_nil assigns(:posts)
+    assert_not_nil assigns(:posts)
   end
 
   test 'get with a search term should have results' do
diff --git a/test/controllers/tags_controller_test.rb b/test/controllers/tags_controller_test.rb
index 176beff95889f389e0f5e9c6d670ad2553566e64..cdb3983d91c65a191c88be491d0b39fcebeedabf 100644
--- a/test/controllers/tags_controller_test.rb
+++ b/test/controllers/tags_controller_test.rb
@@ -12,7 +12,7 @@ class TagsControllerTest < ActionController::TestCase
     assert_not_nil assigns(:tags)
   end
 
-  test 'index with search params should return tags starting with search' do
+  test 'index with search params should return tags including search term' do
     get :index, params: { format: 'json', term: 'dis' }
     assert_response 200
     assert_nothing_raised do
@@ -20,7 +20,19 @@ class TagsControllerTest < ActionController::TestCase
     end
     assert_not_nil assigns(:tags)
     JSON.parse(response.body).each do |tag|
-      assert_equal true, tag['name'].start_with?('dis')
+      assert_equal true, tag['name'].include?('dis') || tag['tag_synonyms'].any? { |ts| ts['name'].include?('syn') }
+    end
+  end
+
+  test 'index with search params should return tags whose synonyms include search term' do
+    get :index, params: { format: 'json', term: 'syn' }
+    assert_response 200
+    assert_nothing_raised do
+      JSON.parse(response.body)
+    end
+    assert_not_nil assigns(:tags)
+    JSON.parse(response.body).each do |tag|
+      assert_equal true, tag['name'].include?('syn') || tag['tag_synonyms'].any? { |ts| ts['name'].include?('syn') }
     end
   end
 
@@ -125,6 +137,26 @@ class TagsControllerTest < ActionController::TestCase
     assert_equal 'things', assigns(:tag).excerpt
   end
 
+  test 'should update tag with synonym addition' do
+    sign_in users(:deleter)
+    patch :update, params: { id: categories(:main).id, tag_id: tags(:topic).id,
+                             tag: { tag_synonyms_attributes: { '1': { name: 'conversation' } } } }
+    assert_response 302
+    assert_redirected_to tag_path(id: categories(:main).id, tag_id: tags(:topic).id)
+    assert_not_nil assigns(:tag)
+    assert_equal 'conversation', assigns(:tag).tag_synonyms.first&.name
+  end
+
+  test 'should update tag with synonym removal' do
+    sign_in users(:deleter)
+    patch :update, params: { id: categories(:main).id, tag_id: tags(:base).id,
+                             tag: { tag_synonyms_attributes: { '1': { id: tag_synonyms(:base_synonym).id, _destroy: 'true' } } } }
+    assert_response 302
+    assert_redirected_to tag_path(id: categories(:main).id, tag_id: tags(:base).id)
+    assert_not_nil assigns(:tag)
+    assert_equal true, (assigns(:tag).tag_synonyms.none? { |ts| ts.name == 'synonym' })
+  end
+
   test 'should prevent a tag being its own parent' do
     sign_in users(:deleter)
     patch :update, params: { id: categories(:main).id, tag_id: tags(:topic).id,
diff --git a/test/controllers/users/sessions_controller_test.rb b/test/controllers/users/sessions_controller_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..b06e1b602c05eceef740f5b1cca6b82f4edf4e1b
--- /dev/null
+++ b/test/controllers/users/sessions_controller_test.rb
@@ -0,0 +1,17 @@
+require 'test_helper'
+
+class Users::SessionsControllerTest < ActionController::TestCase
+  include Devise::Test::ControllerHelpers
+  include ApplicationHelper
+
+  test 'should sign in with 2fa backup code' do
+    @request.env['devise.mapping'] = Devise.mappings[:user]
+    Users::SessionsController.first_factor << users(:enabled_2fa).id
+    post :verify_code, params: { uid: users(:enabled_2fa).id, code: 'M8lENyehyCvo9F9MbyTl1aOL' }
+    assert_response 302
+    assert_not_nil flash[:warning]
+    assert_not_nil current_user
+    assert_nil current_user.backup_2fa_code
+    assert_not current_user.enabled_2fa
+  end
+end
diff --git a/test/controllers/users_controller_test.rb b/test/controllers/users_controller_test.rb
index 17a18c257f006a6440b8bef65b7ceafe03255ad5..0aa5d08633d57e9b7556807f9b9cea9638159184 100644
--- a/test/controllers/users_controller_test.rb
+++ b/test/controllers/users_controller_test.rb
@@ -113,16 +113,41 @@ class UsersControllerTest < ActionController::TestCase
     assert_response 200
   end
 
-  test 'should update profile text' do
+  test 'should redirect & show success notice on profile update' do
     sign_in users(:standard_user)
-    patch :update_profile, params: { user: { profile_markdown: 'ABCDEF GHIJKL', website: 'https://example.com/user',
-                                             twitter: '@standard_user' } }
+    patch :update_profile, params: { user: { username: 'std' } }
     assert_response 302
     assert_not_nil flash[:success]
     assert_not_nil assigns(:user)
     assert_equal users(:standard_user).id, assigns(:user).id
-    assert_not_nil assigns(:user).profile
-    assert_equal 'standard_user', assigns(:user).twitter
+  end
+
+  test 'should update profile text' do
+    sign_in users(:standard_user)
+    patch :update_profile, params: {
+      user: { profile_markdown: 'ABCDEF GHIJKL' }
+    }
+    assert_equal assigns(:user).profile.strip, '<p>ABCDEF GHIJKL</p>'
+  end
+
+  test 'should update websites' do
+    sign_in users(:standard_user)
+    patch :update_profile, params: {
+      user: { user_websites_attributes: {
+        '0': { label: 'web', url: 'example.com' }
+      } }
+    }
+    assert_not_nil assigns(:user).user_websites
+    assert_equal 'web', assigns(:user).user_websites.first.label
+    assert_equal 'example.com', assigns(:user).user_websites.first.url
+  end
+
+  test 'should update user discord link' do
+    sign_in users(:standard_user)
+    patch :update_profile, params: {
+      user: { discord: 'example_user#1234' }
+    }
+    assert_equal 'example_user#1234', assigns(:user).discord
   end
 
   test 'should get full posts list for a user' do
diff --git a/test/fixtures/category_filter_defaults.yml b/test/fixtures/category_filter_defaults.yml
new file mode 100644
index 0000000000000000000000000000000000000000..697b14ee7c7510bbb4a9103c3419463b70d9a4b0
--- /dev/null
+++ b/test/fixtures/category_filter_defaults.yml
@@ -0,0 +1,11 @@
+# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
+
+main_default_filter:
+  user: standard_user
+  filter: one
+  category: main
+
+meta_default_filter:
+  user: standard_user
+  filter: one
+  category: meta
diff --git a/test/fixtures/community_users.yml b/test/fixtures/community_users.yml
index 31d28a04fa272cf2274e1b41de05fa721c2ab043..4d7ac69639e34698622803a8ad1409eb1a08d0ef 100644
--- a/test/fixtures/community_users.yml
+++ b/test/fixtures/community_users.yml
@@ -1,3 +1,10 @@
+sample_basic_user:
+  user: basic_user
+  community: sample
+  is_admin: false
+  is_moderator: false
+  reputation: 1
+
 sample_standard_user:
   user: standard_user
   community: sample
diff --git a/test/fixtures/email_logs.yml b/test/fixtures/email_logs.yml
new file mode 100644
index 0000000000000000000000000000000000000000..7ca782a8cb4b8c4b06b52c5fb157df93ee087daa
--- /dev/null
+++ b/test/fixtures/email_logs.yml
@@ -0,0 +1,11 @@
+# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
+
+one:
+  log_type: MyString
+  destination: MyString
+  data: MyText
+
+two:
+  log_type: MyString
+  destination: MyString
+  data: MyText
diff --git a/test/fixtures/filters.yml b/test/fixtures/filters.yml
new file mode 100644
index 0000000000000000000000000000000000000000..f47aa1e3150f70f81172ee51e494e6d6291d006f
--- /dev/null
+++ b/test/fixtures/filters.yml
@@ -0,0 +1,19 @@
+# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
+
+one:
+  name: MyFilterOne
+  user: standard_user
+  min_score: 1.5
+  max_score: 1.5
+  min_answers: 1
+  max_answers: 1
+  status: MyString
+
+two:
+  name: MyFilterTwo
+  user: standard_user
+  min_score: 1.5
+  max_score: 1.5
+  min_answers: 1
+  max_answers: 1
+  status: MyString
diff --git a/test/fixtures/posts.yml b/test/fixtures/posts.yml
index bd4c76383cb81d2101989a582feb4979711d253c..c3597fb96896631d447eb8c2c3c8e26b0f75ce34 100644
--- a/test/fixtures/posts.yml
+++ b/test/fixtures/posts.yml
@@ -1,8 +1,8 @@
 question_one:
   post_type: question
-  title: Q1 ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
+  title: Q1 - This is test question number one
+  body: This is the body of test question one. Note that we did not include any markdown or HTML in here.
+  body_markdown: This is the body of test question one. Note that we did not include any markdown or HTML in here.
   tags_cache:
     - discussion
     - support
@@ -21,9 +21,9 @@ question_one:
 
 question_two:
   post_type: question
-  title: Q2 ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  title: Q2 - This is test question number two
+  body: This is the body of test question two. Note that we did not include any markdown or HTML in here.
+  body_markdown: This is the body of test question two. Note that we did not include any markdown or HTML in here.
   tags_cache:
     - discussion
     - support
@@ -42,9 +42,9 @@ question_two:
 
 bad_answers:
   post_type: question
-  title: Q1 ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
+  title: Q3B - This question has bad answers
+  body: Q3B - This question is in the category main, posted by standard_user. It has bad answers.
+  body_markdown: Q3B - This question is in the category main, posted by standard_user. It has bad answers.
   tags_cache:
     - discussion
     - support
@@ -63,9 +63,9 @@ bad_answers:
 
 deleted:
   post_type: question
-  title: Q3D ZY XWVUTS RQPONM LKJIHG FEDCBA
-  body: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  title: Q3D - This question is deleted
+  body: Q3D - This question is in the category main, posted by standard_user. It has been deleted by deleter.
+  body_markdown: Q3D - This question is in the category main, posted by standard_user. It has been deleted by deleter.
   tags_cache:
     - discussion
     - support
@@ -87,9 +87,9 @@ deleted:
 
 deleted_mod:
   post_type: question
-  title: Q3D ZY XWVUTS RQPONM LKJIHG FEDCBA
-  body: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  title: Q3DM - This question is deleted by a moderator
+  body: Q3DM - This question is in category main, posted by standard_user. It has been deleted by moderator.
+  body_markdown: Q3DM - This question is in category main, posted by standard_user. It has been deleted by moderator.
   tags_cache:
     - discussion
     - support
@@ -111,9 +111,9 @@ deleted_mod:
 
 closed:
   post_type: question
-  title: Q4C ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  title: Q4C - This question is closed
+  body: Q4C - This question is in the category main, posted by standard_user. It has been closed by closer.
+  body_markdown: Q4C - This question is in the category main, posted by standard_user. It has been closed by closer.
   tags_cache:
     - discussion
     - support
@@ -135,9 +135,9 @@ closed:
 
 locked:
   post_type: question
-  title: Q4C ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  title: Q4L - This question is locked
+  body: Q4L - This question is in the category main, posted by standard_user. It has been locked by deleter.
+  body_markdown: Q4L - This question is in the category main, posted by standard_user. It has been locked by deleter.
   tags_cache:
     - discussion
     - support
@@ -150,7 +150,6 @@ locked:
   locked: true
   locked_by: deleter
   locked_at: 2019-01-01T00:00:00.000000Z
-  locked_until: 2020-01-01T00:00:00.000000Z
   user: standard_user
   community: sample
   category: main
@@ -160,9 +159,9 @@ locked:
 
 locked_mod:
   post_type: question
-  title: LM ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  title: Q4LM - This question is locked by a moderator
+  body: Q4LM - This question is in the category main, posted by standard_user. It has been locked by moderator.
+  body_markdown: Q4LM - This question is in the category main, posted by standard_user. It has been locked by moderator.
   tags_cache:
     - discussion
     - support
@@ -175,7 +174,6 @@ locked_mod:
   locked: true
   locked_by: moderator
   locked_at: 2019-01-01T00:00:00.000000Z
-  locked_until: 2020-01-01T00:00:00.000000Z
   user: standard_user
   community: sample
   category: main
@@ -185,9 +183,9 @@ locked_mod:
 
 free_edit:
   post_type: free_edit
-  title: FE ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
+  title: FE - This is a post of the type free_edit
+  body: FE - This is a free edit post, in the category main, posted by moderator.
+  body_markdown: FE - This is a free edit post, in the category main, posted by moderator.
   tags_cache:
     - discussion
     - support
@@ -206,9 +204,9 @@ free_edit:
 
 high_trust:
   post_type: question
-  title: Q1 ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
+  title: Q5 - This is a question with high trust
+  body: Q5 - By not providing any information, you know you can trust everything said in this post.
+  body_markdown: Q5 - By not providing any information, you know you can trust everything said in this post.
   tags_cache:
     - discussion
     - support
@@ -227,8 +225,8 @@ high_trust:
 
 answer_one:
   post_type: answer
-  body: A1 ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  body: A1 - This is the first answer to question number 1 (Q1). It was posted by standard user.
+  body_markdown: A1 - This is the first answer to question number 1 (Q1). It was posted by standard user.
   score: 0.5
   parent: question_one
   user: standard_user
@@ -240,8 +238,8 @@ answer_one:
 
 answer_two:
   post_type: answer
-  body: A2 ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  body: A2 - This is the second answer to question number 1 (Q1). It was posted by editor.
+  body_markdown: A2 - This is the second answer to question number 1 (Q1). It was posted by editor.
   score: 0.5
   parent: question_one
   user: editor
@@ -253,8 +251,8 @@ answer_two:
 
 comments_disabled:
   post_type: answer
-  body: A2 ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  body: A3CD - This is the third answer to question number 1 (Q1). It has comments disabled. Posted by closer.
+  body_markdown: A3 - This is the third answer to question number 1 (Q1). It has comments disabled. Posted by closer.
   score: 0.5
   parent: question_one
   user: closer
@@ -267,8 +265,8 @@ comments_disabled:
 
 bad_answer:
   post_type: answer
-  body: A2 ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  body: A3BA - This is the fourth answer to question number 1 (Q1). It has a bad score. Posted by editor.
+  body_markdown: A4 - This is the fourth answer to question number 1 (Q1). It has a bad score. Posted by editor.
   score: 0.4
   parent: question_one
   user: editor
@@ -280,8 +278,8 @@ bad_answer:
 
 really_old_answer:
   post_type: answer
-  body: A3RO ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  body: A3RO - This is the fifth answer to question number 1 (Q1). It is very old. Posted by standard_user.
+  body_markdown: A3RO - This is the fifth answer to question number 1 (Q1). It is very old. Posted by standard_user.
   score: 2
   parent: question_one
   user: standard_user
@@ -294,8 +292,8 @@ really_old_answer:
 
 deleted_answer:
   post_type: answer
-  body: A4D ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  body: A4D - This is the sixth answer to question number 1 (Q1). It is deleted. Posted by closer.
+  body_markdown: A4D - This is the sixth answer to question number 1 (Q1). It is deleted. Posted by closer.
   score: 0.5
   parent: question_one
   user: closer
@@ -310,8 +308,8 @@ deleted_answer:
 
 policy_doc:
   post_type: policy_doc
-  body: PD ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ZY XWVUTS RQPONM LKJIHG FEDCBA ZY XWVUTS RQPONM LKJIHG FEDCBA
+  body: PD - This is a policy document called "Terms of Service", or "tos" for short.
+  body_markdown: PD - This is a policy document called "Terms of Service", or "tos" for short.
   title: Terms of Service
   doc_slug: tos
   user: admin
@@ -330,9 +328,9 @@ help_doc:
 
 article_one:
   post_type: article
-  title: Q1 ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
+  title: Art1 - This is the first article
+  body: Testing is an important practice, which should probably be discussed in this test article.
+  body_markdown: Testing is an important practice, which should probably be discussed in this test article.
   tags_cache:
     - discussion
     - support
@@ -351,9 +349,9 @@ article_one:
 
 deleted_article:
   post_type: article
-  title: Q1 ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
+  title: Art2D - This is a deleted article
+  body: Articles are most likely deleted because they have no inherent value, just like this sentence.
+  body_markdown: Articles are most likely deleted because they have no inherent value, just like this sentence.
   tags_cache:
     - discussion
     - support
@@ -375,8 +373,8 @@ deleted_article:
 
 help_article:
   post_type: help_doc
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
+  body: HA1 - This help article is in the category Site information, with "sample" as slug.
+  body_markdown: HA1 - This help article is in the category Site information, with "sample" as slug.
   user: system
   community: sample
   help_category: Site Information
@@ -385,8 +383,8 @@ help_article:
 
 mod_help_article:
   post_type: help_doc
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
+  body: HA2 - This help article is in the category for moderators only, with "sample-mod" as slug.
+  body_markdown: HA2 - This help article is in the category for moderators only, with "sample-mod" as slug.
   user: system
   community: sample
   help_category: $Moderator
@@ -395,8 +393,8 @@ mod_help_article:
 
 disabled_help_article:
   post_type: help_doc
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
+  body: HA3 - This help article is disabled. It is in the category disabled, with "sample-disable" as slug.
+  body_markdown: HA3 - This help article is disabled. It is in the category disabled, with "sample-disable" as slug.
   user: system
   community: sample
   help_category: $Disabled
@@ -405,9 +403,9 @@ disabled_help_article:
 
 blog_post:
   post_type: blog_post
-  title: B1 ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
-  body_markdown: ABCDEF GHIJKL MNOPQR STUVWX YZ ABCDEF GHIJKL MNOPQR STUVWX YZ
+  title: B1 - This is a blog post in the main category
+  body: B1 - This blog post is in the main category, posted by closer.
+  body_markdown: B1 - This blog post is in the main category, posted by closer.
   tags_cache:
     - discussion
     - support
diff --git a/test/fixtures/tag_sets.yml b/test/fixtures/tag_sets.yml
index df6517d05a995070b1ba35905d4dddc5e84db772..57b91b8a2caa6960e6bf2d925d8821dfe49dd3f1 100644
--- a/test/fixtures/tag_sets.yml
+++ b/test/fixtures/tag_sets.yml
@@ -5,3 +5,7 @@ main:
 meta:
   name: Meta
   community: sample
+
+empty:
+  name: 'Empty'
+  community: sample
diff --git a/test/fixtures/tag_synonyms.yml b/test/fixtures/tag_synonyms.yml
new file mode 100644
index 0000000000000000000000000000000000000000..48965bc780407b939a9dad31d36bd15a26422235
--- /dev/null
+++ b/test/fixtures/tag_synonyms.yml
@@ -0,0 +1,5 @@
+# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
+
+base_synonym:
+  name: synonym
+  tag: base
diff --git a/test/fixtures/tags.yml b/test/fixtures/tags.yml
index de28397edc0b3dff9cb435ab40e5e420264e5770..422794b4e61f74d1188c718f7a4dba5eb27ddcb5 100644
--- a/test/fixtures/tags.yml
+++ b/test/fixtures/tags.yml
@@ -10,6 +10,7 @@ support:
 
 bug:
   name: bug
+  excerpt: use for bug reports
   community: sample
   tag_set: main
 
@@ -38,3 +39,8 @@ child:
   community: sample
   tag_set: main
   parent: topic
+
+base:
+  name: base
+  community: sample
+  tag_set: main
diff --git a/test/fixtures/users.yml b/test/fixtures/users.yml
index e60c01e6c45a1425b6b6260cfdf0a01d88b7cf05..9cc566f84a57fac559eda8ccd97cfe2a3faed289 100644
--- a/test/fixtures/users.yml
+++ b/test/fixtures/users.yml
@@ -1,6 +1,15 @@
+basic_user:
+  email: basic@qpixel-test.net
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
+  sign_in_count: 42
+  username: basic_user
+  is_global_admin: false
+  is_global_moderator: false
+  confirmed_at: 2020-01-01T00:00:00.000000Z
+
 standard_user:
   email: standard@qpixel-test.net
-  encrypted_password: abcdefghijklmnopqrstuvwxyz
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
   sign_in_count: 1337
   username: standard_user
   is_global_admin: false
@@ -9,7 +18,7 @@ standard_user:
 
 closer:
   email: closer@qpixel-test.net
-  encrypted_password: abcdefghijklmnopqrstuvwxyz
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
   sign_in_count: 1337
   username: closer
   website: https://example.com/closer
@@ -22,7 +31,7 @@ closer:
 
 editor:
   email: editor@qpixel-test.net
-  encrypted_password: abcdefghijklmnopqrstuvwxyz
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
   sign_in_count: 1337
   username: editor
   is_global_admin: false
@@ -33,7 +42,7 @@ editor:
 
 deleter:
   email: delete@qpixel-test.net
-  encrypted_password: abcdefghijklmnopqrstuvwxyz
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
   sign_in_count: 1337
   username: deleter
   is_global_admin: false
@@ -42,7 +51,7 @@ deleter:
 
 moderator:
   email: moderator@qpixel-test.net
-  encrypted_password: abcdefghijklmnopqrstuvwxyz
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
   sign_in_count: 1337
   username: moderator
   is_global_admin: false
@@ -51,7 +60,7 @@ moderator:
 
 admin:
   email: admin@qpixel-test.net
-  encrypted_password: abcdefghijklmnopqrstuvwxyz
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
   sign_in_count: 1337
   username: admin
   is_global_admin: false
@@ -60,7 +69,7 @@ admin:
 
 global_moderator:
   email: global-moderator@qpixel-test.net
-  encrypted_password: abcdefghijklmnopqrstuvwxyz
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
   sign_in_count: 1337
   username: global-moderator
   is_global_admin: false
@@ -69,7 +78,7 @@ global_moderator:
 
 global_admin:
   email: global-admin@qpixel-test.net
-  encrypted_password: abcdefghijklmnopqrstuvwxyz
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
   sign_in_count: 1337
   username: global-admin
   is_global_admin: true
@@ -78,7 +87,7 @@ global_admin:
 
 no_community_user:
   email: no_community_user@qpixel-test.net
-  encrypted_password: abcdefghijklmnopqrstuvwxyz
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
   sign_in_count: 1337
   username: no_community_user
   is_global_admin: false
@@ -88,7 +97,7 @@ no_community_user:
 system:
   id: -99
   email: system@qpixel-test.net
-  encrypted_password: abcdefghijklmnopqrstuvwxyz
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
   sign_in_count: 1337
   username: system
   is_global_admin: true
@@ -97,7 +106,7 @@ system:
 
 deleted_account:
   email: deleted@qpixel-test.net
-  encrypted_password: abcdefghijklmnopqrstuvwxyz
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
   sign_in_count: 1337
   username: deleted_account
   confirmed_at: 2020-01-01T00:00:00.000000Z
@@ -107,7 +116,17 @@ deleted_account:
 
 deleted_profile:
   email: deleted_profile@qpixel-test.net
-  encrypted_password: abcdefghijklmnopqrstuvwxyz
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
   sign_in_count: 1337
   username: deleted_profile
   confirmed_at: 2020-01-01T00:00:00.000000Z
+
+enabled_2fa:
+  email: 2fa@qpixel-test.net
+  encrypted_password: '$2a$11$roUHXKxecjyQ72Qn7DWs3.9eRCCoRn176kX/UNb/xiue3aGqf7xEW'
+  sign_in_count: 1337
+  username: enabled_2fa
+  confirmed_at: 2020-01-01T00:00:00.000000Z
+  enabled_2fa: true
+  two_factor_token: WT65ANYXBB2SBR7III7IVWNJDS4PQF2T
+  backup_2fa_code: M8lENyehyCvo9F9MbyTl1aOL
diff --git a/test/helpers/comments_helper_test.rb b/test/helpers/comments_helper_test.rb
index fca0964bd13acc515b30657c0ab9c8f6ed54e6c6..42c6bde74c2fa405a88a350e68110c9542dd6d19 100644
--- a/test/helpers/comments_helper_test.rb
+++ b/test/helpers/comments_helper_test.rb
@@ -28,7 +28,7 @@ class CommentsHelperTest < ActionView::TestCase
   test '[flags?] substitution' do
     expected = {
       '[flag] me if you can' => "<a href=\"#{flag_history_url(users(:standard_user).id)}\">flag</a> me if you can",
-      '\'cause it\'s our [flags]hip product' => "\'cause it\'s our <a href=\"#{flag_history_url(users(:standard_user).id)}\">flags</a>hip product",
+      '\'cause it\'s our [flags]hip product' => "'cause it's our <a href=\"#{flag_history_url(users(:standard_user).id)}\">flags</a>hip product",
       'yeah bad pun - [flagged] and downvoted' => 'yeah bad pun - [flagged] and downvoted'
     }
     expected.each do |input, expect|
diff --git a/test/jobs/clean_up_spammy_users_job_test.rb b/test/jobs/clean_up_spammy_users_job_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..8d84e8a91b4151c332c29ce243a20a13d2511a44
--- /dev/null
+++ b/test/jobs/clean_up_spammy_users_job_test.rb
@@ -0,0 +1,7 @@
+require 'test_helper'
+
+class CleanUpSpammyUsersJobTest < ActiveJob::TestCase
+  # test "the truth" do
+  #   assert true
+  # end
+end
diff --git a/test/mailers/previews/two_factor_mailer_preview.rb b/test/mailers/previews/two_factor_mailer_preview.rb
index 804f8701237ecca0ba46da27aa3d14343fdbe1ca..2857de292e2894bb3dcd3c570520e208a5cc517e 100644
--- a/test/mailers/previews/two_factor_mailer_preview.rb
+++ b/test/mailers/previews/two_factor_mailer_preview.rb
@@ -7,4 +7,8 @@ class TwoFactorMailerPreview < ActionMailer::Preview
   def login_email_preview
     TwoFactorMailer.with(user: User.last, host: 'testhost.codidact.com').login_email
   end
+
+  def backup_code_preview
+    TwoFactorMailer.with(user: User.last, host: 'testhost.codidact.com').backup_code
+  end
 end
diff --git a/test/models/category_filter_default_test.rb b/test/models/category_filter_default_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..72358b7c03902c854b32be4b350bdea5572f410d
--- /dev/null
+++ b/test/models/category_filter_default_test.rb
@@ -0,0 +1,7 @@
+require 'test_helper'
+
+class CategoryFilterDefaultTest < ActiveSupport::TestCase
+  # test "the truth" do
+  #   assert true
+  # end
+end
diff --git a/test/models/email_log_test.rb b/test/models/email_log_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..12c90407fb8aa6c6ecbb07cb9d2a946130423cba
--- /dev/null
+++ b/test/models/email_log_test.rb
@@ -0,0 +1,7 @@
+require 'test_helper'
+
+class EmailLogTest < ActiveSupport::TestCase
+  # test "the truth" do
+  #   assert true
+  # end
+end
diff --git a/test/models/filter_test.rb b/test/models/filter_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..418de9e94f6cae1f83207e6f94d0a9dbb5834612
--- /dev/null
+++ b/test/models/filter_test.rb
@@ -0,0 +1,7 @@
+require 'test_helper'
+
+class FilterTest < ActiveSupport::TestCase
+  # test "the truth" do
+  #   assert true
+  # end
+end
diff --git a/test/models/tag_set_test.rb b/test/models/tag_set_test.rb
index c5ea5ed117f34bd83c271398c1eaa0357116f15e..39e8b086bf2762fa366eec284a8eecc06330dc1d 100644
--- a/test/models/tag_set_test.rb
+++ b/test/models/tag_set_test.rb
@@ -1,7 +1,18 @@
 require 'test_helper'
 
 class TagSetTest < ActiveSupport::TestCase
-  # test "the truth" do
-  #   assert true
-  # end
+  include CommunityRelatedHelper
+
+  test 'is community related' do
+    assert_community_related(TagSet)
+  end
+
+  test 'with_paths method should respect no_excerpt' do
+    main = TagSet.main
+
+    all = main.with_paths.size
+    excerptless = main.with_paths(true).size
+
+    assert_not_equal(all, excerptless)
+  end
 end
diff --git a/test/models/tag_synonym_test.rb b/test/models/tag_synonym_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..a5ea423be62aeaf90e7264ab4ee7d2b32bc15c1d
--- /dev/null
+++ b/test/models/tag_synonym_test.rb
@@ -0,0 +1,7 @@
+require 'test_helper'
+
+class TagSynonymTest < ActiveSupport::TestCase
+  # test "the truth" do
+  #   assert true
+  # end
+end
diff --git a/test/models/user_test.rb b/test/models/user_test.rb
index 2335370686d0a6a5dbcbb0493622b59417cc4d95..9d8cfd64b967213a8c3afe0aa4bdbf4d229ddeb2 100644
--- a/test/models/user_test.rb
+++ b/test/models/user_test.rb
@@ -25,6 +25,39 @@ class UserTest < ActiveSupport::TestCase
     assert_equal 'example.com', users(:closer).website_domain
   end
 
+  test 'can_update should determine if the user can update a given post' do
+    basic_user = users(:basic_user)
+    post_owner = users(:standard_user)
+    category = categories(:main)
+    license = licenses(:cc_by_sa)
+    post_type = post_types(:question)
+    post = Post.create(body_markdown: 'rev 1',
+                       body: '<p>rev 1</p>',
+                       title: 'test post',
+                       tags_cache: ['test'],
+                       license: license,
+                       score: 0,
+                       user: post_owner,
+                       post_type: post_type,
+                       category: category)
+
+    assert_equal true, post_owner.can_update(post, post_type)
+    assert_equal false, basic_user.can_update(post, post_type)
+    assert_equal true, users(:moderator).can_update(post, post_type)
+    assert_equal true, users(:editor).can_update(post, post_type)
+
+    basic_user.community_user.grant_privilege!('unrestricted')
+    assert_equal false, basic_user.can_update(post, post_type)
+    assert_equal true, basic_user.can_update(post, post_types(:free_edit))
+  end
+
+  test 'can_push_to_network should determine if the user can push updates to network' do
+    post_type = post_types(:help_doc)
+    assert_equal false, users(:standard_user).can_push_to_network(post_type)
+    assert_equal true, users(:global_moderator).can_push_to_network(post_type)
+    assert_equal true, users(:global_admin).can_push_to_network(post_type)
+  end
+
   test 'community_user is based on context' do
     user = users(:standard_user)
     community = Community.create(host: 'other', name: 'Other')
diff --git a/test/system/login_test.rb b/test/system/login_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..5f799dbc5b1d1bc943de6e4c089a98951a52497a
--- /dev/null
+++ b/test/system/login_test.rb
@@ -0,0 +1,57 @@
+require 'application_system_test_case'
+
+class LoginTest < ApplicationSystemTestCase
+  test 'User can register a new account and sign-in to it after confirming their email' do
+    email = 'test@test.com'
+    username = 'Test User'
+    password = 'login_test_1'
+
+    # Sign up for an account
+    visit root_url
+    click_on 'Sign Up'
+    fill_in 'Email', with: email
+    fill_in 'Username', with: username
+    fill_in 'Password', with: password
+    fill_in 'Password confirmation', with: password
+
+    # Check that the user is created in the DB
+    assert_difference 'User.count' do
+      click_on 'Sign up'
+    end
+
+    user = User.last
+
+    # Try logging in directly, this should fail because not confirmed yet
+    log_in user, password
+    assert_selector '.notice', text: 'You have to confirm your email address before continuing.'
+
+    # Confirm email and sign in again, should succeed this time
+    confirm_email user
+    log_in user, password
+    assert_selector '.notice', text: 'Signed in successfully.'
+  end
+
+  test 'User can sign in and is redirected back to the page they were on' do
+    # Start on the users page
+    visit users_url
+
+    # Click the sign in button (top right)
+    # Don't go through log_in helper, since we want to test the sign-in fully here
+    click_on 'Sign In'
+    fill_in 'Email', with: users(:standard_user).email
+    fill_in 'Password', with: 'test123'
+    click_button 'Sign in'
+
+    # We should see a message that we have signed in, and we should be on the users page again.
+    assert_selector '.notice', text: 'Signed in successfully.'
+    assert_current_path users_url
+  end
+
+  test 'User can sign out' do
+    log_in :standard_user
+    assert_selector '.notice', text: 'Signed in successfully.'
+
+    log_out
+    assert_selector '.notice', text: 'Signed out successfully.'
+  end
+end
diff --git a/test/system/post_test.rb b/test/system/post_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..d99cd266cab58145a3a2c59942ef050ee184fa33
--- /dev/null
+++ b/test/system/post_test.rb
@@ -0,0 +1,166 @@
+require 'application_system_test_case'
+
+class PostTest < ApplicationSystemTestCase
+  # -------------------------------------------------------
+  # Create
+  # -------------------------------------------------------
+
+  test 'Not-signed in user cannot create a post' do
+    visit root_url
+    click_on 'Create Post'
+
+    assert_current_path new_user_session_url
+  end
+
+  test 'Signed in user can create a question' do
+    category = categories(:meta)
+    log_in :standard_user
+    visit category_path(category)
+    click_on 'Create Post'
+
+    body_text = 'When running QPixel, users are generally supposed to be able to create posts. ' \
+                'Does that actually work?'
+    title_text = 'Can a signed-in user create a post?'
+
+    fill_in 'Body', with: body_text
+    fill_in 'Summarize your post with a title:', with: title_text
+    post_form_select_tag tags(:faq).name
+
+    # Check that the post is actually created
+    assert_difference 'Post.count' do
+      click_on "Save Post in #{category.name}"
+    end
+
+    # Verify that the post is correctly created
+    new_post = Post.last
+    assert_equal body_text, new_post.body_markdown
+    assert_equal title_text, new_post.title
+    assert_equal [tags(:faq)], new_post.tags
+  end
+
+  test 'Creating a question is blocked when body is too short' do
+    category = categories(:meta)
+    log_in :standard_user
+    visit category_path(category)
+    click_on 'Create Post'
+
+    fill_in 'Summarize your post with a title:', with: 'Initial title is of sufficient length'
+    post_form_select_tag tags(:faq).name
+    fill_in 'Body', with: 'Short'
+
+    # Check that the button is disabled
+    find_button "Save Post in #{category.name}", disabled: true
+
+    # After filling out body correctly, verify that the button becomes enabled
+    fill_in 'Body', with: 'This body should pass the minimum length requirements for questions in the meta category.'
+    find_button "Save Post in #{category.name}", disabled: false
+  end
+
+  test 'Creating a question is blocked when title is too short' do
+    category = categories(:meta)
+    log_in :standard_user
+    visit category_path(category)
+    click_on 'Create Post'
+
+    fill_in 'Body', with: 'This body should pass the minimum length requirements for questions in the meta category.'
+    post_form_select_tag tags(:faq).name
+    fill_in 'Summarize your post with a title:', with: 'Too short'
+
+    # Check that the button is disabled
+    find_button "Save Post in #{category.name}", disabled: true
+
+    # After filling out the title, verify that the button becomes enabled
+    fill_in 'Summarize your post with a title:', with: 'Updated title is of sufficient length'
+    find_button "Save Post in #{category.name}", disabled: false
+  end
+
+  test 'Signed in user gets to pick post type for post creation in categories with multiple types' do
+    category = categories(:main)
+    log_in :standard_user
+    visit category_path(category)
+    click_on 'Create Post'
+
+    # All the top level post types set should be present
+    category.post_types.where(is_top_level: true).each do |pt|
+      assert_link pt.name.underscore.humanize
+    end
+
+    # Pick a non-question post type
+    post_type = category.post_types.where(is_top_level: true).where.not(name: 'Question').first
+
+    # After clicking on a post type, we should be on the creation page of the correct category and post type.
+    click_on post_type.name.underscore.humanize
+    assert_current_path new_category_post_url(post_type.id, category.id)
+  end
+
+  test 'Signed in user can answer question' do
+    log_in :standard_user
+    post = posts(:question_two)
+    visit post_path(post)
+
+    # Answer the question
+    answer_text = 'You can do this by running the rails system tests, rails test:system.'
+    fill_in 'Body', with: answer_text
+    assert_difference 'Post.count' do
+      click_on "Save Post in #{post.category.name}"
+    end
+
+    # We should now be looking at our answer, look for the text on the page
+    assert_text answer_text
+
+    # The original post should also still be on the page
+    assert_text post.body
+  end
+
+  # -------------------------------------------------------
+  # Show
+  # -------------------------------------------------------
+
+  test 'Anyone can view question' do
+    post = posts(:question_one)
+    visit post_url(post)
+
+    # Check that the post is displayed somewhere on the page
+    assert_text post.title
+    assert_text post.body
+
+    # Check that answers are displayed somewhere on the page
+    assert post.children.any?, 'The post for this system test should have answers'
+    post.children.where(deleted: false).each do |child|
+      assert_text child.body
+    end
+  end
+
+  test 'Anyone can sort answers' do
+    post = posts(:question_one)
+    visit post_url(post)
+
+    click_on 'Active'
+
+    assert_current_path post_url(post, sort: 'active')
+  end
+
+  # -------------------------------------------------------
+  # Edit
+  # -------------------------------------------------------
+
+  test 'User with edit permissions can directly edit question' do
+    log_in :editor
+    post = posts(:question_two)
+    visit post_url(post)
+
+    within ".post[data-post-id=\"#{post.id}\"]" do
+      click_on 'Edit'
+    end
+
+    updated_text = 'This is the updated body text, which should be quite different from the original text!'
+    fill_in 'Body', with: updated_text
+    fill_in 'Edit Comment', with: 'Major Rewrite for Tests'
+
+    click_on "Save Post in #{post.category.name}"
+    assert_current_path post_url(post)
+
+    # Check that the page shows the updated text
+    assert_text updated_text
+  end
+end
diff --git a/test/test_helper.rb b/test/test_helper.rb
index c41d9b27cb6b2dfb3db4aeaf003e4aca620fc94d..638cf4085e5b6ac873fb83e28d4444f1e82bf004 100644
--- a/test/test_helper.rb
+++ b/test/test_helper.rb
@@ -14,18 +14,39 @@ class ActiveSupport::TestCase
   # Setup all fixtures in test/fixtures/*.yml for all tests in alphabetical order.
   fixtures :all
 
-  setup :load_seeds
+  setup :set_request_context
 
   teardown :clear_cache
 
   protected
 
-  def load_seeds
+  # Overrides minitest' load_fixtures method to also load our seeds when fixtures are loaded.
+  # This means that we can leverage it's smart transaction behavior to significantly speed up our tests (by a factor of 6).
+  def load_fixtures(config)
+    # Loading a fixture deletes all data in the same tables, so it has to happen before we load our normal seeds.
+    fixture_data = super(config)
+    load_tags_paths
+    load_seeds
+
+    # We do need to return the same thing as the original method to not break fixtures
+    fixture_data
+  end
+
+  # Ensures that a community is set for all requests that will be made (on this thread)
+  def set_request_context
     comm = Community.first || Community.create(name: 'Test', host: 'test.host')
     RequestContext.community = comm
+  end
+
+  def load_seeds
+    set_request_context
     Rails.application.load_seed
   end
 
+  def load_tags_paths
+    ActiveRecord::Base.connection.execute File.read(Rails.root.join('db/scripts/create_tags_path_view.sql'))
+  end
+
   def clear_cache
     Rails.cache.clear
   end