2014-03-05 20:52:20 +08:00
|
|
|
class Badge < ActiveRecord::Base
|
2014-07-03 15:29:44 +08:00
|
|
|
# badge ids
|
|
|
|
Welcome = 5
|
|
|
|
NicePost = 6
|
|
|
|
GoodPost = 7
|
|
|
|
GreatPost = 8
|
|
|
|
Autobiographer = 9
|
2014-07-07 15:55:25 +08:00
|
|
|
Editor = 10
|
2014-07-09 10:17:39 +08:00
|
|
|
FirstLike = 11
|
|
|
|
FirstShare = 12
|
|
|
|
FirstFlag = 13
|
2014-07-10 09:18:02 +08:00
|
|
|
FirstLink = 14
|
2014-07-11 12:17:01 +08:00
|
|
|
FirstQuote = 15
|
2014-07-16 14:26:22 +08:00
|
|
|
ReadGuidelines = 16
|
2014-07-15 13:16:41 +08:00
|
|
|
Reader = 17
|
2014-09-11 10:36:37 +08:00
|
|
|
NiceTopic = 18
|
|
|
|
GoodTopic = 19
|
|
|
|
GreatTopic = 20
|
2014-09-11 11:10:37 +08:00
|
|
|
NiceShare = 21
|
|
|
|
GoodShare = 22
|
|
|
|
GreatShare = 23
|
2015-02-19 02:30:07 +08:00
|
|
|
OneYearAnniversary = 24
|
2016-03-17 01:48:14 +08:00
|
|
|
|
2015-05-15 10:04:41 +08:00
|
|
|
Promoter = 25
|
|
|
|
Campaigner = 26
|
|
|
|
Champion = 27
|
2016-03-17 01:48:14 +08:00
|
|
|
|
2015-08-28 00:52:31 +08:00
|
|
|
PopularLink = 28
|
|
|
|
HotLink = 29
|
|
|
|
FamousLink = 30
|
2016-03-17 01:48:14 +08:00
|
|
|
|
2016-03-17 01:03:17 +08:00
|
|
|
Appreciated = 36
|
|
|
|
Respected = 37
|
2016-03-16 02:14:10 +08:00
|
|
|
Admired = 31
|
2016-03-17 01:48:14 +08:00
|
|
|
|
2016-03-17 00:31:26 +08:00
|
|
|
OutOfLove = 33
|
2016-03-17 23:42:22 +08:00
|
|
|
HigherLove = 34
|
2016-03-17 00:31:26 +08:00
|
|
|
CrazyInLove = 35
|
2015-05-15 10:04:41 +08:00
|
|
|
|
2016-03-17 01:48:14 +08:00
|
|
|
ThankYou = 38
|
|
|
|
GivesBack = 32
|
|
|
|
Empathetic = 39
|
|
|
|
|
2014-07-03 15:29:44 +08:00
|
|
|
# other consts
|
|
|
|
AutobiographerMinBioLength = 10
|
|
|
|
|
2014-07-24 16:28:09 +08:00
|
|
|
def self.trigger_hash
|
|
|
|
Hash[*(
|
|
|
|
Badge::Trigger.constants.map{|k|
|
|
|
|
[k.to_s.underscore, Badge::Trigger.const_get(k)]
|
|
|
|
}.flatten
|
|
|
|
)]
|
|
|
|
end
|
|
|
|
|
2014-07-22 10:46:31 +08:00
|
|
|
module Trigger
|
2014-07-24 16:28:09 +08:00
|
|
|
None = 0
|
2014-07-22 09:11:30 +08:00
|
|
|
PostAction = 1
|
2014-07-23 09:42:24 +08:00
|
|
|
PostRevision = 2
|
|
|
|
TrustLevelChange = 4
|
|
|
|
UserChange = 8
|
FEATURE: Badge query validation, preview results, and EXPLAIN
Upon saving a badge or requesting a badge result preview,
BadgeGranter.contract_checks! will examine the provided badge SQL for
some contractual obligations - namely, the returned columns and use of
trigger parameters.
Saving the badge is wrapped in a transaction to make this easier, by
raising ActiveRecord::Rollback on a detected violation.
On the client, a modal view is added for the badge query sample run
results, named admin-badge-preview.
The preview action is moved up to the route.
The save action, on failure, triggers a 'saveError' action (also in the
route).
The preview action gains a new parameter, 'explain', which will give the
output of an EXPLAIN query for the badge sql, which can be used by forum
admins to estimate the cost of their badge queries.
The preview link is replaced by two links, one which omits (false) and
includes (true) the EXPLAIN query.
The Badge.save() method is amended to propogate errors.
Badge::Trigger gets some utility methods for use in the
BadgeGranter.contract_checks! method.
Additionally, extra checks outside of BadgeGranter.contract_checks! are
added in the preview() method, to cover cases of null granted_at
columns.
An uninitialized variable path is removed in the backfill() method.
TODO - it would be nice to be able to get the actual names of all
columns the provided query returns, so we could give more errors
2014-08-26 06:17:29 +08:00
|
|
|
|
|
|
|
def self.is_none?(trigger)
|
2014-09-01 10:36:31 +08:00
|
|
|
[None].include? trigger
|
FEATURE: Badge query validation, preview results, and EXPLAIN
Upon saving a badge or requesting a badge result preview,
BadgeGranter.contract_checks! will examine the provided badge SQL for
some contractual obligations - namely, the returned columns and use of
trigger parameters.
Saving the badge is wrapped in a transaction to make this easier, by
raising ActiveRecord::Rollback on a detected violation.
On the client, a modal view is added for the badge query sample run
results, named admin-badge-preview.
The preview action is moved up to the route.
The save action, on failure, triggers a 'saveError' action (also in the
route).
The preview action gains a new parameter, 'explain', which will give the
output of an EXPLAIN query for the badge sql, which can be used by forum
admins to estimate the cost of their badge queries.
The preview link is replaced by two links, one which omits (false) and
includes (true) the EXPLAIN query.
The Badge.save() method is amended to propogate errors.
Badge::Trigger gets some utility methods for use in the
BadgeGranter.contract_checks! method.
Additionally, extra checks outside of BadgeGranter.contract_checks! are
added in the preview() method, to cover cases of null granted_at
columns.
An uninitialized variable path is removed in the backfill() method.
TODO - it would be nice to be able to get the actual names of all
columns the provided query returns, so we could give more errors
2014-08-26 06:17:29 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.uses_user_ids?(trigger)
|
2014-09-01 10:36:31 +08:00
|
|
|
[TrustLevelChange, UserChange].include? trigger
|
FEATURE: Badge query validation, preview results, and EXPLAIN
Upon saving a badge or requesting a badge result preview,
BadgeGranter.contract_checks! will examine the provided badge SQL for
some contractual obligations - namely, the returned columns and use of
trigger parameters.
Saving the badge is wrapped in a transaction to make this easier, by
raising ActiveRecord::Rollback on a detected violation.
On the client, a modal view is added for the badge query sample run
results, named admin-badge-preview.
The preview action is moved up to the route.
The save action, on failure, triggers a 'saveError' action (also in the
route).
The preview action gains a new parameter, 'explain', which will give the
output of an EXPLAIN query for the badge sql, which can be used by forum
admins to estimate the cost of their badge queries.
The preview link is replaced by two links, one which omits (false) and
includes (true) the EXPLAIN query.
The Badge.save() method is amended to propogate errors.
Badge::Trigger gets some utility methods for use in the
BadgeGranter.contract_checks! method.
Additionally, extra checks outside of BadgeGranter.contract_checks! are
added in the preview() method, to cover cases of null granted_at
columns.
An uninitialized variable path is removed in the backfill() method.
TODO - it would be nice to be able to get the actual names of all
columns the provided query returns, so we could give more errors
2014-08-26 06:17:29 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.uses_post_ids?(trigger)
|
2014-09-01 10:36:31 +08:00
|
|
|
[PostAction, PostRevision].include? trigger
|
FEATURE: Badge query validation, preview results, and EXPLAIN
Upon saving a badge or requesting a badge result preview,
BadgeGranter.contract_checks! will examine the provided badge SQL for
some contractual obligations - namely, the returned columns and use of
trigger parameters.
Saving the badge is wrapped in a transaction to make this easier, by
raising ActiveRecord::Rollback on a detected violation.
On the client, a modal view is added for the badge query sample run
results, named admin-badge-preview.
The preview action is moved up to the route.
The save action, on failure, triggers a 'saveError' action (also in the
route).
The preview action gains a new parameter, 'explain', which will give the
output of an EXPLAIN query for the badge sql, which can be used by forum
admins to estimate the cost of their badge queries.
The preview link is replaced by two links, one which omits (false) and
includes (true) the EXPLAIN query.
The Badge.save() method is amended to propogate errors.
Badge::Trigger gets some utility methods for use in the
BadgeGranter.contract_checks! method.
Additionally, extra checks outside of BadgeGranter.contract_checks! are
added in the preview() method, to cover cases of null granted_at
columns.
An uninitialized variable path is removed in the backfill() method.
TODO - it would be nice to be able to get the actual names of all
columns the provided query returns, so we could give more errors
2014-08-26 06:17:29 +08:00
|
|
|
end
|
2014-07-22 09:11:30 +08:00
|
|
|
end
|
2014-07-03 15:29:44 +08:00
|
|
|
|
|
|
|
module Queries
|
2014-07-11 15:32:29 +08:00
|
|
|
|
2014-07-15 13:16:41 +08:00
|
|
|
Reader = <<SQL
|
2014-08-27 16:02:13 +08:00
|
|
|
SELECT id user_id, current_timestamp granted_at
|
2014-07-15 13:16:41 +08:00
|
|
|
FROM users
|
|
|
|
WHERE id IN
|
|
|
|
(
|
|
|
|
SELECT pt.user_id
|
|
|
|
FROM post_timings pt
|
|
|
|
JOIN badge_posts b ON b.post_number = pt.post_number AND
|
|
|
|
b.topic_id = pt.topic_id
|
|
|
|
JOIN topics t ON t.id = pt.topic_id
|
|
|
|
LEFT JOIN user_badges ub ON ub.badge_id = 17 AND ub.user_id = pt.user_id
|
2014-07-18 13:57:03 +08:00
|
|
|
WHERE ub.id IS NULL AND t.posts_count > 100
|
2014-07-15 13:16:41 +08:00
|
|
|
GROUP BY pt.user_id, pt.topic_id, t.posts_count
|
2014-09-16 05:07:47 +08:00
|
|
|
HAVING count(*) >= t.posts_count
|
2014-07-15 13:16:41 +08:00
|
|
|
)
|
|
|
|
SQL
|
|
|
|
|
2014-07-16 14:26:22 +08:00
|
|
|
ReadGuidelines = <<SQL
|
2014-08-27 16:02:13 +08:00
|
|
|
SELECT user_id, read_faq granted_at
|
2014-07-11 15:35:34 +08:00
|
|
|
FROM user_stats
|
2014-08-27 16:02:13 +08:00
|
|
|
WHERE read_faq IS NOT NULL AND (user_id IN (:user_ids) OR :backfill)
|
2014-07-11 15:32:29 +08:00
|
|
|
SQL
|
|
|
|
|
2014-07-11 12:17:01 +08:00
|
|
|
FirstQuote = <<SQL
|
2014-07-15 15:47:24 +08:00
|
|
|
SELECT ids.user_id, q.post_id, q.created_at granted_at
|
2014-07-11 12:17:01 +08:00
|
|
|
FROM
|
|
|
|
(
|
2014-07-15 15:47:24 +08:00
|
|
|
SELECT p1.user_id, MIN(q1.id) id
|
|
|
|
FROM quoted_posts q1
|
|
|
|
JOIN badge_posts p1 ON p1.id = q1.post_id
|
|
|
|
JOIN badge_posts p2 ON p2.id = q1.quoted_post_id
|
2014-08-08 08:00:10 +08:00
|
|
|
WHERE (:backfill OR ( p1.id IN (:post_ids) ))
|
2014-07-15 15:47:24 +08:00
|
|
|
GROUP BY p1.user_id
|
2014-07-11 12:17:01 +08:00
|
|
|
) ids
|
2014-07-15 15:47:24 +08:00
|
|
|
JOIN quoted_posts q ON q.id = ids.id
|
2014-07-11 12:17:01 +08:00
|
|
|
SQL
|
2014-07-09 10:17:39 +08:00
|
|
|
|
2014-07-10 09:18:02 +08:00
|
|
|
FirstLink = <<SQL
|
|
|
|
SELECT l.user_id, l.post_id, l.created_at granted_at
|
|
|
|
FROM
|
|
|
|
(
|
|
|
|
SELECT MIN(l1.id) id
|
|
|
|
FROM topic_links l1
|
|
|
|
JOIN badge_posts p1 ON p1.id = l1.post_id
|
|
|
|
JOIN badge_posts p2 ON p2.id = l1.link_post_id
|
2014-08-08 08:00:10 +08:00
|
|
|
WHERE NOT reflection AND p1.topic_id <> p2.topic_id AND not quote AND
|
|
|
|
(:backfill OR ( p1.id in (:post_ids) ))
|
2014-07-10 09:18:02 +08:00
|
|
|
GROUP BY l1.user_id
|
|
|
|
) ids
|
|
|
|
JOIN topic_links l ON l.id = ids.id
|
|
|
|
SQL
|
|
|
|
|
2014-07-09 10:17:39 +08:00
|
|
|
FirstShare = <<SQL
|
2014-08-04 14:43:57 +08:00
|
|
|
SELECT views.user_id, i2.post_id, i2.created_at granted_at
|
2014-07-09 10:17:39 +08:00
|
|
|
FROM
|
|
|
|
(
|
|
|
|
SELECT i.user_id, MIN(i.id) i_id
|
|
|
|
FROM incoming_links i
|
2014-08-04 14:43:57 +08:00
|
|
|
JOIN badge_posts p on p.id = i.post_id
|
2014-07-10 09:18:02 +08:00
|
|
|
WHERE i.user_id IS NOT NULL
|
2014-07-09 10:17:39 +08:00
|
|
|
GROUP BY i.user_id
|
|
|
|
) as views
|
|
|
|
JOIN incoming_links i2 ON i2.id = views.i_id
|
|
|
|
SQL
|
|
|
|
|
|
|
|
FirstFlag = <<SQL
|
2014-07-23 09:42:24 +08:00
|
|
|
SELECT pa1.user_id, pa1.created_at granted_at, pa1.post_id
|
|
|
|
FROM (
|
|
|
|
SELECT pa.user_id, min(pa.id) id
|
|
|
|
FROM post_actions pa
|
|
|
|
JOIN badge_posts p on p.id = pa.post_id
|
2014-08-08 08:00:10 +08:00
|
|
|
WHERE post_action_type_id IN (#{PostActionType.flag_types.values.join(",")}) AND
|
|
|
|
(:backfill OR pa.post_id IN (:post_ids) )
|
2014-07-23 09:42:24 +08:00
|
|
|
GROUP BY pa.user_id
|
|
|
|
) x
|
|
|
|
JOIN post_actions pa1 on pa1.id = x.id
|
2014-07-09 10:17:39 +08:00
|
|
|
SQL
|
|
|
|
|
|
|
|
FirstLike = <<SQL
|
2014-07-23 09:42:24 +08:00
|
|
|
SELECT pa1.user_id, pa1.created_at granted_at, pa1.post_id
|
|
|
|
FROM (
|
|
|
|
SELECT pa.user_id, min(pa.id) id
|
|
|
|
FROM post_actions pa
|
|
|
|
JOIN badge_posts p on p.id = pa.post_id
|
2014-08-08 08:00:10 +08:00
|
|
|
WHERE post_action_type_id = 2 AND
|
|
|
|
(:backfill OR pa.post_id IN (:post_ids) )
|
2014-07-23 09:42:24 +08:00
|
|
|
GROUP BY pa.user_id
|
|
|
|
) x
|
|
|
|
JOIN post_actions pa1 on pa1.id = x.id
|
2014-07-08 12:26:53 +08:00
|
|
|
SQL
|
2014-07-07 15:55:25 +08:00
|
|
|
|
2014-07-23 09:42:24 +08:00
|
|
|
# Incorrect, but good enough - (earlies post edited vs first edit)
|
2014-07-07 15:55:25 +08:00
|
|
|
Editor = <<SQL
|
|
|
|
SELECT p.user_id, min(p.id) post_id, min(p.created_at) granted_at
|
2014-07-10 09:18:02 +08:00
|
|
|
FROM badge_posts p
|
2014-08-08 08:00:10 +08:00
|
|
|
WHERE p.self_edits > 0 AND
|
|
|
|
(:backfill OR p.id IN (:post_ids) )
|
2014-07-07 15:55:25 +08:00
|
|
|
GROUP BY p.user_id
|
|
|
|
SQL
|
|
|
|
|
2014-07-03 15:29:44 +08:00
|
|
|
Welcome = <<SQL
|
|
|
|
SELECT p.user_id, min(post_id) post_id, min(pa.created_at) granted_at
|
|
|
|
FROM post_actions pa
|
2014-07-10 09:18:02 +08:00
|
|
|
JOIN badge_posts p on p.id = pa.post_id
|
2014-08-08 08:00:10 +08:00
|
|
|
WHERE post_action_type_id = 2 AND
|
|
|
|
(:backfill OR pa.post_id IN (:post_ids) )
|
2014-07-03 15:29:44 +08:00
|
|
|
GROUP BY p.user_id
|
|
|
|
SQL
|
|
|
|
|
|
|
|
Autobiographer = <<SQL
|
2014-08-27 16:02:13 +08:00
|
|
|
SELECT u.id user_id, current_timestamp granted_at
|
2014-07-03 15:29:44 +08:00
|
|
|
FROM users u
|
|
|
|
JOIN user_profiles up on u.id = up.user_id
|
|
|
|
WHERE bio_raw IS NOT NULL AND LENGTH(TRIM(bio_raw)) > #{Badge::AutobiographerMinBioLength} AND
|
2014-08-08 08:00:10 +08:00
|
|
|
uploaded_avatar_id IS NOT NULL AND
|
|
|
|
(:backfill OR u.id IN (:user_ids) )
|
2014-07-03 15:29:44 +08:00
|
|
|
SQL
|
|
|
|
|
2015-02-19 02:30:07 +08:00
|
|
|
# member for a year + has posted at least once during that year
|
|
|
|
OneYearAnniversary = <<-SQL
|
|
|
|
SELECT u.id AS user_id, MIN(u.created_at + interval '1 year') AS granted_at
|
|
|
|
FROM users u
|
|
|
|
JOIN posts p ON p.user_id = u.id
|
|
|
|
WHERE u.id > 0
|
|
|
|
AND u.active
|
|
|
|
AND NOT u.blocked
|
|
|
|
AND u.created_at + interval '1 year' < now()
|
|
|
|
AND p.deleted_at IS NULL
|
|
|
|
AND NOT p.hidden
|
|
|
|
AND p.created_at + interval '1 year' > now()
|
|
|
|
AND (:backfill OR u.id IN (:user_ids))
|
|
|
|
GROUP BY u.id
|
|
|
|
HAVING COUNT(p.id) > 0
|
|
|
|
SQL
|
|
|
|
|
2015-05-15 10:04:41 +08:00
|
|
|
def self.invite_badge(count,trust_level)
|
|
|
|
"
|
|
|
|
SELECT u.id user_id, current_timestamp granted_at
|
|
|
|
FROM users u
|
|
|
|
WHERE u.id IN (
|
|
|
|
SELECT invited_by_id
|
|
|
|
FROM invites i
|
|
|
|
JOIN users u2 ON u2.id = i.user_id
|
|
|
|
WHERE i.deleted_at IS NULL AND u2.active AND u2.trust_level >= #{trust_level.to_i} AND not u2.blocked
|
|
|
|
GROUP BY invited_by_id
|
2015-10-13 09:32:29 +08:00
|
|
|
HAVING COUNT(*) >= #{count.to_i}
|
2015-05-15 10:04:41 +08:00
|
|
|
) AND u.active AND NOT u.blocked AND u.id > 0 AND
|
|
|
|
(:backfill OR u.id IN (:user_ids) )
|
|
|
|
"
|
|
|
|
end
|
|
|
|
|
2014-09-11 10:36:37 +08:00
|
|
|
def self.like_badge(count, is_topic)
|
2014-07-03 15:29:44 +08:00
|
|
|
# we can do better with dates, but its hard work
|
|
|
|
"
|
2014-07-10 09:18:02 +08:00
|
|
|
SELECT p.user_id, p.id post_id, p.updated_at granted_at
|
|
|
|
FROM badge_posts p
|
2014-09-11 10:36:37 +08:00
|
|
|
WHERE #{is_topic ? "p.post_number = 1" : "p.post_number > 1" } AND p.like_count >= #{count.to_i} AND
|
2014-08-08 08:00:10 +08:00
|
|
|
(:backfill OR p.id IN (:post_ids) )
|
2014-07-03 15:29:44 +08:00
|
|
|
"
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.trust_level(level)
|
|
|
|
# we can do better with dates, but its hard work figuring this out historically
|
|
|
|
"
|
2014-08-27 16:02:13 +08:00
|
|
|
SELECT u.id user_id, current_timestamp granted_at FROM users u
|
2014-08-08 08:00:10 +08:00
|
|
|
WHERE trust_level >= #{level.to_i} AND (
|
|
|
|
:backfill OR u.id IN (:user_ids)
|
|
|
|
)
|
2014-07-03 15:29:44 +08:00
|
|
|
"
|
|
|
|
end
|
2014-09-11 11:10:37 +08:00
|
|
|
|
|
|
|
def self.sharing_badge(count)
|
|
|
|
<<SQL
|
2016-03-17 07:33:34 +08:00
|
|
|
SELECT views.user_id, i2.post_id, current_timestamp granted_at
|
2014-09-11 11:10:37 +08:00
|
|
|
FROM
|
|
|
|
(
|
|
|
|
SELECT i.user_id, MIN(i.id) i_id
|
|
|
|
FROM incoming_links i
|
|
|
|
JOIN badge_posts p on p.id = i.post_id
|
|
|
|
WHERE i.user_id IS NOT NULL
|
|
|
|
GROUP BY i.user_id,i.post_id
|
|
|
|
HAVING COUNT(*) > #{count}
|
|
|
|
) as views
|
|
|
|
JOIN incoming_links i2 ON i2.id = views.i_id
|
|
|
|
SQL
|
|
|
|
end
|
2015-08-28 00:52:31 +08:00
|
|
|
|
|
|
|
def self.linking_badge(count)
|
|
|
|
<<-SQL
|
2016-03-17 07:33:34 +08:00
|
|
|
SELECT tl.user_id, post_id, current_timestamp granted_at
|
2015-08-28 00:52:31 +08:00
|
|
|
FROM topic_links tl
|
|
|
|
JOIN posts p ON p.id = post_id AND p.deleted_at IS NULL
|
|
|
|
JOIN topics t ON t.id = p.topic_id AND t.deleted_at IS NULL AND t.archetype <> 'private_message'
|
|
|
|
WHERE NOT tl.internal
|
|
|
|
AND tl.clicks >= #{count}
|
|
|
|
GROUP BY tl.user_id, tl.post_id
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
|
2016-03-17 01:03:17 +08:00
|
|
|
def self.liked_posts(post_count, like_count)
|
|
|
|
<<-SQL
|
|
|
|
SELECT p.user_id, current_timestamp AS granted_at
|
|
|
|
FROM posts AS p
|
|
|
|
WHERE p.like_count >= #{like_count}
|
|
|
|
AND (:backfill OR p.user_id IN (:user_ids))
|
|
|
|
GROUP BY p.user_id
|
|
|
|
HAVING count(*) > #{post_count}
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
|
2016-03-17 00:31:26 +08:00
|
|
|
def self.like_rate_limit(count)
|
|
|
|
<<-SQL
|
2016-03-18 02:41:00 +08:00
|
|
|
SELECT gdl.user_id, current_timestamp AS granted_at
|
|
|
|
FROM given_daily_likes AS gdl
|
|
|
|
WHERE gdl.limit_reached
|
|
|
|
AND (:backfill OR gdl.user_id IN (:user_ids))
|
|
|
|
GROUP BY gdl.user_id
|
2016-03-17 00:31:26 +08:00
|
|
|
HAVING COUNT(*) >= #{count}
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
|
2016-03-22 22:51:42 +08:00
|
|
|
def self.liked_back(likes_received, likes_given)
|
2016-03-17 01:48:14 +08:00
|
|
|
<<-SQL
|
2016-03-22 22:51:42 +08:00
|
|
|
SELECT us.user_id, current_timestamp AS granted_at
|
|
|
|
FROM user_stats AS us
|
|
|
|
WHERE us.likes_received >= #{likes_received}
|
|
|
|
AND us.likes_given >= #{likes_given}
|
|
|
|
AND (:backfill OR us.user_id IN (:user_ids))
|
2016-03-17 01:48:14 +08:00
|
|
|
SQL
|
|
|
|
end
|
2014-07-03 15:29:44 +08:00
|
|
|
end
|
|
|
|
|
2014-03-05 20:52:20 +08:00
|
|
|
belongs_to :badge_type
|
2014-07-18 13:46:36 +08:00
|
|
|
belongs_to :badge_grouping
|
|
|
|
|
2014-03-21 20:26:06 +08:00
|
|
|
has_many :user_badges, dependent: :destroy
|
2014-03-05 20:52:20 +08:00
|
|
|
|
|
|
|
validates :name, presence: true, uniqueness: true
|
|
|
|
validates :badge_type, presence: true
|
2014-04-26 02:25:29 +08:00
|
|
|
validates :allow_title, inclusion: [true, false]
|
2014-05-24 10:33:46 +08:00
|
|
|
validates :multiple_grant, inclusion: [true, false]
|
2014-05-05 02:15:38 +08:00
|
|
|
|
2014-07-14 15:40:01 +08:00
|
|
|
scope :enabled, ->{ where(enabled: true) }
|
|
|
|
|
2014-08-06 08:51:39 +08:00
|
|
|
before_create :ensure_not_system
|
|
|
|
|
2014-07-24 16:28:09 +08:00
|
|
|
# fields that can not be edited on system badges
|
|
|
|
def self.protected_system_fields
|
2016-03-28 15:38:38 +08:00
|
|
|
[
|
|
|
|
:badge_type_id, :multiple_grant,
|
|
|
|
:target_posts, :show_posts, :query,
|
|
|
|
:trigger, :auto_revoke, :listable
|
|
|
|
]
|
2014-07-24 16:28:09 +08:00
|
|
|
end
|
|
|
|
|
2014-05-05 02:15:38 +08:00
|
|
|
def self.trust_level_badge_ids
|
|
|
|
(1..4).to_a
|
|
|
|
end
|
2014-05-16 08:34:06 +08:00
|
|
|
|
2014-07-03 15:29:44 +08:00
|
|
|
def self.like_badge_counts
|
|
|
|
@like_badge_counts ||= {
|
|
|
|
NicePost => 10,
|
|
|
|
GoodPost => 25,
|
2014-09-11 11:30:47 +08:00
|
|
|
GreatPost => 50,
|
|
|
|
NiceTopic => 10,
|
|
|
|
GoodTopic => 25,
|
|
|
|
GreatTopic => 50
|
2014-07-03 15:29:44 +08:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2014-05-16 08:34:06 +08:00
|
|
|
def reset_grant_count!
|
|
|
|
self.grant_count = UserBadge.where(badge_id: id).count
|
|
|
|
save!
|
|
|
|
end
|
|
|
|
|
2014-05-21 15:22:42 +08:00
|
|
|
def single_grant?
|
|
|
|
!self.multiple_grant?
|
|
|
|
end
|
|
|
|
|
2014-10-02 13:55:33 +08:00
|
|
|
def default_icon=(val)
|
|
|
|
self.icon ||= val
|
2014-10-07 09:01:06 +08:00
|
|
|
self.icon = val if self.icon = "fa-certificate"
|
2014-10-02 13:55:33 +08:00
|
|
|
end
|
|
|
|
|
2014-07-17 10:25:16 +08:00
|
|
|
def default_name=(val)
|
|
|
|
self.name ||= val
|
|
|
|
end
|
2014-07-17 14:10:44 +08:00
|
|
|
|
2014-07-30 06:46:46 +08:00
|
|
|
def default_allow_title=(val)
|
|
|
|
self.allow_title ||= val
|
|
|
|
end
|
|
|
|
|
2014-07-17 14:10:44 +08:00
|
|
|
def default_badge_grouping_id=(val)
|
2014-07-18 13:55:42 +08:00
|
|
|
# allow to correct orphans
|
|
|
|
if !self.badge_grouping_id || self.badge_grouping_id < 0
|
|
|
|
self.badge_grouping_id = val
|
|
|
|
end
|
2014-07-17 14:10:44 +08:00
|
|
|
end
|
2014-08-06 08:51:39 +08:00
|
|
|
|
2015-08-14 19:03:49 +08:00
|
|
|
def self.ensure_consistency!
|
2016-03-29 13:41:03 +08:00
|
|
|
exec_sql <<SQL
|
|
|
|
DELETE FROM user_badges
|
|
|
|
USING user_badges ub
|
|
|
|
LEFT JOIN users u ON u.id = ub.user_id
|
2016-03-29 20:54:18 +08:00
|
|
|
WHERE u.id IS NULL AND user_badges.id = ub.id
|
2016-03-29 13:41:03 +08:00
|
|
|
SQL
|
|
|
|
|
2015-08-14 19:03:49 +08:00
|
|
|
Badge.find_each(&:reset_grant_count!)
|
|
|
|
end
|
|
|
|
|
2015-09-24 04:52:43 +08:00
|
|
|
def display_name
|
|
|
|
if self.system?
|
2016-03-29 02:29:53 +08:00
|
|
|
key = "badges.#{i18n_name}.name"
|
2015-09-24 04:52:43 +08:00
|
|
|
I18n.t(key, default: self.name)
|
|
|
|
else
|
|
|
|
self.name
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def long_description
|
|
|
|
if self[:long_description].present?
|
|
|
|
self[:long_description]
|
|
|
|
else
|
2016-03-28 15:38:38 +08:00
|
|
|
key = "badges.#{i18n_name}.long_description"
|
2015-09-24 04:52:43 +08:00
|
|
|
I18n.t(key, default: '')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-03-28 15:38:38 +08:00
|
|
|
def long_description=(val)
|
|
|
|
if val != long_description
|
|
|
|
self[:long_description] = val
|
|
|
|
end
|
|
|
|
|
|
|
|
val
|
|
|
|
end
|
|
|
|
|
|
|
|
def description
|
|
|
|
if self[:description].present?
|
|
|
|
self[:description]
|
|
|
|
else
|
|
|
|
key = "badges.#{i18n_name}.description"
|
|
|
|
I18n.t(key, default: '')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def description=(val)
|
|
|
|
if val != description
|
|
|
|
self[:description] = val
|
|
|
|
end
|
|
|
|
|
|
|
|
val
|
|
|
|
end
|
|
|
|
|
|
|
|
|
2015-09-24 04:52:43 +08:00
|
|
|
def slug
|
|
|
|
Slug.for(self.display_name, '-')
|
|
|
|
end
|
|
|
|
|
2014-08-06 08:51:39 +08:00
|
|
|
protected
|
|
|
|
def ensure_not_system
|
|
|
|
unless id
|
|
|
|
self.id = [Badge.maximum(:id) + 1, 100].max
|
|
|
|
end
|
|
|
|
end
|
2015-09-24 04:52:43 +08:00
|
|
|
|
|
|
|
def i18n_name
|
|
|
|
self.name.downcase.gsub(' ', '_')
|
|
|
|
end
|
2014-03-05 20:52:20 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
# == Schema Information
|
|
|
|
#
|
|
|
|
# Table name: badges
|
|
|
|
#
|
2014-07-17 14:10:44 +08:00
|
|
|
# id :integer not null, primary key
|
2016-02-23 07:33:53 +08:00
|
|
|
# name :string not null
|
2014-07-17 14:10:44 +08:00
|
|
|
# description :text
|
|
|
|
# badge_type_id :integer not null
|
|
|
|
# grant_count :integer default(0), not null
|
2014-08-27 13:19:25 +08:00
|
|
|
# created_at :datetime not null
|
|
|
|
# updated_at :datetime not null
|
2014-07-17 14:10:44 +08:00
|
|
|
# allow_title :boolean default(FALSE), not null
|
|
|
|
# multiple_grant :boolean default(FALSE), not null
|
2016-02-23 07:33:53 +08:00
|
|
|
# icon :string default("fa-certificate")
|
2014-07-17 14:10:44 +08:00
|
|
|
# listable :boolean default(TRUE)
|
|
|
|
# target_posts :boolean default(FALSE)
|
|
|
|
# query :text
|
|
|
|
# enabled :boolean default(TRUE), not null
|
|
|
|
# auto_revoke :boolean default(TRUE), not null
|
2014-07-22 10:46:31 +08:00
|
|
|
# badge_grouping_id :integer default(5), not null
|
|
|
|
# trigger :integer
|
2014-07-24 16:28:09 +08:00
|
|
|
# show_posts :boolean default(FALSE), not null
|
2014-08-07 11:33:11 +08:00
|
|
|
# system :boolean default(FALSE), not null
|
2014-11-20 11:53:15 +08:00
|
|
|
# image :string(255)
|
2015-09-18 08:41:10 +08:00
|
|
|
# long_description :text
|
2014-03-05 20:52:20 +08:00
|
|
|
#
|
|
|
|
# Indexes
|
|
|
|
#
|
2016-02-23 07:33:53 +08:00
|
|
|
# index_badges_on_badge_type_id (badge_type_id)
|
|
|
|
# index_badges_on_name (name) UNIQUE
|
2014-03-05 20:52:20 +08:00
|
|
|
#
|