mirror of
https://github.com/ENSL/ensl.org.git
synced 2025-01-13 05:10:59 +00:00
db9d4c1564
* Logrotation for production * added contributors class method to Group * added contributor? method to User * added contributor flag to User-API * added Contributor-Tab to staff-page
341 lines
10 KiB
Ruby
Executable file
341 lines
10 KiB
Ruby
Executable file
# == Schema Information
|
|
#
|
|
# Table name: users
|
|
#
|
|
# id :integer not null, primary key
|
|
# username :string(255)
|
|
# password :string(255)
|
|
# firstname :string(255)
|
|
# lastname :string(255)
|
|
# email :string(255)
|
|
# steamid :string(255)
|
|
# team_id :integer
|
|
# lastvisit :datetime
|
|
# created_at :datetime
|
|
# updated_at :datetime
|
|
# lastip :string(255)
|
|
# country :string(255)
|
|
# birthdate :date
|
|
# time_zone :string(255)
|
|
# version :integer
|
|
# public_email :boolean default(FALSE), not null
|
|
#
|
|
|
|
|
|
require 'digest/md5'
|
|
require File.join(Rails.root, 'vendor', 'plugins', 'acts_as_versioned', 'lib', 'acts_as_versioned.rb')
|
|
|
|
class User < ActiveRecord::Base
|
|
include Extra
|
|
|
|
VERIFICATION_TIME = 604800
|
|
|
|
attr_protected :id, :created_at, :updated_at, :lastvisit, :lastip, :password, :version
|
|
attr_accessor :raw_password
|
|
|
|
belongs_to :team
|
|
has_one :profile, :dependent => :destroy
|
|
has_many :bans, :dependent => :destroy
|
|
has_many :articles, :dependent => :destroy
|
|
has_many :movies, :dependent => :destroy
|
|
has_many :servers, :dependent => :destroy
|
|
has_many :votes, :dependent => :destroy
|
|
has_many :gatherers, :dependent => :destroy
|
|
has_many :gathers, :through => :gatherers
|
|
has_many :groupers, :dependent => :destroy
|
|
has_many :posts, :dependent => :destroy
|
|
has_many :groups, :through => :groupers
|
|
has_many :shoutmsgs, :dependent => :destroy
|
|
has_many :issues, :foreign_key => "author_id", :dependent => :destroy
|
|
has_many :open_issues, :class_name => "Issue", :foreign_key => "assigned_id",
|
|
:conditions => ["issues.status = ?", Issue::STATUS_OPEN]
|
|
has_many :posted_comments, :dependent => :destroy, :class_name => "Comment"
|
|
has_many :comments, :class_name => "Comment", :as => :commentable, :order => "created_at ASC", :dependent => :destroy
|
|
has_many :teamers, :dependent => :destroy
|
|
has_many :active_teams, :through => :teamers, :source => "team",
|
|
:conditions => ["teamers.rank >= ? AND teams.active = ?", Teamer::RANK_MEMBER, true]
|
|
has_many :active_contesters, :through => :active_teams, :source => "contesters",
|
|
:conditions => {"contesters.active" => true}
|
|
has_many :active_contests, :through => :active_contesters, :source => "contest",
|
|
:conditions => ["contests.status != ?", Contest::STATUS_CLOSED]
|
|
has_many :past_teams, :through => :teamers, :source => "team", :group => "user_id, team_id"
|
|
has_many :matchers, :dependent => :destroy
|
|
has_many :matches, :through => :matchers
|
|
has_many :predictions, :dependent => :destroy
|
|
has_many :challenges_received, :through => :active_contesters, :source => "challenges_received"
|
|
has_many :challenges_sent, :through => :active_contesters, :source => "challenges_sent"
|
|
has_many :upcoming_team_matches, :through => :active_contesters, :source => "matches",
|
|
:conditions => "match_time > UTC_TIMESTAMP()"
|
|
has_many :upcoming_ref_matches, :class_name => "Match", :foreign_key => "referee_id",
|
|
:conditions => "match_time > UTC_TIMESTAMP()"
|
|
has_many :past_team_matches, :through => :active_contesters, :source => "matches",
|
|
:conditions => "match_time < UTC_TIMESTAMP()"
|
|
has_many :past_ref_matches, :class_name => "Match", :foreign_key => "referee_id",
|
|
:conditions => "match_time < UTC_TIMESTAMP()"
|
|
has_many :received_personal_messages, :class_name => "Message", :as => "recipient", :dependent => :destroy
|
|
has_many :sent_personal_messages, :class_name => "Message", :as => "sender", :dependent => :destroy
|
|
has_many :sent_team_messages, :through => :active_teams, :source => :sent_messages
|
|
has_many :match_teams, :through => :matchers, :source => :teams, :uniq => true
|
|
|
|
scope :active, :conditions => {:banned => false}
|
|
scope :with_age,
|
|
:select => "DATE_FORMAT(FROM_DAYS(TO_DAYS(NOW())-TO_DAYS(birthdate)), '%Y')+0 AS aged, COUNT(*) as num, username",
|
|
:group => "aged",
|
|
:having => "num > 8 AND aged > 0"
|
|
scope :country_stats,
|
|
:select => "country, COUNT(*) as num",
|
|
:conditions => "country is not null and country != '' and country != '--'",
|
|
:group => "country",
|
|
:having => "num > 15",
|
|
:order => "num DESC"
|
|
scope :posts_stats,
|
|
:select => "users.id, username, COUNT(posts.id) as num",
|
|
:joins => "LEFT JOIN posts ON posts.user_id = users.id",
|
|
:group => "users.id",
|
|
:order => "num DESC"
|
|
scope :banned,
|
|
:joins => "LEFT JOIN bans ON bans.user_id = users.id AND expiry > UTC_TIMESTAMP()",
|
|
:conditions => "bans.id IS NOT NULL"
|
|
scope :idle,
|
|
:conditions => ["lastvisit < ?", 30.minutes.ago.utc]
|
|
scope :lately,
|
|
:conditions => ["lastvisit > ?", 30.days.ago.utc]
|
|
|
|
before_validation :update_password
|
|
|
|
validates :username, uniqueness: true, length: {in: 2..20}, format: /\A[A-Za-z0-9_\-\+]{2,20}\Z/
|
|
validates :firstname, length: {in: 1..15}, allow_blank: true
|
|
validates :lastname, length: {in: 1..25}, allow_blank: true
|
|
validates :raw_password, presence: {on: :create}
|
|
validates :email, uniqueness: true, length: {maximum: 50}, format: /\A([^@\s]+)@((?:[-a-z0-9]+\.)+[a-z]{2,})\Z/i
|
|
validates :steamid, uniqueness: {allow_nil: true}, length: {maximum: 14}, presence: {on: :create}
|
|
validate :validate_steamid
|
|
validates :time_zone, length: {maximum: 100}, allow_blank: true
|
|
validates :public_email, inclusion: [true, false], allow_nil: true
|
|
validate :validate_team
|
|
|
|
before_create :init_variables
|
|
|
|
before_save :correct_steamid_universe
|
|
|
|
accepts_nested_attributes_for :profile
|
|
|
|
acts_as_versioned
|
|
non_versioned_columns << 'firstname'
|
|
non_versioned_columns << 'lastname'
|
|
non_versioned_columns << 'email'
|
|
non_versioned_columns << 'password'
|
|
non_versioned_columns << 'team_id'
|
|
non_versioned_columns << 'lastvisit'
|
|
non_versioned_columns << 'team_id'
|
|
non_versioned_columns << 'country'
|
|
non_versioned_columns << 'birthdate'
|
|
non_versioned_columns << 'time_zone'
|
|
non_versioned_columns << 'public_email'
|
|
non_versioned_columns << 'created_at'
|
|
|
|
def to_s
|
|
username
|
|
end
|
|
|
|
def email_s
|
|
email.gsub /@/, " (at) "
|
|
end
|
|
|
|
def country_s
|
|
country = ISO3166::Country[self.country]
|
|
country ? country.name : 'Earth'
|
|
end
|
|
|
|
def realname
|
|
if firstname and lastname
|
|
"#{firstname} #{lastname}"
|
|
elsif firstname
|
|
firstname
|
|
elsif lastname
|
|
lastname
|
|
else
|
|
""
|
|
end
|
|
end
|
|
|
|
def from
|
|
if profile.town && profile.town.length > 0
|
|
"#{profile.town}, #{country_s}"
|
|
else
|
|
"#{country_s}"
|
|
end
|
|
end
|
|
|
|
def age
|
|
return 0 unless birthdate
|
|
a = Date.today.year - birthdate.year
|
|
a-= 1 if Date.today < birthdate + a.years
|
|
a
|
|
end
|
|
|
|
def current_layout
|
|
profile.layout || 'default'
|
|
end
|
|
|
|
def joined
|
|
created_at.strftime("%d %b %y")
|
|
end
|
|
|
|
def banned? type = Ban::TYPE_SITE
|
|
Ban.first :conditions => ["expiry > UTC_TIMESTAMP() AND user_id = ? AND ban_type = ?", self.id, type]
|
|
end
|
|
|
|
def admin?
|
|
groups.exists? id: Group::ADMINS
|
|
end
|
|
|
|
def ref?
|
|
groups.exists? id: Group::REFEREES
|
|
end
|
|
|
|
def staff?
|
|
groups.exists? id: Group::STAFF
|
|
end
|
|
|
|
def caster?
|
|
groups.exists? id: Group::CASTERS
|
|
end
|
|
|
|
# might seem redundant but allows for later extensions like forum moderators
|
|
def moderator?
|
|
groups.exists? id: Group::GATHER_MODERATORS
|
|
end
|
|
|
|
def gather_moderator?
|
|
groups.exists? id: Group::GATHER_MODERATORS
|
|
end
|
|
|
|
def contributor?
|
|
groups.exists? id: Group::CONTRIBUTORS
|
|
end
|
|
|
|
def allowed_to_ban?
|
|
admin? or moderator?
|
|
end
|
|
|
|
def verified?
|
|
# created_at < DateTime.now.ago(VERIFICATION_TIME)
|
|
true
|
|
end
|
|
|
|
def has_access? group
|
|
admin? or groups.exists?(:id => group)
|
|
end
|
|
|
|
def new_messages
|
|
received_personal_messages.unread_by(self) + received_team_messages.unread_by(self)
|
|
end
|
|
|
|
def received_messages
|
|
received_personal_messages + received_team_messages
|
|
end
|
|
|
|
def received_team_messages
|
|
Message.where(recipient_id: team_id, recipient_type: 'Team' )
|
|
end
|
|
|
|
def sent_messages
|
|
sent_personal_messages + sent_team_messages
|
|
end
|
|
|
|
def upcoming_matches
|
|
upcoming_team_matches.ordered | upcoming_ref_matches.ordered
|
|
end
|
|
|
|
def past_matches
|
|
past_team_matches.unfinished.ordered | past_ref_matches.unfinished.ordered
|
|
end
|
|
|
|
def unread_issues
|
|
issues.unread_by(self)
|
|
end
|
|
|
|
def validate_steamid
|
|
errors.add :steamid unless
|
|
steamid.nil? ||
|
|
(m = steamid.match(/\A([01]):([01]):(\d{1,10})\Z/)) &&
|
|
(id = m[3].to_i) &&
|
|
id >= 1 && id <= 2147483647
|
|
end
|
|
|
|
def correct_steamid_universe
|
|
if steamid.present?
|
|
steamid[0] = "0"
|
|
end
|
|
end
|
|
|
|
def validate_team
|
|
if team and !active_teams.exists?({:id => team.id})
|
|
errors.add :team
|
|
end
|
|
end
|
|
|
|
def init_variables
|
|
self.public_email = false
|
|
self.time_zone = "Amsterdam"
|
|
end
|
|
|
|
def update_password
|
|
self.password = Digest::MD5.hexdigest(raw_password) if raw_password and raw_password.length > 0
|
|
end
|
|
|
|
def send_new_password
|
|
newpass = Verification.random_string 10
|
|
update_attribute :password, Digest::MD5.hexdigest(newpass)
|
|
Notifications.password(self, newpass).deliver
|
|
end
|
|
|
|
def can_play?
|
|
(gathers.count(:conditions => ["gathers.status > ?", Gather::STATE_RUNNING]) > 0) or created_at < 2.years.ago
|
|
end
|
|
|
|
def can_create? cuser
|
|
true
|
|
end
|
|
|
|
def can_update? cuser
|
|
cuser and (self == cuser or cuser.admin?)
|
|
end
|
|
|
|
def can_change_name? cuser
|
|
cuser and cuser.admin?
|
|
end
|
|
|
|
def can_destroy? cuser
|
|
cuser and cuser.admin?
|
|
end
|
|
|
|
def self.authenticate(username, password)
|
|
where("LOWER(username) = LOWER(?)", username).where(:password => Digest::MD5.hexdigest(password)).first
|
|
end
|
|
|
|
def self.get id
|
|
id ? find(id) : ""
|
|
end
|
|
|
|
def self.historic steamid
|
|
if u = User.find_by_sql(["SELECT * FROM user_versions WHERE steamid = ? ORDER BY updated_at", steamid]) and u.length > 0
|
|
User.find u[0]['user_id']
|
|
else
|
|
nil
|
|
end
|
|
end
|
|
|
|
def self.search(search)
|
|
search ? where("LOWER(username) LIKE LOWER(?) OR steamid LIKE ?", "%#{search}%", "%#{search}%") : scoped
|
|
end
|
|
|
|
def self.refadmins
|
|
Group.find(Group::REFEREES).users.order(:username) + Group.find(Group::ADMINS).users.order(:username)
|
|
end
|
|
|
|
def self.casters
|
|
Group.find(Group::CASTERS).users.order(:username)
|
|
end
|
|
end
|