2023-11-09 15:44:05 -05:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
class Statistics
|
2024-08-20 17:03:42 -04:00
|
|
|
EU_COUNTRIES = %w[
|
|
|
|
AT
|
|
|
|
BE
|
|
|
|
BG
|
|
|
|
CY
|
|
|
|
CZ
|
|
|
|
DE
|
|
|
|
DK
|
|
|
|
EE
|
|
|
|
ES
|
|
|
|
FI
|
|
|
|
FR
|
|
|
|
GR
|
|
|
|
HR
|
|
|
|
HU
|
|
|
|
IE
|
|
|
|
IT
|
|
|
|
LT
|
|
|
|
LU
|
|
|
|
LV
|
|
|
|
MT
|
|
|
|
NL
|
|
|
|
PL
|
|
|
|
PT
|
|
|
|
RO
|
|
|
|
SE
|
|
|
|
SI
|
|
|
|
SK
|
|
|
|
]
|
|
|
|
|
2023-11-09 15:44:05 -05:00
|
|
|
def self.active_users
|
|
|
|
{
|
2024-08-12 17:47:13 -04:00
|
|
|
last_day: User.where("last_seen_at > ?", 1.day.ago).count,
|
2023-11-09 15:44:05 -05:00
|
|
|
"7_days": User.where("last_seen_at > ?", 7.days.ago).count,
|
|
|
|
"30_days": User.where("last_seen_at > ?", 30.days.ago).count,
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.likes
|
|
|
|
{
|
|
|
|
last_day:
|
2024-08-12 17:47:13 -04:00
|
|
|
UserAction.where(action_type: UserAction::LIKE).where("created_at > ?", 1.day.ago).count,
|
2023-11-09 15:44:05 -05:00
|
|
|
"7_days":
|
|
|
|
UserAction.where(action_type: UserAction::LIKE).where("created_at > ?", 7.days.ago).count,
|
|
|
|
"30_days":
|
|
|
|
UserAction.where(action_type: UserAction::LIKE).where("created_at > ?", 30.days.ago).count,
|
|
|
|
count: UserAction.where(action_type: UserAction::LIKE).count,
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.posts
|
|
|
|
{
|
2024-08-12 17:47:13 -04:00
|
|
|
last_day: Post.where("created_at > ?", 1.day.ago).count,
|
2023-11-09 15:44:05 -05:00
|
|
|
"7_days": Post.where("created_at > ?", 7.days.ago).count,
|
|
|
|
"30_days": Post.where("created_at > ?", 30.days.ago).count,
|
|
|
|
count: Post.count,
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.topics
|
|
|
|
{
|
2024-08-12 17:47:13 -04:00
|
|
|
last_day: Topic.listable_topics.where("created_at > ?", 1.day.ago).count,
|
2023-11-09 15:44:05 -05:00
|
|
|
"7_days": Topic.listable_topics.where("created_at > ?", 7.days.ago).count,
|
|
|
|
"30_days": Topic.listable_topics.where("created_at > ?", 30.days.ago).count,
|
|
|
|
count: Topic.listable_topics.count,
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.users
|
2024-09-12 15:28:49 -04:00
|
|
|
query = User.real.not_silenced.activated
|
|
|
|
|
|
|
|
query = query.where(approved: true) if SiteSetting.must_approve_users
|
|
|
|
|
2023-11-09 15:44:05 -05:00
|
|
|
{
|
2024-09-12 15:28:49 -04:00
|
|
|
last_day: query.where("created_at > ?", 1.day.ago).count,
|
|
|
|
"7_days": query.where("created_at > ?", 7.days.ago).count,
|
|
|
|
"30_days": query.where("created_at > ?", 30.days.ago).count,
|
|
|
|
count: query.count,
|
2023-11-09 15:44:05 -05:00
|
|
|
}
|
|
|
|
end
|
2024-08-12 17:47:13 -04:00
|
|
|
|
|
|
|
def self.participating_users
|
|
|
|
{
|
|
|
|
last_day: participating_users_count(1.day.ago),
|
|
|
|
"7_days": participating_users_count(7.days.ago),
|
|
|
|
"30_days": participating_users_count(30.days.ago),
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2024-08-20 17:03:42 -04:00
|
|
|
def self.visitors
|
|
|
|
periods = [[1.day.ago, :last_day], [7.days.ago, :"7_days"], [30.days.ago, :"30_days"]]
|
|
|
|
|
|
|
|
periods
|
|
|
|
.map do |(period, key)|
|
|
|
|
anon_page_views =
|
|
|
|
ApplicationRequest.request_type_count_for_period(:page_view_anon_browser, period)
|
|
|
|
|
|
|
|
logged_in_visitors = logged_in_visitors_count(period)
|
|
|
|
next key, anon_page_views if logged_in_visitors == 0
|
|
|
|
|
|
|
|
logged_in_page_views =
|
|
|
|
ApplicationRequest.request_type_count_for_period(:page_view_logged_in_browser, period)
|
|
|
|
next key, anon_page_views + logged_in_visitors if logged_in_page_views == 0
|
|
|
|
|
|
|
|
total_visitors = logged_in_visitors
|
|
|
|
avg_logged_in_page_view_per_user = logged_in_page_views.to_f / logged_in_visitors
|
|
|
|
anon_visitors = (anon_page_views / avg_logged_in_page_view_per_user).round
|
|
|
|
total_visitors += anon_visitors
|
|
|
|
[key, total_visitors]
|
|
|
|
end
|
|
|
|
.to_h
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.eu_visitors
|
|
|
|
periods = [[1.day.ago, :last_day], [7.days.ago, :"7_days"], [30.days.ago, :"30_days"]]
|
|
|
|
|
|
|
|
periods
|
|
|
|
.map do |(period, key)|
|
|
|
|
logged_in_page_views =
|
|
|
|
ApplicationRequest.request_type_count_for_period(:page_view_logged_in_browser, period)
|
|
|
|
anon_page_views =
|
|
|
|
ApplicationRequest.request_type_count_for_period(:page_view_anon_browser, period)
|
|
|
|
|
|
|
|
all_logged_in_visitors = logged_in_visitors_count(period)
|
|
|
|
eu_logged_in_visitors = eu_logged_in_visitors_count(period)
|
|
|
|
|
|
|
|
next key, 0 if all_logged_in_visitors == 0 || eu_logged_in_visitors == 0
|
|
|
|
next key, eu_logged_in_visitors if logged_in_page_views == 0
|
|
|
|
|
|
|
|
avg_logged_in_page_view_per_user = logged_in_page_views / all_logged_in_visitors.to_f
|
|
|
|
|
|
|
|
eu_logged_in_visitors_ratio = eu_logged_in_visitors / all_logged_in_visitors.to_f
|
|
|
|
|
|
|
|
eu_anon_visitors =
|
|
|
|
((anon_page_views / avg_logged_in_page_view_per_user) * eu_logged_in_visitors_ratio).round
|
|
|
|
eu_visitors = eu_logged_in_visitors + eu_anon_visitors
|
|
|
|
[key, eu_visitors]
|
|
|
|
end
|
|
|
|
.to_h
|
|
|
|
end
|
|
|
|
|
2024-08-12 17:47:13 -04:00
|
|
|
private
|
|
|
|
|
|
|
|
def self.participating_users_count(date)
|
|
|
|
subqueries = [
|
|
|
|
"SELECT DISTINCT user_id FROM user_actions WHERE created_at > :date AND action_type IN (:action_types)",
|
|
|
|
]
|
|
|
|
|
|
|
|
if ActiveRecord::Base.connection.data_source_exists?("chat_messages")
|
|
|
|
subqueries << "SELECT DISTINCT user_id FROM chat_messages WHERE created_at > :date"
|
|
|
|
end
|
|
|
|
|
|
|
|
if ActiveRecord::Base.connection.data_source_exists?("chat_message_reactions")
|
|
|
|
subqueries << "SELECT DISTINCT user_id FROM chat_message_reactions WHERE created_at > :date"
|
|
|
|
end
|
|
|
|
|
|
|
|
sql = "SELECT COUNT(user_id) FROM (#{subqueries.join(" UNION ")}) u"
|
|
|
|
|
|
|
|
DB.query_single(sql, date: date, action_types: UserAction::USER_ACTED_TYPES).first
|
|
|
|
end
|
2024-08-20 17:03:42 -04:00
|
|
|
|
|
|
|
def self.logged_in_visitors_count(since)
|
|
|
|
DB.query_single(<<~SQL, since:).first
|
|
|
|
SELECT COUNT(DISTINCT user_id)
|
|
|
|
FROM user_visits
|
|
|
|
WHERE visited_at >= :since
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.eu_logged_in_visitors_count(since)
|
|
|
|
results = DB.query_hash(<<~SQL, since:)
|
|
|
|
SELECT DISTINCT(user_id), ip_address
|
|
|
|
FROM user_visits uv
|
|
|
|
INNER JOIN users u
|
|
|
|
ON u.id = uv.user_id
|
|
|
|
WHERE visited_at >= :since AND ip_address IS NOT NULL
|
|
|
|
SQL
|
|
|
|
|
|
|
|
results.reduce(0) do |sum, hash|
|
|
|
|
ip_info = DiscourseIpInfo.get(hash["ip_address"].to_s)
|
|
|
|
sum + (EU_COUNTRIES.include?(ip_info[:country_code]) ? 1 : 0)
|
|
|
|
end
|
|
|
|
end
|
2023-11-09 15:44:05 -05:00
|
|
|
end
|