tilde.news/app/controllers/application_controller.rb

201 lines
6.6 KiB
Ruby

# typed: false
class ApplicationController < ActionController::Base
include IntervalHelper
protect_from_forgery
before_action :authenticate_user
before_action :heinous_inline_partials, if: -> { Rails.env.development? }
before_action :mini_profiler
before_action :prepare_exception_notifier
before_action :set_traffic_style
# 2023-10-07 one user in one of their browser envs is getting a CSRF failure, I'm reverting
# because I'll be AFK a while.
# after_action :clear_lobster_trap
# match this nginx config for bypassing the file cache
TAG_FILTER_COOKIE = :tag_filters
CACHE_PAGE = proc { @user.blank? && cookies[TAG_FILTER_COOKIE].blank? }
rescue_from ActionController::UnknownFormat do
render plain: "404 Not Found", status: :not_found, content_type: "text/plain"
end
rescue_from ActionController::UnpermittedParameters do
render plain: "400 Unpermitted query or form paramater", status: :bad_request, content_type: "text/plain"
end
rescue_from ActionDispatch::Http::MimeNegotiation::InvalidType do
render plain: "fix the mime type in your HTTP_ACCEPT header",
status: :bad_request, content_type: "text/plain"
end
def agent_is_spider?
ua = request.env["HTTP_USER_AGENT"].to_s
ua == "" || ua.match(/(Google|bing|Slack|Twitter)bot|Slurp|crawler|Feedly|FeedParser|RSS/)
end
def authenticate_user
# eagerly evaluate, in case this triggers an IpSpoofAttackError
request.remote_ip
if Rails.application.read_only?
return true
end
if session[:u] &&
(user = User.find_by(session_token: session[:u].to_s)) &&
user.is_active?
@user = user
end
Rails.logger.info(
" Request #{request.remote_ip} #{request.request_method} #{request.fullpath} user: " +
(@user ? "#{@user.id} #{@user.username}" : "0 nobody")
)
true
end
def check_for_read_only_mode
if Rails.application.read_only?
flash.now[:error] = "Site is currently in read-only mode."
return redirect_to "/"
end
true
end
# clear Rails session cookie if not logged in so nginx uses the page cache
# https://ryanfb.xyz/etc/2021/08/29/going_cookie-free_with_rails.html
def clear_lobster_trap
key = Rails.application.config.session_options[:key] # "lobster_trap"
cookies.delete(key) if @user.blank?
# this probably should test session.empty? && controller...
request.session_options[:skip] = @user.blank? && controller_name != "login"
end
def find_user_from_rss_token
if !@user && params[:format] == "rss" && params[:token].to_s.present?
@user = User.where(rss_token: params[:token].to_s).first
end
end
def flag_warning
@flag_warning_int ||= time_interval("1m")
return false if Rails.env.development? # expensive because Rails doesn't cache in dev
@show_flag_warning ||= @user && !!FlaggedCommenters.new(@flag_warning_int[:param], 1.day).check_list_for(@user)
end
def heinous_inline_partials
do_heinous_inline_partial_replacement
end
def mini_profiler
if @user&.is_moderator?
Rack::MiniProfiler.authorize_request
end
end
def prepare_exception_notifier
exception_data = {}
exception_data[:username] = @user.username unless @user.nil?
request.env["exception_notifier.exception_data"] = exception_data
end
# https://web.archive.org/web/20180108083712/http://umaine.edu/lobsterinstitute/files/2011/12/LobsterColorsWeb.pdf
def set_traffic_style
@traffic_intensity = "?"
@traffic_style = "background-color: #ac130d;"
return true if Rails.application.read_only? ||
agent_is_spider? ||
%w[json rss].include?(params[:format])
if (skip = TrafficHelper.novelty_logo)
@traffic_style = skip
return
end
@traffic_intensity = TrafficHelper.cached_current_intensity
# map intensity to 80-255 so there's always a little red
hex = sprintf("%02x", (@traffic_intensity * 1.75 + 80).round)
@traffic_style = "background-color: ##{hex}0000;"
return true unless @user
color = :red
[
[2_000_000, :blue, "background-color: #0000#{hex};"],
[6, :yellow, "background-color: ##{hex}#{hex}00;"],
[3, :calico, "background: url() no-repeat center"],
[2, :split, "background: linear-gradient(90deg, ##{hex}0000 50%, #0000#{hex} 50%)"],
[2, :albino, "filter: invert(100%);"]
].each do |cumulative_odds, name, style|
break unless rand(cumulative_odds) == 0
color = name
@traffic_style = style
end
if color != :red
Rails.logger.info " Lucky user #{@user.username} saw #{color} logo"
end
end
def require_logged_in_user
if @user
true
else
if request.get?
session[:redirect_to] = request.original_fullpath
end
redirect_to "/login"
end
end
def require_logged_in_moderator
require_logged_in_user
if @user
if @user.is_moderator?
true
else
flash[:error] = "You are not authorized to access that resource."
redirect_to "/"
end
end
end
def require_logged_in_admin
require_logged_in_user
if @user
if @user.is_admin?
true
else
flash[:error] = "You are not authorized to access that resource."
redirect_to "/"
end
end
end
def require_logged_in_user_or_400
if @user
true
else
render plain: "not logged in", status: 400
false
end
end
def require_no_user_or_redirect
redirect_to "/" if @user
end
def show_title_h1
@title_h1 = true
end
def tags_filtered_by_cookie
@_tags_filtered ||= Tag.where(
tag: cookies[TAG_FILTER_COOKIE].to_s.split(",")
)
end
end