jam-cloud/lib/jam_ruby/models/user.rb

505 lines
17 KiB
Ruby
Raw Normal View History

module JamRuby
class User < ActiveRecord::Base
include Tire::Model::Search
attr_accessible :first_name, :last_name, :email, :password, :password_confirmation, :city, :state, :country
2012-10-30 00:56:51 +00:00
attr_accessor :updating_password
2012-10-07 04:57:23 +00:00
self.primary_key = 'id'
2012-11-13 07:40:06 +00:00
# authorizations (for facebook, etc -- omniauth)
2012-11-13 21:21:04 +00:00
has_many :user_authorizations, :class_name => "JamRuby::UserAuthorization"
2012-11-13 02:52:05 +00:00
2012-10-29 10:45:47 +00:00
# connections (websocket-gateway)
has_many :connections, :class_name => "JamRuby::Connection"
2012-10-01 21:27:32 +00:00
2012-10-29 10:45:47 +00:00
# friend requests
2012-10-14 02:18:20 +00:00
has_many :friend_requests, :class_name => "JamRuby::FriendRequest"
2012-10-29 10:45:47 +00:00
# instruments
has_many :musician_instruments, :class_name => "JamRuby::MusicianInstrument"
has_many :instruments, :through => :musician_instruments, :class_name => "JamRuby::Instrument"
2012-10-29 10:45:47 +00:00
# bands
has_many :band_musicians, :class_name => "JamRuby::BandMusician"
has_many :bands, :through => :band_musicians, :class_name => "JamRuby::Band"
2012-10-14 02:18:20 +00:00
2012-11-16 02:08:37 +00:00
# recordings
has_many :musician_recordings, :class_name => "JamRuby::MusicianRecording", :foreign_key => "user_id"
has_many :recordings, :through => :musician_recordings, :class_name => "JamRuby::Recording"
2012-11-16 02:08:37 +00:00
2012-11-04 13:33:51 +00:00
# followers
has_many :followers, :class_name => "JamRuby::UserFollower", :foreign_key => "user_id"
has_many :inverse_followers, :through => :followers, :class_name => "JamRuby::User", :foreign_key => "follower_id"
# user likers (a musician has likers and may have likes too; fans do not have likers)
has_many :likers, :class_name => "JamRuby::UserLiker", :foreign_key => "user_id"
has_many :inverse_likers, :through => :likers, :class_name => "JamRuby::User", :foreign_key => "liker_id"
2012-11-04 22:54:53 +00:00
# user followings
2012-11-04 13:33:51 +00:00
has_many :followings, :class_name => "JamRuby::UserFollowing", :foreign_key => "follower_id"
has_many :inverse_followings, :through => :followings, :class_name => "JamRuby::User", :foreign_key => "user_id"
# user likes (fans and musicians have likes)
has_many :likes, :class_name => "JamRuby::UserLike", :foreign_key => "liker_id"
has_many :inverse_likes, :through => :followings, :class_name => "JamRuby::User", :foreign_key => "user_id"
2012-11-04 13:33:51 +00:00
2012-11-04 22:54:53 +00:00
# band followings
has_many :band_followings, :class_name => "JamRuby::BandFollower", :foreign_key => "follower_id"
2012-11-11 04:23:38 +00:00
has_many :inverse_band_followings, :through => :band_followings, :class_name => "JamRuby::Band", :foreign_key => "band_id"
2012-11-04 22:54:53 +00:00
# band likes
has_many :band_likes, :class_name => "JamRuby::BandLike", :foreign_key => "liker_id"
has_many :inverse_band_likes, :through => :band_likes, :class_name => "JamRuby::Band", :foreign_key => "band_id"
2012-11-16 02:08:37 +00:00
# favorites
has_many :favorites, :class_name => "JamRuby::UserFavorite", :foreign_key => "user_id"
has_many :inverse_favorites, :through => :favorites, :class_name => "JamRuby::User"
2012-10-29 10:45:47 +00:00
# friends
has_many :friendships, :class_name => "JamRuby::Friendship", :foreign_key => "user_id"
has_many :friends, :through => :friendships, :class_name => "JamRuby::User"
2012-10-01 21:27:32 +00:00
has_many :inverse_friendships, :class_name => "JamRuby::Friendship", :foreign_key => "friend_id"
has_many :inverse_friends, :through => :inverse_friendships, :source => :user, :class_name => "JamRuby::User"
2012-11-03 19:32:27 +00:00
# connections / music sessions
has_many :created_music_sessions, :foreign_key => "user_id", :inverse_of => :user, :class_name => "JamRuby::MusicSession" # sessions *created* by the user
2012-11-02 06:51:52 +00:00
has_many :music_sessions, :through => :connections, :class_name => "JamRuby::MusicSession"
# invitations
has_many :received_invitations, :foreign_key => "receiver_id", :inverse_of => :receiver, :class_name => "JamRuby::Invitation"
has_many :sent_invitations, :foreign_key => "sender_id", :inverse_of => :sender, :class_name => "JamRuby::Invitation"
# fan invitations
has_many :received_fan_invitations, :foreign_key => "receiver_id", :inverse_of => :receiver, :class_name => "JamRuby::FanInvitation"
has_many :sent_fan_invitations, :foreign_key => "sender_id", :inverse_of => :sender, :class_name => "JamRuby::FanInvitation"
2012-11-24 18:22:44 +00:00
# band invitations
has_many :received_band_invitations, :inverse_of => :receiver, :foreign_key => "user_id", :class_name => "JamRuby::BandInvitation"
has_many :sent_band_invitations, :inverse_of => :sender, :foreign_key => "creator_id", :class_name => "JamRuby::BandInvitation"
2012-11-09 06:51:17 +00:00
# This causes the authenticate method to be generated (among other stuff)
has_secure_password
before_save { |user| user.email = email.downcase }
2012-12-09 04:05:54 +00:00
before_save :create_remember_token, :if => :should_validate_password?
2012-11-11 04:23:38 +00:00
validates :first_name, presence: true, length: {maximum: 50}
validates :last_name, presence: true, length: {maximum: 50}
VALID_EMAIL_REGEX = /\A[\w+\-.]+@[a-z\d\-.]+\.[a-z]+\z/i
validates :email, presence: true, format: {with: VALID_EMAIL_REGEX},
uniqueness: {case_sensitive: false}
validates_length_of :password, minimum: 6, maximum: 100, :if => :should_validate_password?
2012-10-07 04:57:23 +00:00
validates_presence_of :password_confirmation, :if => :should_validate_password?
validates_confirmation_of :password, :if => :should_validate_password?
2012-10-14 02:18:20 +00:00
def online
2012-11-09 06:29:05 +00:00
@online ||= !self.connections.nil? && self.connections.size > 0
2012-10-14 02:18:20 +00:00
end
2012-11-18 02:59:59 +00:00
def name
return "#{first_name} #{last_name}"
end
def location
2012-11-16 02:08:37 +00:00
return "#{self.city}, #{self.state}, #{self.country}"
end
2012-10-07 04:57:23 +00:00
def should_validate_password?
updating_password || new_record?
end
2012-10-07 18:02:26 +00:00
def friends?(user)
return self.friends.exists?(user)
end
2012-11-06 04:47:50 +00:00
def friend_count
return self.friends.size
end
def liker_count
return 0
end
2012-11-06 04:47:50 +00:00
def follower_count
return self.followers.size
end
def following_count
2012-11-13 02:52:05 +00:00
return self.followings.size + self.band_followings.size
2012-11-06 04:47:50 +00:00
end
def favorite_count
return self.favorites.size
end
def recording_count
return self.recordings.size
end
def session_count
return self.music_sessions.size
end
def confirm_email!
2012-12-09 04:05:54 +00:00
self.email_confirmed = true
end
def my_session_settings
unless self.session_settings.nil?
return JSON.parse(self.session_settings)
else
return ""
end
end
2012-10-07 04:57:23 +00:00
def to_s
return email unless email.nil?
if !first_name.nil? && !last_name.nil?
return first_name + ' ' + last_name
end
2012-10-07 04:57:23 +00:00
return id
end
# helper method for creating / updating a User
2012-11-22 08:27:23 +00:00
def self.save(id, updater_id, first_name, last_name, email, password, password_confirmation, musician, gender,
birth_date, internet_service_provider, city, state, country, instruments, photo_url)
if id.nil?
validate_instruments(instruments, musician)
2012-10-29 10:45:47 +00:00
user = User.new()
else
validate_instruments(instruments, true)
user = User.find(id)
2012-11-13 02:52:05 +00:00
end
2012-11-22 08:27:23 +00:00
if user.id != updater_id
raise PermissionError, ValidationMessages::PERMISSION_VALIDATION_ERROR
2012-11-22 08:27:23 +00:00
end
2012-11-11 04:23:38 +00:00
# first name
unless first_name.nil?
user.first_name = first_name
2012-11-11 04:23:38 +00:00
end
# last name
unless last_name.nil?
user.last_name = last_name
2012-11-11 04:23:38 +00:00
end
# email
unless email.nil?
user.email = email
end
# password
unless password.nil?
user.password = password
2012-10-29 10:45:47 +00:00
end
# password confirmation
unless password_confirmation.nil?
user.password_confirmation = password_confirmation
end
# musician flag
unless musician.nil?
user.musician = musician
end
2012-11-13 02:52:05 +00:00
# gender
unless gender.nil?
account.gender = gender
2012-11-13 02:52:05 +00:00
end
# birthdate
unless birth_date.nil?
account.birth_date = birth_date
2012-11-13 02:52:05 +00:00
end
# ISP
unless internet_service_provider.nil?
account.internet_service_provider = internet_service_provider
2012-11-13 02:52:05 +00:00
end
2012-11-06 02:55:08 +00:00
# city
unless city.nil?
user.city = city
2012-11-06 02:55:08 +00:00
end
# state
unless state.nil?
user.state = state
2012-11-06 02:55:08 +00:00
end
# country
unless country.nil?
user.country = country
2012-11-06 02:55:08 +00:00
end
# instruments
unless instruments.nil?
UserManager.active_record_transaction do |user_manager|
# delete all instruments for this user first
2012-11-03 19:32:27 +00:00
unless user.id.nil? || user.id.length == 0
MusicianInstrument.delete_all(["user_id = ?", user.id])
end
# loop through each instrument in the array and save to the db
instruments.each do |musician_instrument_param|
instrument = Instrument.find(musician_instrument_param[:instrument_id])
musician_instrument = MusicianInstrument.new
musician_instrument.user = user
musician_instrument.instrument = instrument
musician_instrument.proficiency_level = musician_instrument_param[:proficiency_level]
musician_instrument.priority = musician_instrument_param[:priority]
musician_instrument.save
user.musician_instruments << musician_instrument
end
end
2012-10-29 10:45:47 +00:00
end
# photo url
unless photo_url.nil?
user.photo_url = photo_url
end
user.updated_at = Time.now.getutc
user.save
2012-10-29 10:45:47 +00:00
return user
end
def self.create_user_following(user_id, follower_id)
2012-11-22 08:27:23 +00:00
follower = UserFollower.new()
follower.user_id = user_id
follower.follower_id = follower_id
follower.save
end
def self.delete_following(user_id, band_id, follower_id)
if !user_id.nil?
JamRuby::UserFollower.delete_all "(user_id = '#{user_id}' AND follower_id = '#{follower_id}')"
elsif !band_id.nil?
JamRuby::BandFollower.delete_all "(band_id = '#{band_id}' AND follower_id = '#{follower_id}')"
end
end
def self.create_band_following(band_id, follower_id)
2012-11-22 08:27:23 +00:00
follower = BandFollower.new()
follower.band_id = band_id
follower.follower_id = follower_id
follower.save
end
def self.delete_band_following(band_id, follower_id)
JamRuby::BandFollower.delete_all "(band_id = '#{band_id}' AND follower_id = '#{follower_id}')"
end
def self.create_favorite(user_id, recording_id)
2012-11-22 08:27:23 +00:00
favorite = UserFavorite.new()
favorite.user_id = user_id
favorite.recording_id = recording_id
favorite.save
end
def self.delete_favorite(user_id, recording_id)
JamRuby::UserFavorite.delete_all "(user_id = '#{user_id}' AND recording_id = '#{recording_id}')"
end
2012-12-09 04:05:54 +00:00
def self.save_session_settings(user, music_session)
unless user.nil?
session_settings = { :band_id => music_session.band_id,
:musician_access => music_session.musician_access,
:approval_required => music_session.approval_required,
:fan_chat => music_session.fan_chat,
:fan_access => music_session.fan_access,
:description => music_session.description,
:genres => music_session.genres
}.to_json
user.session_settings = session_settings
user.save
end
end
# throws ActiveRecord::RecordNotFound if instrument is invalid
# throws an email delivery error if unable to connect out to SMTP
def self.signup(first_name, last_name, email, password, password_confirmation,
city, state, country, instruments, photo_url, signup_confirm_url)
user = User.new
UserManager.active_record_transaction do |user_manager|
user.first_name = first_name
user.last_name = last_name
user.email = email
#FIXME: Setting random password for social network logins. This
# is because we have validations all over the place on this.
# The right thing would be to have this null
if password.nil?
user.password = "blahblahblah"
user.password_confirmation = "blahblahblah"
else
user.password = password
user.password_confirmation = password_confirmation
end
user.admin = false
user.email_confirmed = false
user.musician = true
user.city = city
user.state = state
user.country = country
unless instruments.nil?
instruments.each do |musician_instrument_param|
instrument = Instrument.find(musician_instrument_param[:instrument_id])
musician_instrument = MusicianInstrument.new
musician_instrument.user = user
musician_instrument.instrument = instrument
musician_instrument.proficiency_level = musician_instrument_param[:proficiency_level]
musician_instrument.priority = musician_instrument_param[:priority]
musician_instrument.save
user.musician_instruments << musician_instrument
end
end
user.photo_url = photo_url
user.signup_token = SecureRandom.urlsafe_base64
2012-11-15 09:30:55 +00:00
user.save
2012-11-15 09:30:55 +00:00
if user.errors.any?
raise ActiveRecord::Rollback
else
# FIXME:
# It's not standard to require a confirmation when a user signs up with Facebook.
# We should stop asking for it.
#
# any errors here should also rollback the transaction; that's OK. If emails aren't going to be delivered,
# it's already a really bad situation; make user signup again
UserMailer.welcome_message(user, signup_confirm_url.nil? ? nil : (signup_confirm_url + "/" + user.signup_token) ).deliver
end
end
return user
end
# throws RecordNotFound if signup token is invalid; i.e., if it's nil, empty string, or not belonging to a user
def self.signup_confirm(signup_token)
if signup_token.nil? || signup_token.empty?
# there are plenty of confirmed users with nil signup_tokens, so we can't look on it
raise ActiveRecord::RecordNotFound
else
UserManager.active_record_transaction do |user_manager|
# throws ActiveRecord::RecordNotFound if invalid
user = User.find_by_signup_token!(signup_token)
user.signup_token = nil
2012-11-22 07:57:32 +00:00
user.confirm_email!
user.save
return user
end
end
end
# if valid credentials are supplied for an 'active' user, returns the user
# if not authenticated, returns nil
def self.authenticate(email, password)
# we only allow users that have confirmed email to authenticate
user = User.where('email_confirmed=true').find_by_email(email)
if user && user.authenticate(password)
return user
else
return nil
end
end
### Elasticsearch/Tire integration ###
#
# Define the name based on the environment
# We wouldn't like to erase dev data during
# test runs!
#
index_name("#{Environment.mode}-#{Environment.application}-users")
def to_indexed_json
{
2012-11-11 04:23:38 +00:00
:first_name => first_name,
:last_name => last_name,
:photo_url => photo_url,
:location => location,
:musician => musician
}.to_json
end
2012-11-12 12:28:44 +00:00
# only put users into the search index if their eail is confirmed
after_save do
update_index if email_confirmed
end
after_destroy do
update_index
end
class << self
def create_search_index
Tire.index(User.index_name) do
create(
:settings => Search.index_settings,
:mappings => {
"jam_ruby/user" => {
:properties => {
:photo_url => { :type => :string, :index => :not_analyzed, :include_in_all => false},
:location => { :type => :string },
2012-11-11 04:23:38 +00:00
:first_name => { :type => :string, :boost => 100 },
:last_name => { :type => :string, :boost => 100 },
:musician => { :type => :boolean, :index => :not_analyzed, :include_in_all => false}
}
}
}
)
end
end
def delete_search_index
search_index.delete
end
def search_index
Tire.index(User.index_name)
end
end
### Elasticsearch/Tire integration
2012-10-07 04:57:23 +00:00
private
def create_remember_token
self.remember_token = SecureRandom.urlsafe_base64
end
def self.validate_instruments(instruments, is_nil_ok)
if is_nil_ok && instruments.nil?
return
end
if instruments.nil?
raise JamRuby::JamArgumentError, ValidationMessages::INSTRUMENT_MINIMUM_NOT_MET
else
if instruments.size < Limits::MIN_INSTRUMENTS_PER_MUSICIAN
raise JamRuby::JamArgumentError, ValidationMessages::INSTRUMENT_MINIMUM_NOT_MET
end
if instruments.size > Limits::MAX_INSTRUMENTS_PER_MUSICIAN
raise JamRuby::JamArgumentError, ValidationMessages::INSTRUMENT_LIMIT_EXCEEDED
end
end
end
end
end