tilde.news/app/controllers/application_controller.rb

171 lines
5.3 KiB
Ruby

class ApplicationController < ActionController::Base
protect_from_forgery
before_action :authenticate_user
before_action :increase_traffic_counter
TRAFFIC_DECREMENTER = 0.50
# match this in your nginx config for bypassing the file cache
TAG_FILTER_COOKIE = :tag_filters
CACHE_PAGE = proc { @user.blank? && cookies[TAG_FILTER_COOKIE].blank? }
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
Rails.logger.info " Logged in as user #{@user.id} (#{@user.username})"
end
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
def increase_traffic_counter
if Rails.application.read_only?
return true
end
@traffic = 1.0
Keystore.transaction do
now_i = Time.now.to_i
date_kv = Keystore.find_or_create_key_for_update("traffic:date", now_i)
traffic_kv = Keystore.find_or_create_key_for_update("traffic:hits", 0)
traffic = traffic_kv.value.to_i
# don't increase traffic counter for bots or api requests
unless agent_is_spider? || ["json", "rss"].include?(params[:format])
traffic += 100
end
# every second, decrement traffic by some amount
traffic -= (100.0 * (now_i - date_kv.value) * TRAFFIC_DECREMENTER).to_i
# clamp to 100, 1000
traffic = [[100, traffic].max, 10_000].min
@traffic = traffic * 0.01
traffic_kv.value = traffic
traffic_kv.save!
date_kv.value = now_i
date_kv.save!
Rails.logger.info " Traffic level: #{@traffic.to_i}"
end
# logo background intensity is based on traffic
intensity = sprintf('%02x', [(@traffic * 7).floor + 50.0, 255].min)
set_traffic_style intensity
true
end
# http://umaine.edu/lobsterinstitute/files/2011/12/LobsterColorsWeb.pdf
def set_traffic_style intensity
@traffic_style = "background-color: ##{intensity}0000;"
return unless @user
color = :red
[
# rubocop:disable Metrics/LineLength,
[2_000_000, :blue, "background-color: #0000#{intensity};"],
[6, :yellow, "background-color: ##{intensity}#{intensity}00;"],
[3, :calico, "background: url() no-repeat center"],
[2, :split, "background: linear-gradient(90deg, ##{intensity}0000 50%, #0000#{intensity} 50%)"],
[2, :albino, "filter: invert(100%);"],
# rubocop:enable Metrics/LineLength,
].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."
return 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."
return redirect_to "/"
end
end
end
def require_logged_in_user_or_400
if @user
true
else
render :plain => "not logged in", :status => 400
return false
end
end
def tags_filtered_by_cookie
@_tags_filtered ||= Tag.where(
:tag => cookies[TAG_FILTER_COOKIE].to_s.split(",")
)
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 find_user_from_rss_token
if !@user && request[:format] == "rss" && params[:token].to_s.present?
@user = User.where(:rss_token => params[:token].to_s).first
end
end
end