2019-01-21 15:17:56 +01:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
# Provides helper methods for listing Users
|
|
|
|
class Members::ListService
|
|
|
|
class << self
|
|
|
|
def list(params)
|
2023-05-29 10:34:14 +02:00
|
|
|
@query = User.includes(:profile, :group, :statistic_profile, :children)
|
2019-01-21 15:17:56 +01:00
|
|
|
.joins(:profile,
|
2019-06-04 16:50:23 +02:00
|
|
|
:statistic_profile,
|
2019-01-21 15:17:56 +01:00
|
|
|
:group,
|
|
|
|
:roles,
|
2019-06-05 12:11:51 +02:00
|
|
|
'LEFT JOIN (
|
|
|
|
SELECT *
|
|
|
|
FROM "subscriptions" AS s1
|
|
|
|
INNER JOIN (
|
|
|
|
SELECT MAX("created_at") AS "s2_created_at", "statistic_profile_id" AS "s2_statistic_profile_id"
|
|
|
|
FROM "subscriptions"
|
|
|
|
GROUP BY "statistic_profile_id"
|
2020-05-13 10:59:16 +02:00
|
|
|
) AS s2
|
2019-06-05 12:11:51 +02:00
|
|
|
ON "s1"."statistic_profile_id" = "s2"."s2_statistic_profile_id"
|
|
|
|
WHERE "s1"."expiration_date" > now()::date
|
|
|
|
) AS "subscriptions" ON "subscriptions"."statistic_profile_id" = "statistic_profiles"."id" ' \
|
2019-01-21 15:17:56 +01:00
|
|
|
'LEFT JOIN "plans" ON "plans"."id" = "subscriptions"."plan_id"')
|
|
|
|
.where("users.is_active = 'true' AND roles.name = 'member'")
|
|
|
|
.order(list_order(params))
|
|
|
|
|
|
|
|
# ILIKE => PostgreSQL case-insensitive LIKE
|
|
|
|
if params[:search].size.positive?
|
2022-07-21 13:21:06 -03:00
|
|
|
@query = @query.where('users.username ILIKE :search OR ' \
|
2023-05-29 10:34:14 +02:00
|
|
|
"profiles.first_name || ' ' || profiles.last_name ILIKE :search OR " \
|
2019-01-21 15:17:56 +01:00
|
|
|
'profiles.phone ILIKE :search OR ' \
|
2023-05-29 10:34:14 +02:00
|
|
|
'users.email ILIKE :search OR ' \
|
|
|
|
"children.first_name || ' ' || children.last_name ILIKE :search OR " \
|
2019-01-21 15:17:56 +01:00
|
|
|
'groups.name ILIKE :search OR ' \
|
|
|
|
'plans.base_name ILIKE :search', search: "%#{params[:search]}%")
|
|
|
|
end
|
|
|
|
|
2023-08-30 11:56:24 +02:00
|
|
|
filter = params[:filter].presence_in(%w[inactive_for_3_years not_confirmed not_validated]) || nil
|
2019-12-20 15:47:42 +01:00
|
|
|
@query = @query.send(filter) if filter
|
|
|
|
|
2019-01-21 15:17:56 +01:00
|
|
|
@query
|
|
|
|
end
|
|
|
|
|
2022-10-24 17:39:16 +02:00
|
|
|
def search(current_user, query, subscription)
|
2022-12-30 17:52:28 +01:00
|
|
|
members = User.includes(:profile, :statistic_profile, invoicing_profile: [:address])
|
2019-01-21 15:17:56 +01:00
|
|
|
.joins(:profile,
|
2019-06-04 16:50:23 +02:00
|
|
|
:statistic_profile,
|
2019-01-21 15:17:56 +01:00
|
|
|
:roles,
|
2019-06-04 16:50:23 +02:00
|
|
|
'LEFT JOIN "subscriptions" ON "subscriptions"."statistic_profile_id" = "statistic_profiles"."id" AND ' \
|
2019-01-21 15:17:56 +01:00
|
|
|
'"subscriptions"."created_at" = ( ' \
|
2022-10-24 17:39:16 +02:00
|
|
|
'SELECT max("created_at") ' \
|
|
|
|
'FROM "subscriptions" ' \
|
|
|
|
'WHERE "statistic_profile_id" = "statistic_profiles"."id")')
|
2022-05-30 13:59:01 +02:00
|
|
|
.where("users.is_active = 'true'")
|
2019-03-11 12:47:46 +01:00
|
|
|
.limit(50)
|
2022-10-24 17:39:16 +02:00
|
|
|
query.downcase.split.each do |word|
|
2022-09-26 13:41:50 -03:00
|
|
|
members = members.where('lower(f_unaccent(users.username)) ~ :search OR ' \
|
|
|
|
'lower(f_unaccent(profiles.first_name)) ~ :search OR ' \
|
2019-01-21 15:17:56 +01:00
|
|
|
'lower(f_unaccent(profiles.last_name)) ~ :search',
|
|
|
|
search: word)
|
|
|
|
end
|
|
|
|
|
|
|
|
if current_user.member?
|
|
|
|
# non-admin can only retrieve users with "public profiles"
|
|
|
|
members = members.where("users.is_allow_contact = 'true'")
|
|
|
|
elsif subscription == 'true'
|
|
|
|
# only admins have the ability to filter by subscription
|
2022-10-24 17:39:16 +02:00
|
|
|
members = members.where('subscriptions.id IS NOT NULL AND subscriptions.expiration_date >= :now', now: Time.zone.today.to_s)
|
2019-01-21 15:17:56 +01:00
|
|
|
elsif subscription == 'false'
|
2022-10-24 17:39:16 +02:00
|
|
|
members = members.where('subscriptions.id IS NULL OR subscriptions.expiration_date < :now', now: Time.zone.today.to_s)
|
2019-01-21 15:17:56 +01:00
|
|
|
end
|
|
|
|
|
2022-06-29 13:39:41 +02:00
|
|
|
members.to_a.filter(&:valid?)
|
2019-01-21 15:17:56 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def list_order(params)
|
|
|
|
direction = (params[:order_by][0] == '-' ? 'DESC' : 'ASC')
|
|
|
|
order_key = (params[:order_by][0] == '-' ? params[:order_by][1, params[:order_by].size] : params[:order_by])
|
2019-06-05 12:11:51 +02:00
|
|
|
limit = params[:size]
|
2020-05-13 10:59:16 +02:00
|
|
|
offset = ((params[:page]&.to_i || 1) - 1) * (params[:size]&.to_i || 1)
|
2019-01-21 15:17:56 +01:00
|
|
|
|
|
|
|
order_key = case order_key
|
2022-07-14 16:11:44 -03:00
|
|
|
when 'username'
|
2022-07-21 13:21:06 -03:00
|
|
|
'users.username'
|
2019-01-21 15:17:56 +01:00
|
|
|
when 'last_name'
|
|
|
|
'profiles.last_name'
|
|
|
|
when 'first_name'
|
|
|
|
'profiles.first_name'
|
|
|
|
when 'email'
|
|
|
|
'users.email'
|
|
|
|
when 'phone'
|
|
|
|
'profiles.phone'
|
|
|
|
when 'group'
|
|
|
|
'groups.name'
|
|
|
|
when 'plan'
|
|
|
|
'plans.base_name'
|
|
|
|
else
|
|
|
|
'users.id'
|
|
|
|
end
|
|
|
|
|
2023-03-28 14:01:05 +02:00
|
|
|
Arel.sql("#{order_key} #{direction}, users.id ASC LIMIT #{limit} OFFSET #{offset}")
|
2019-01-21 15:17:56 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|