-
Notifications
You must be signed in to change notification settings - Fork 6
/
acc_monitor.rb
233 lines (192 loc) · 10.2 KB
/
acc_monitor.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
$LOAD_PATH << '..'
require 'musikbot'
require 'httparty'
module ACCMonitor
def self.run
@mb = MusikBot::Session.new(inspect)
@getter = HTTParty
normal_header = "{| class='wikitable sortable'\n! Username\n! Total actions\n! style='min-width:100px' " \
"| Last action\n! Reason granted\n! style='min-width:85px' | Rights log\n|-\n"
@acc_markup = "==Account creation team==\n{| class='wikitable sortable'\n! Username\n! Total actions\n! style='min-width:100px' " \
"| Last ACC action\n! Reason granted\n! style='min-width:85px' | Rights log\n|-\n"
@educators_markup = "==Education program==\n{| class='wikitable sortable'\n! Username\n! EP rights\n! Total actions\n" \
"! style='min-width:100px' | Last action\n! Reason granted\n! style='min-width:85px' | Rights log\n|-\n"
@event_coordinators_markup = "==Event coordinators==\n{| class='wikitable sortable'\n! Username\n! Event date\n! Total actions\n" \
"! style='min-width:100px' | Last action\n! Reason granted\n! style='min-width:85px' | Rights log\n|-\n"
@other_users_markup = "==Other==\n" + normal_header
@user_count = 0
process_accounts
@coordinator_count = 0
process_event_coordinators
issue_report
rescue => e
@mb.report_error('Fatal error', e)
end
def self.process_accounts
account_creators.each_with_index do |account_creator, index|
username = account_creator['user_name']
puts "Checking #{account_creator}, index #{index} of #{account_creators.to_a.length}"
next if whitelisted_users.include?(username)
# queries
user_actions = logged_actions(username).reject { |la| la['log_title'] == username }
rights_log = rights_changes(username).select { |rc| rc['log_params'].scan(/accountcreator/).length == 1 }
last_action = @mb.parse_date(user_actions.first['log_timestamp']) rescue nil
permissions = user_groups(username).collect { |ug| ug['ug_group'] }
acc_info = acc_stats(username)
if acc_info
acc_last_action = @mb.parse_date(acc_info['lastactive']) rescue nil
acc_active = acc_last_action > @mb.today - num_days rescue nil
end
next if acc_active || (last_action && last_action > @mb.today - num_days)
# next unless acc_inactive || last_action.nil? || (last_action < @mb.today - num_days && acc_info.nil?)
puts ' meets inactivity threshold'
@user_count += 1
user = {
username: username,
rights_log: rights_log.reverse,
num_actions: user_actions.length,
last_action: last_action ? last_action.strftime('%Y %B %-d') : '-'
}
if (ep_groups & permissions).any?
user[:user_groups] = permissions
@educators_markup += ep_entry(user)
elsif acc_info.present?
user[:acc_type] = acc_info['status']
user[:acc_last_action] = acc_last_action ? acc_last_action.strftime('%Y %B %-d') : '-'
@acc_markup += acc_entry(user)
elsif acc_info.nil?
@other_users_markup += normal_entry(user)
end
end
end
def self.process_event_coordinators
usernames = []
event_coordinators.each do |user|
next if user[:event_date] > @mb.today || whitelisted_users.include?(user[:username])
usernames << user[:username]
permissions = user_groups(user[:username]).collect { |ug| ug['ug_group'] }
if permissions.include?('accountcreator') || ['Example', 'Test', 'Test user'].include?(user[:username])
user[:num_actions] = logged_actions(user[:username]).reject { |la| la['log_title'] == user[:username] }.length
user[:last_action] = @mb.parse_date(user_actions.first['log_timestamp']).strftime('%Y %B %-d') rescue '-'
user[:rights_log] = rights_changes(user[:username]).select { |rc| rc['log_params'].scan(/accountcreator/).length == 1 }.reverse
@coordinator_count += 1
@event_coordinators_markup += ec_entry(user)
end
@event_coordinators_text.gsub!("\n#{@event_coordinator_entries[user[:index]]}", '')
end
edit_page('User:MusikBot/ACCMonitor/Event coordinators', @event_coordinators_text, "Removing event coordinators: #{usernames.join(', ')}")
end
def self.issue_report
total = @user_count + @coordinator_count
percentage = ((total.to_f / account_creators.to_a.length.to_f) * 100).round
content = "<div style='font-size:24px'>Inactive account creators as of #{@mb.today.strftime('%-d %B %Y')}</div>\n" \
"'''#{total}''' out of #{account_creators.to_a.length} (#{percentage}%) account creators eligible for revocation\n\n" \
"'''{{User:MusikBot/ACCMonitor/Count}}''' users with no account creation activity in the past {{User:MusikBot/ACCMonitor/Offset}} days\n\n" \
"'''{{User:MusikBot/ACCMonitor/Coordinator count}}''' event coordinators with expired account creator privileges\n\n" \
"<small>''NOTE: Total actions excludes creation of their own account. " \
"Rights log only shows entries where {{mono|accountcreator}} was granted or revoked.''\n\n" \
"''If an account creator is inactive but not eligible for revocation of the right (such as an alternate account), they can be added to the [[User:MusikBot/ACCMonitor/Whitelist|whitelist]].''</small>\n\n"
content += @acc_markup.chomp("|-\n") + "|}\n\n"
content += @educators_markup.chomp("|-\n") + "|}\n\n"
content += @event_coordinators_markup.chomp("|-\n") + "|}\n\n"
content += @other_users_markup.chomp("|-\n") + "|}\n\n"
edit_page('User:MusikBot/ACCMonitor/Tracking', content, "Reporting account creation inactivity of #{@user_count} users")
edit_page('User:MusikBot/ACCMonitor/Count', @user_count.to_s, "Reporting #{@user_count} inactive account creators")
edit_page('User:MusikBot/ACCMonitor/Coordinator count', @coordinator_count.to_s, "Reporting #{@coordinator_count} event coordinators with expired account creator privileges")
end
def self.normal_entry(user)
most_recent_reason = user[:rights_log].first['log_comment'] rescue ''
rights_log_content = rights_log_markup(user[:username], user[:rights_log])
"| {{user-multi|user=#{user[:username]}|t|cr}}\n| #{user[:num_actions]}\n| #{user[:last_action]}\n| #{most_recent_reason}\n| #{rights_log_content}\n|-\n"
end
def self.acc_entry(user)
most_recent_reason = user[:rights_log].first['log_comment'] rescue ''
rights_log_content = rights_log_markup(user[:username], user[:rights_log])
"| {{user-multi|user=#{user[:username]}|t|cr}}\n| #{user[:num_actions]}\n| #{user[:acc_last_action]}\n| #{most_recent_reason}\n| #{rights_log_content}\n|-\n"
end
def self.ep_entry(user)
most_recent_reason = user[:rights_log].first['log_comment'] rescue ''
rights_log_content = rights_log_markup(user[:username], user[:rights_log])
ep_rights = user[:user_groups].select { |ug| ep_groups.include?(ug) }
"| {{user-multi|user=#{user[:username]}|t|cr}}\n| #{ep_rights.join(', ')}\n| #{user[:num_actions]}\n| #{user[:last_action]}\n| #{most_recent_reason}\n| #{rights_log_content}\n|-\n"
end
def self.ec_entry(user)
most_recent_reason = user[:rights_log].first['log_comment'] rescue ''
rights_log_content = rights_log_markup(user[:username], user[:rights_log])
"| {{user-multi|user=#{user[:username]}|t|cr}}\n| #{user[:event_date]}\n| #{user[:num_actions]}\n| #{user[:last_action]}\n| #{most_recent_reason}\n| #{rights_log_content}\n|-\n"
end
def self.rights_log_markup(username, rights_log)
markup = "{{collapse top|bg=transparent|bg2=transparent|border=0|border2=transparent|padding=0|title={{/Log link|#{username}}}}}\n"
rights_log.each do |entry|
date = @mb.parse_date(entry['log_timestamp']).strftime('%-d %B %Y')
userlinks = "{{u|#{entry['log_user_text']}}}"
granted = entry['log_params'] =~ /oldgroups.*accountcreator.*newgroups/ ? 'Revoked' : 'Granted'
markup += "* #{date} - #{granted} by #{userlinks} - ''#{entry['log_comment']}''\n"
end
markup += '{{collapse bottom}}'
end
def self.ep_groups
%w(epcampus epinstructor eponline epcoordinator)
end
def self.event_coordinators
return @event_coordinators if @event_coordinators
@event_coordinator_entries = event_coordinators_text.split(/\=\=\s*Event\s+coordinators\s*\=\=/)[1].split(/\n/).drop(2)
@event_coordinators = []
@event_coordinator_entries.each_with_index do |entry, index|
matches = entry.scan(/{{\s*no\s*ping\s*\|(.*?)}}(.*)/i).flatten
@event_coordinators << {
username: matches[0].strip,
event_date: @mb.parse_date(matches[1]),
index: index
}
end
@event_coordinators
rescue => e
@mb.report_error('Unable to parse event coordinators page', e)
end
# API related
def self.event_coordinators_text
@event_coordinators_text ||= @mb.get('User:MusikBot/ACCMonitor/Event coordinators')
end
def self.whitelisted_users
@whitelisted_users ||= @mb.get('User:MusikBot/ACCMonitor/Whitelist').split(/^\*/).drop(1).map { |u| u.chomp("\n") }
end
def self.num_days
@num_days ||= @mb.get('User:MusikBot/ACCMonitor/Offset').to_i
end
def self.edit_page(page, content, summary)
@mb.edit(page,
content: content,
summary: summary
)
end
private
# Replication database related
def self.account_creators
return @account_creators if @account_creators
@mb.repl_client.query('SELECT user_name FROM enwiki_p.user JOIN user_groups ' \
"WHERE ug_group = 'accountcreator' AND ug_user = user_id;")
end
def self.user_groups(username)
@mb.repl_client.query('SELECT ug_group FROM enwiki_p.user_groups ' \
"JOIN user WHERE user_name = '#{username}' AND ug_user = user_id;")
end
def self.rights_changes(username)
@mb.repl_client.query('SELECT log_timestamp, log_comment, log_user_text, log_params ' \
"FROM logging_logindex WHERE log_title = '#{username}' AND log_type = 'rights';")
end
def self.logged_actions(username)
@mb.repl_client.query('SELECT log_timestamp, log_title FROM logging_userindex ' \
"WHERE log_user_text = '#{username}' AND log_type = 'newusers' " \
'ORDER BY log_timestamp DESC;')
end
# HTTParty
def self.acc_stats(username)
res = @getter.get('http://accounts.wmflabs.org/api.php', query: {
action: 'stats',
user: username
})['api']['user']
res['missing'] ? nil : res
end
end
ACCMonitor.run