jam-cloud/ruby/spec/jam_ruby/models/music_session_spec.rb

946 lines
43 KiB
Ruby

require 'spec_helper'
describe MusicSession do
let(:creator) {FactoryGirl.create(:user, last_jam_locidispid: 1)}
let(:some_user) { FactoryGirl.create(:user) }
let(:music_session1) { FactoryGirl.create(:music_session) }
let(:music_session2) { FactoryGirl.create(:music_session) }
let(:music_session3) { FactoryGirl.create(:music_session) }
let(:music_session4) { FactoryGirl.create(:music_session) }
describe "validations" do
it "genre must be set" do
music_session = FactoryGirl.build(:music_session)
music_session.genre = nil
music_session.save.should be_false
music_session.errors[:genre].should == ["can't be blank"]
end
it "updates the fields of a music session properly" do
genre1 = FactoryGirl.create(:genre)
genre2 = FactoryGirl.create(:genre)
genre3 = FactoryGirl.create(:genre)
genre4 = FactoryGirl.create(:genre)
creator = FactoryGirl.create(:user)
session = FactoryGirl.create(:music_session, :creator => creator, :description => "Session", :genre => genre3)
session.update_attributes({:description => "Session2", :genre => genre1})
session.reload
session.description.should == "Session2"
session.genre.should == genre1
end
it "must have legal_terms accepted" do
user1 = FactoryGirl.create(:user)
music_session = FactoryGirl.build(:music_session, :creator => user1, legal_terms: false)
music_session.save
music_session.valid?.should be_false
music_session.errors["legal_terms"].should == ["is not included in the list"]
end
it "cannot have profanity in the description" do
user1 = FactoryGirl.create(:user)
music_session = FactoryGirl.build(:music_session, :creator => user1, legal_terms: false, :description => "fuck you")
music_session.save
music_session.valid?.should be_false
end
end
describe "create" do
let(:open_params) {
{
name: "session 1",
description: "my session",
genres: ['ambient'],
musician_access: true,
fan_access: true,
approval_required: true,
fan_chat: true,
legal_policy: 'Standard',
language: 'eng',
start: "Thu Jul 10 2014 10:00 PM",
duration: 30,
timezone: "Central Time (US & Canada),America/Chicago",
open_rsvps: true,
legal_terms: true,
recurring_mode: 'once',
isUnstructuredRsvp: true,
rsvp_slots: [{ instrument_id: "other", proficiency_level: 1, approve: true}]
}
}
it "wide open scheduled session" do
session = MusicSession.create(creator, open_params)
session.valid?.should be_true
# verify that scheduled_start is now 5 hours ahead of what was specified (CST during summer is -5 offset)
session.scheduled_start.utc.should == DateTime.new(2014,07,11,3,00,0)
# verify that the update_scheduled_start does not disturb scheduled_start
session.save!
session.scheduled_start.utc.should == DateTime.new(2014,07,11,3,00,0)
end
it "works with UTC timezone" do
open_params[:timezone] = 'UTC,Etc/UTC'
session = MusicSession.create(creator, open_params)
session.valid?.should be_true
# verify that scheduled_start is now 5 hours ahead of what was specified (CST during summer is -5 offset)
session.scheduled_start.should == DateTime.new(2014,07,10,22,00,0)
end
it "no scheduled_start" do
open_params[:timezone] = nil
open_params[:scheduled_start] = nil
open_params[:scheduled_duration] = nil
session = MusicSession.create(creator, open_params)
session.valid?.should be_true
session.scheduled_start.should be_nil
end
end
describe "pretty_scheduled_start" do
it "displays central time correctly" do
time = MusicSession.parse_scheduled_start("Thu Jul 10 2014 10:00 PM", "Central Time (US & Canada),America/Chicago")
music_session = FactoryGirl.create(:music_session, scheduled_start: time, timezone: "Central Time (US & Canada),America/Chicago")
music_session.pretty_scheduled_start(true).should == 'Thursday, July 10, 10:00-11:00 PM Central Time (US & Canada)'
music_session.pretty_scheduled_start(false).should == 'Thursday, July 10 - 10:00pm'
end
it "displays default correctly" do
music_session = FactoryGirl.create(:music_session, scheduled_start: nil)
music_session.pretty_scheduled_start(true).should == 'Date and time TBD'
music_session.pretty_scheduled_start(false).should == 'Date and time TBD'
end
end
describe "nindex" do
it "nindex orders two sessions by created_at starting with most recent" do
creator = FactoryGirl.create(:user)
creator2 = FactoryGirl.create(:user)
earlier_session = FactoryGirl.create(:active_music_session, :creator => creator, :description => "Earlier Session")
c1 = FactoryGirl.create(:connection, user: creator, music_session: earlier_session, addr: 0x01020304, locidispid: 1)
later_session = FactoryGirl.create(:active_music_session, :creator => creator2, :description => "Later Session")
c2 = FactoryGirl.create(:connection, user: creator2, music_session: later_session, addr: 0x21020304, locidispid: 2)
user = FactoryGirl.create(:user)
c3 = FactoryGirl.create(:connection, user: user, locidispid: 3)
Score.createx(c1.locidispid, c1.client_id, c1.addr, c3.locidispid, c3.client_id, c3.addr, 20, nil);
Score.createx(c2.locidispid, c2.client_id, c2.addr, c3.locidispid, c3.client_id, c3.addr, 30, nil);
# scores!
#ActiveRecord::Base.logger = Logger.new(STDOUT)
music_sessions = ActiveMusicSession.nindex(user, client_id: c3.client_id).take(100)
#music_sessions = MusicSession.index(user).take(100)
#ActiveRecord::Base.logger = nil
music_sessions.length.should == 2
music_sessions[0].id.should == later_session.id
music_sessions[1].id.should == earlier_session.id
end
end
it 'uninvited users cant join approval-required sessions without invitation' do
user1 = FactoryGirl.create(:user) # in the jam session
user2 = FactoryGirl.create(:user) # in the jam session
music_session = FactoryGirl.create(:active_music_session, :creator => user1, :musician_access => true, :approval_required => true)
connection1 = FactoryGirl.create(:connection, :user => user1, :music_session => music_session)
expect { FactoryGirl.create(:connection, :user => user2, :music_session => music_session, :joining_session => true) }.to raise_error(ActiveRecord::RecordInvalid)
end
it "is_recording? returns false if not recording" do
user1 = FactoryGirl.create(:user)
music_session = FactoryGirl.build(:active_music_session, :creator => user1)
music_session.is_recording?.should be_false
end
describe "recordings" do
before(:each) do
@user1 = FactoryGirl.create(:user)
@connection = FactoryGirl.create(:connection, :user => @user1)
@instrument = FactoryGirl.create(:instrument, :description => 'a great instrument')
@track = FactoryGirl.create(:track, :connection => @connection, :instrument => @instrument)
@music_session = FactoryGirl.create(:active_music_session, :creator => @user1, :musician_access => true)
# @music_session.connections << @connection
@music_session.save!
@connection.join_the_session(@music_session, true, nil, @user1, 10)
end
describe "not recording" do
it "stop_recording should return nil if not recording" do
@music_session.stop_recording.should be_nil
end
end
describe "currently recording" do
before(:each) do
@recording = FactoryGirl.create(:recording, :music_session => @music_session, :owner => @user1)
end
it "is_recording? returns true if recording" do
@music_session.is_recording?.should be_true
end
it "stop_recording should return recording object if recording" do
@music_session.stop_recording.should == @recording
end
end
describe "claim a recording" do
before(:each) do
@recording = Recording.start(@music_session, @user1)
@recording.errors.any?.should be_false
@recording.stop
@recording.reload
@claimed_recording = @recording.claim(@user1, "name", "description", Genre.first, true)
@claimed_recording.errors.any?.should be_false
end
it "allow a claimed recording to be associated" do
@music_session.claimed_recording_start(@user1, @claimed_recording)
@music_session.errors.any?.should be_false
@music_session.reload
@music_session.claimed_recording.should == @claimed_recording
@music_session.claimed_recording_initiator.should == @user1
end
it "allow a claimed recording to be removed" do
@music_session.claimed_recording_start(@user1, @claimed_recording)
@music_session.errors.any?.should be_false
@music_session.claimed_recording_stop
@music_session.errors.any?.should be_false
@music_session.reload
@music_session.claimed_recording.should be_nil
@music_session.claimed_recording_initiator.should be_nil
end
it "disallow a claimed recording to be started when already started by someone else" do
@user2 = FactoryGirl.create(:user)
@music_session.claimed_recording_start(@user1, @claimed_recording)
@music_session.errors.any?.should be_false
@music_session.claimed_recording_start(@user2, @claimed_recording)
@music_session.errors.any?.should be_true
@music_session.errors[:claimed_recording] == [ValidationMessages::CLAIMED_RECORDING_ALREADY_IN_PROGRESS]
end
it "allow a claimed recording to be started when already started by self" do
@user2 = FactoryGirl.create(:user)
@claimed_recording2 = @recording.claim(@user1, "name", "description", Genre.first, true)
@music_session.claimed_recording_start(@user1, @claimed_recording)
@music_session.errors.any?.should be_false
@music_session.claimed_recording_start(@user1, @claimed_recording2)
@music_session.errors.any?.should be_true
end
end
end
describe "get_connection_ids" do
before(:each) do
@user1 = FactoryGirl.create(:user)
@user2 = FactoryGirl.create(:user)
@music_session = FactoryGirl.create(:active_music_session, :creator => @user1, :musician_access => true)
@connection1 = FactoryGirl.create(:connection, :user => @user1, :music_session => @music_session, :as_musician => true)
@connection2 = FactoryGirl.create(:connection, :user => @user2, :music_session => @music_session, :as_musician => false)
end
it "get all connections" do
@music_session.get_connection_ids().should == [@connection1.client_id, @connection2.client_id]
end
it "exclude non-musicians" do
@music_session.get_connection_ids(as_musician: true).should == [@connection1.client_id]
end
it "exclude musicians" do
@music_session.get_connection_ids(as_musician: false).should == [@connection2.client_id]
end
it "exclude particular client" do
@music_session.get_connection_ids(exclude_client_id: @connection1.client_id).should == [@connection2.client_id]
end
it "exclude particular client and exclude non-musicians" do
@music_session.get_connection_ids(exclude_client_id: @connection2.client_id, as_musician: true).should == [@connection1.client_id]
end
end
describe "approved_rsvps" do
it "aggregrates instrument info" do
creators_slot = music_session1.rsvp_slots[0]
music_session1.approved_rsvps.length.should == 1
approved_user = music_session1.approved_rsvps[0]
JSON.parse(approved_user[:instrument_ids])[0].should == creators_slot.instrument.id
JSON.parse(approved_user[:instrument_descriptions])[0].should == creators_slot.instrument.description
JSON.parse(approved_user[:instrument_proficiencies])[0].should == creators_slot.proficiency_level
JSON.parse(approved_user[:rsvp_request_ids])[0].should == creators_slot.rsvp_requests[0].id
end
it "unstructured rsvps should still be returned" do
music_session1.rsvp_slots.length.should == 1
creators_slot = music_session1.rsvp_slots[0]
# now take out the instrument and proficiency of the rsvp_slot (make it unstructured)
creators_slot.is_unstructured_rsvp = true
creators_slot.instrument = nil
creators_slot.proficiency_level = nil
creators_slot.save!
music_session = MusicSession.find(music_session1.id)
approved_rsvps = music_session.approved_rsvps
approved_rsvps.length.should == 1
approved_user = approved_rsvps[0]
JSON.parse(approved_user[:instrument_ids])[0].should == nil
JSON.parse(approved_user[:instrument_descriptions])[0].should == nil
JSON.parse(approved_user[:instrument_proficiencies])[0].should == nil
JSON.parse(approved_user[:rsvp_request_ids])[0].should == creators_slot.rsvp_requests[0].id
end
it "handles 2 instruments for a single request correctly" do
rsvp_request = FactoryGirl.create(:rsvp_request_for_multiple_slots, user: some_user, music_session: music_session1, number: 2, chosen:true)
approved_rsvps = music_session1.approved_rsvps
approved_rsvps.length.should == 2
# find the user who made the request for 2 rsvp slots in the approved_users array
approved_some_user = approved_rsvps.find {|s| s.id == some_user.id}
instrument_ids = JSON.parse(approved_some_user[:instrument_ids])
instrument_ids.should =~ rsvp_request.rsvp_slots.map {|slot| slot.instrument_id }
instrument_descriptions = JSON.parse(approved_some_user[:instrument_descriptions])
instrument_descriptions.should =~ rsvp_request.rsvp_slots.map {|slot| slot.instrument.description }
instrument_proficiencies = JSON.parse(approved_some_user[:instrument_proficiencies])
instrument_proficiencies.should =~ rsvp_request.rsvp_slots.map {|slot| slot.proficiency_level }
JSON.parse(approved_some_user[:rsvp_request_ids])[0].should == rsvp_request.id
end
end
describe "parse_scheduled_start" do
it "converts central time correctly" do
# CST has -5 offset in summery
time = DateTime.new(2004,10,15,1,30,0).strftime('%Y-%m-%d %H:%M:%S')
converted = MusicSession.parse_scheduled_start(time, 'Central Time (US & Canada),America/Chicago')
converted.should == DateTime.new(2004,10,15,6,30,0, '+0')
# CST has -6 offset in winter
time = DateTime.new(2004,11,15,1,30,0).strftime('%Y-%m-%d %H:%M:%S')
converted = MusicSession.parse_scheduled_start(time, 'Central Time (US & Canada),America/Chicago')
converted.should == DateTime.new(2004,11,15,7,30,0, '+0')
end
it "converts UTC correctly" do
# should not shift
time = DateTime.new(2004,10,15,1,30,0, '+0').strftime('%Y-%m-%d %H:%M:%S')
converted = MusicSession.parse_scheduled_start(time, 'UTC,Etc/UTC')
converted.should == DateTime.new(2004,10,15,1,30,0, '+0')
# should not shift
time = DateTime.new(2004,11,15,1,30,0).strftime('%Y-%m-%d %H:%M:%S')
converted = MusicSession.parse_scheduled_start(time, 'UTC,Etc/UTC')
converted.should == DateTime.new(2004,11,15,1,30,0, '+0')
end
end
describe "scheduled" do
it "includes any RSVP'ed" do
rsvp_request = FactoryGirl.create(:rsvp_request_for_multiple_slots, user: some_user, music_session: music_session1, number: 2, chosen:true)
approved_rsvps = music_session1.approved_rsvps
approved_rsvps.length.should == 2
sessions = MusicSession.scheduled(approved_rsvps[0])
sessions.length.should == 1
sessions = MusicSession.scheduled(approved_rsvps[1])
sessions.length.should == 1
end
it "includes invited" do
invitee = FactoryGirl.create(:user, last_jam_audio_latency: 30, last_jam_locidispid: 3)
FactoryGirl.create(:friendship, user: creator, friend: invitee)
FactoryGirl.create(:friendship, user: invitee, friend: creator)
music_session = FactoryGirl.create(:music_session, creator: creator)
FactoryGirl.create(:invitation, receiver:invitee, sender:creator, music_session: music_session)
sessions = MusicSession.scheduled(invitee)
sessions.length.should == 1
end
it "excludes based on time-range" do
session = FactoryGirl.create(:music_session, scheduled_start: Time.now)
sessions = MusicSession.scheduled(session.creator)
sessions.length.should == 1
session.scheduled_start = 11.hours.ago
session.save!
sessions = MusicSession.scheduled(session.creator)
sessions.length.should == 1
session.scheduled_start = 13.hours.ago
session.save!
sessions = MusicSession.scheduled(session.creator)
sessions.length.should == 0
session.scheduled_start = 13.hours.from_now
session.save!
sessions = MusicSession.scheduled(session.creator)
sessions.length.should == 1
session.scheduled_start = 300.days.from_now
session.save!
sessions = MusicSession.scheduled(session.creator)
sessions.length.should == 1
session.scheduled_start = nil
session.save!
sessions = MusicSession.scheduled(session.creator)
sessions.length.should == 1
end
it "excludes canceled" do
session = FactoryGirl.create(:music_session, scheduled_start: Time.now)
sessions = MusicSession.scheduled(session.creator)
sessions.length.should == 1
session.canceled = true
session.save!
MusicSession.scheduled(session.creator).length.should == 0
end
end
def sms(user, params)
ActiveRecord::Base.transaction do
return MusicSession.sms_index(user, params)
end
end
def session_with_scores(user, music_session_id)
ActiveRecord::Base.transaction do
return MusicSession.session_with_scores(user, music_session_id)
end
end
describe "session_with_scores", no_transaction: true do
let(:conn) { FactoryGirl.create(:connection, user: creator, locidispid: creator.last_jam_locidispid) }
let(:searcher) { FactoryGirl.create(:user, last_jam_locidispid: 2) }
let(:searcher_conn) { FactoryGirl.create(:connection, user: searcher, ip_address: '2.2.2.2', locidispid: searcher.last_jam_locidispid) }
let(:default_opts) { {client_id: searcher_conn.client_id} }
let(:network_score) { 20 }
before(:each) do
Score.createx(conn.locidispid, conn.client_id, conn.addr, searcher_conn.locidispid, searcher_conn.client_id, searcher_conn.addr, network_score, nil, nil, {auserid: creator.id, buserid: searcher.id})
end
it "invalid session ID" do
expect {session_with_scores(searcher, 'blah')}.to raise_error(ActiveRecord::RecordNotFound)
end
it "one session with scores" do
session = FactoryGirl.create(:music_session, creator: creator)
music_session, user_scores = session_with_scores(searcher, session.id)
music_session.latency.should be_nil # we don't return music_session.latency with session_with_scores, because it's used for sorting among many sessions
user_scores.length.should == 1
user_scores[creator.id][:full_score].should == (network_score + searcher.last_jam_audio_latency + creator.last_jam_audio_latency )
end
end
describe "sms_index", no_transaction: true do
describe "simple" do
let(:conn) { FactoryGirl.create(:connection, user: creator, locidispid: creator.last_jam_locidispid) }
let(:searcher) { FactoryGirl.create(:user, last_jam_locidispid: 2) }
let(:searcher_conn) { FactoryGirl.create(:connection, user: searcher, ip_address: '2.2.2.2', locidispid: searcher.last_jam_locidispid) }
let(:default_opts) { {client_id: searcher_conn.client_id} }
let(:network_score) { 20 }
before(:each) do
Score.createx(conn.locidispid, conn.client_id, conn.addr, searcher_conn.locidispid, searcher_conn.client_id, searcher_conn.addr, network_score, nil, nil, {auserid: creator.id, buserid: searcher.id})
end
it "no results" do
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 0
user_scores.length.should == 0
end
it "one session shows/hides based on open_rsvps" do
creator.last_jam_locidispid = conn.locidispid
creator.save!
music_session = FactoryGirl.create(:music_session, creator: creator, scheduled_start: nil)
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 1
music_session.open_rsvps = false
music_session.save!
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 0
end
it "one session with no scheduled_start time" do
creator.last_jam_locidispid = conn.locidispid
creator.save!
music_session = FactoryGirl.create(:music_session, creator: creator, scheduled_start: nil)
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 1
end
it "one session, one RSVP (creator)" do
creator.last_jam_locidispid = conn.locidispid
creator.save!
music_session = FactoryGirl.create(:music_session, creator: creator)
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 1
music_sessions[0].tag.should == 3 # open session sort
music_sessions[0].latency.should == (network_score + searcher.last_jam_audio_latency + creator.last_jam_audio_latency )
user_scores.length.should == 1
user_scores[creator.id][:full_score].should == (network_score + searcher.last_jam_audio_latency + creator.last_jam_audio_latency )
end
it "skip session with past due scheduled_start time" do
interval = MusicSession::UNSTARTED_INTERVAL_DAYS_SKIP
dd = Time.now - (interval.to_i + 1).days
Timecop.travel(dd)
msess1 = FactoryGirl.create(:music_session, creator: creator, scheduled_start: dd)
msess2 = FactoryGirl.create(:music_session, creator: creator)
music_sessions, user_scores = sms(searcher, default_opts)
expect(music_sessions.length).to be(1)
expect(music_sessions[0].id).to eq(msess2.id)
end
it "filters sessions in the past" do
music_session = FactoryGirl.create(:music_session, creator: creator)
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 1
# 15 minutes is the edge of forgiveness
music_session.scheduled_start = 16.minutes.ago
music_session.save!
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 0
# this should still fall in time
music_session.scheduled_start = 14.minutes.ago
music_session.save!
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 1
end
it "one session, one RSVP (creator), one invitation" do
# create an invitee, and friend them with the creator (you have to be friends to send an invite)
invitee = FactoryGirl.create(:user, last_jam_audio_latency: 30, last_jam_locidispid: 3)
FactoryGirl.create(:friendship, user: creator, friend: invitee)
FactoryGirl.create(:friendship, user: invitee, friend: creator)
music_session = FactoryGirl.create(:music_session, creator: creator)
FactoryGirl.create(:invitation, receiver:invitee, sender:creator, music_session: music_session)
# create a score between creator, searcher
Score.createx(creator.last_jam_locidispid, conn.client_id, conn.addr, searcher.last_jam_locidispid, searcher_conn.client_id, searcher_conn.addr, network_score, nil, nil, {auserid: creator.id, buserid: searcher.id})
# create a score between invitee, and searcher
Score.createx(invitee.last_jam_locidispid, 'immaterial', 1, searcher.last_jam_locidispid, searcher_conn.client_id, searcher_conn.addr, network_score, nil, nil, {auserid: invitee.id, buserid: searcher.id})
# create a score between invitee, and creator
Score.createx(invitee.last_jam_locidispid, 'immaterial', 1, creator.last_jam_locidispid, conn.client_id, conn.addr, network_score, nil, nil)
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 1
music_sessions[0].tag.should == 3 # open session sort
music_sessions[0].latency.should == (network_score + searcher.last_jam_audio_latency + creator.last_jam_audio_latency )
user_scores.length.should == 1 # the creator (invitees are not included)
user_scores[creator.id][:full_score].should == (network_score + searcher.last_jam_audio_latency + creator.last_jam_audio_latency )
#search with the invitee this time.
invitee_conn = FactoryGirl.create(:connection, user: invitee, ip_address: '3.3.3.3', locidispid: invitee.last_jam_locidispid)
music_sessions, user_scores = sms(invitee, {client_id: invitee_conn.client_id})
music_sessions.length.should == 1
music_sessions[0].tag.should == 2 # invited sort
music_sessions[0].latency.should == ((network_score + invitee.last_jam_audio_latency + creator.last_jam_audio_latency )).ceil
user_scores.length.should == 1 # the creator, and the invitee
user_scores[creator.id][:full_score].should == ((network_score + invitee.last_jam_audio_latency + creator.last_jam_audio_latency )).ceil
end
it "does not show when it goes active" do
# we create a scheduled session--it should return
music_session = FactoryGirl.create(:music_session, creator: creator, scheduled_start: nil)
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 1
# but then make an active session for this scheduled session
ams = FactoryGirl.create(:active_music_session, music_session: music_session, creator: creator, musician_access: true)
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 0
# finally, delete the active session, and see results go back to one
ams.delete
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 1
end
it "should allow a null locidispid to search" do
searcher_conn.locidispid = nil
searcher_conn.save!
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 0
end
describe "keywords" do
before(:each) do
creator.last_jam_locidispid = conn.locidispid
creator.save!
FactoryGirl.create(:music_session, creator: creator, scheduled_start: nil, description: 'chunky icecream for the fools')
end
it "handles single keyword" do
default_opts[:keyword] = 'chunky'
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 1
user_scores.length.should == 1
end
it "handles two keyword" do
default_opts[:keyword] = 'chunky for'
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 1
user_scores.length.should == 1
end
it "handles single quote" do
default_opts[:keyword] = "chun'ky fo'r"
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 0
user_scores.length.should == 1
default_opts[:keyword] = "chunky for'"
music_sessions, user_scores = sms(searcher, default_opts)
music_sessions.length.should == 1
user_scores.length.should == 1
end
end
end
describe "sorting" do
let!(:creator_1) { FactoryGirl.create(:user, last_jam_locidispid: 4, last_jam_audio_latency: 8) }
let!(:creator_conn_1) { FactoryGirl.create(:connection, user: creator_1, ip_address: '4.4.4.4', locidispid: 4, addr:4) }
let!(:creator_2) { FactoryGirl.create(:user, last_jam_locidispid: 1, last_jam_audio_latency: 10) }
let!(:creator_conn_2) { FactoryGirl.create(:connection, user: creator_2, ip_address: '4.4.4.4', locidispid: 1, addr:1) }
let!(:creator_3) { FactoryGirl.create(:user, last_jam_locidispid: 2, last_jam_audio_latency: 12) }
let!(:creator_conn_3) { FactoryGirl.create(:connection, user: creator_3, ip_address: '5.5.5.5', locidispid: 2, addr:2) }
let!(:searcher_1) { FactoryGirl.create(:user, last_jam_locidispid: 5, last_jam_audio_latency: 6) }
let!(:searcher_conn_1) { FactoryGirl.create(:connection, user: searcher_1, ip_address: '8.8.8.8', locidispid: 5, addr:5) }
let!(:searcher_2) { FactoryGirl.create(:user, last_jam_locidispid: 3, last_jam_audio_latency: 14) }
let!(:searcher_conn_2) { FactoryGirl.create(:connection, user: searcher_2, ip_address: '9.9.9.9', locidispid: 3, addr:3) }
let!(:music_session_1) { FactoryGirl.create(:music_session, creator: creator_1, genre: Genre.find('african'), language: 'eng', description: "Bunny Jumps") }
let!(:music_session_2) { FactoryGirl.create(:music_session, creator: creator_2, genre: Genre.find('ambient'), language: 'spa', description: "Play with us as we jam to beatles and bunnies") }
let!(:music_session_3) { FactoryGirl.create(:music_session, creator: creator_3) }
let(:good_network_score) { 20 }
let(:fair_network_score) { 30 }
let(:bad_network_score) { 40 }
before(:each) do
# add an RSVP for searcher_1 to music_session_1
searcher_rsvp_slot = FactoryGirl.create(:rsvp_slot, music_session: music_session_1, instrument: Instrument.find('piano'))
searcher_rsvp_request = FactoryGirl.create(:rsvp_request, user: searcher_1)
searcher_rsvp_request_rsvp_slot = FactoryGirl.create(:rsvp_request_rsvp_slot, chosen:true, rsvp_request: searcher_rsvp_request, rsvp_slot: searcher_rsvp_slot)
# add an invitation to searcher_1 to music_session_2
FactoryGirl.create(:friendship, user: creator_2, friend: searcher_1)
FactoryGirl.create(:friendship, user: searcher_1, friend: creator_2)
FactoryGirl.create(:invitation, receiver:searcher_1, sender:creator_2, music_session: music_session_2)
end
it "searcher_1" do
# create a bad score between searcher_1 and creator_1 (but we should still see it sort 1st because it's got an RSVP to the searcher)
Score.createx(searcher_conn_1.locidispid, searcher_conn_1.client_id, searcher_conn_1.addr, creator_conn_1.locidispid, creator_conn_1.client_id, creator_conn_1.addr, bad_network_score, nil, nil, {auserid: searcher_1.id, buserid: creator_1.id})
# create a fair score between searcher_1 and creator_2 (but we should still see it sort 2st because it's got an invitation to the searcher)
Score.createx(searcher_conn_1.locidispid, searcher_conn_1.client_id, searcher_conn_1.addr, creator_conn_2.locidispid, creator_conn_2.client_id, creator_conn_2.addr, fair_network_score, nil, nil, {auserid: searcher_1.id, buserid: creator_2.id})
# create a good score between searcher_1 and creator_3 (but we should still see it sort last because it's an open session; no affiliation with the searcher)
Score.createx(searcher_conn_1.locidispid, searcher_conn_1.client_id, searcher_conn_1.addr, creator_conn_3.locidispid, creator_conn_3.client_id, creator_conn_3.addr, good_network_score, nil, nil, {auserid: searcher_1.id, buserid: creator_3.id})
music_sessions, user_scores = sms(searcher_1, {client_id: searcher_conn_1.client_id})
music_sessions.length.should == 3
music_session = music_sessions[0]
music_session.should == music_session_1
music_session.tag.should == 1 # RSVP
music_session.latency.should == (bad_network_score + searcher_1.last_jam_audio_latency + creator_1.last_jam_audio_latency )
music_session = music_sessions[1]
music_session.should == music_session_2
music_session.tag.should == 2 # INVITE
music_session.latency.should == (fair_network_score + searcher_1.last_jam_audio_latency + creator_2.last_jam_audio_latency )
music_session = music_sessions[2]
music_session.should == music_session_3
music_session.tag.should == 3 # OPEN
music_session.latency.should == (good_network_score + searcher_1.last_jam_audio_latency + creator_3.last_jam_audio_latency )
user_scores.length.should == 3 # the creator, and the invitee
user_scores[creator_1.id][:full_score].should == (bad_network_score + searcher_1.last_jam_audio_latency + creator_1.last_jam_audio_latency )
# let's make music_session_3 invisible, and verify the count goes to 2
music_session_3.open_rsvps = false
music_session_3.save!
music_sessions, user_scores = sms(searcher_1, {client_id: searcher_conn_1.client_id})
music_sessions.length.should == 2
# let's make music_session_2 invisible, but still the count should be the same (because searcher_1 have an invite)
music_session_2.open_rsvps = false
music_session_2.save!
music_sessions, user_scores = sms(searcher_1, {client_id: searcher_conn_1.client_id})
music_sessions.length.should == 2
# and lastly with music_session_1, make it invisible, and still it should be visible to the searcher (because searcher_1 has an invite)
music_session_1.open_rsvps = false
music_session_1.save!
music_sessions, user_scores = sms(searcher_1, {client_id: searcher_conn_1.client_id})
music_sessions.length.should == 2
end
end
describe "parameters" do
let(:creator_1) { FactoryGirl.create(:user, last_jam_locidispid: 4, last_jam_audio_latency: 8) }
let(:creator_conn_1) { FactoryGirl.create(:connection, user: creator_1, ip_address: '4.4.4.4', locidispid: 4, addr:4) }
let(:creator_2) { FactoryGirl.create(:user, last_jam_locidispid: 1, last_jam_audio_latency: 10) }
let(:creator_conn_2) { FactoryGirl.create(:connection, user: creator_2, ip_address: '4.4.4.4', locidispid: 1, addr:1) }
let(:creator_3) { FactoryGirl.create(:user, last_jam_locidispid: 2, last_jam_audio_latency: 12) }
let(:creator_conn_3) { FactoryGirl.create(:connection, user: creator_3, ip_address: '5.5.5.5', locidispid: 2, addr:2) }
let(:searcher_1) { FactoryGirl.create(:user, last_jam_locidispid: 5, last_jam_audio_latency: 6) }
let(:searcher_conn_1) { FactoryGirl.create(:connection, user: searcher_1, ip_address: '8.8.8.8', locidispid: 5, addr:5) }
let(:searcher_2) { FactoryGirl.create(:user, last_jam_locidispid: 3, last_jam_audio_latency: 14) }
let(:searcher_conn_2) { FactoryGirl.create(:connection, user: searcher_2, ip_address: '9.9.9.9', locidispid: 3, addr:3) }
let!(:music_session_1) { FactoryGirl.create(:music_session, :creator => creator_1, genre: Genre.find('african'), language: 'eng', description: "Bunny Jumps" ) }
let!(:music_session_2) { FactoryGirl.create(:music_session, :creator => creator_2, genre: Genre.find('ambient'), language: 'spa', description: "Play with us as we jam to beatles and bunnies") }
let(:good_network_score) { 20 }
let(:fair_network_score) { 30 }
it "offset/limit" do
# set up some scores to control sorting
Score.createx(searcher_conn_1.locidispid, searcher_conn_1.client_id, searcher_conn_1.addr, creator_conn_1.locidispid, creator_conn_1.client_id, creator_conn_1.addr, good_network_score, nil)
Score.createx(searcher_conn_1.locidispid, searcher_conn_1.client_id, searcher_conn_1.addr, creator_conn_2.locidispid, creator_conn_2.client_id, creator_conn_2.addr, fair_network_score, nil)
# verify we can get all 2 sessions
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id)
music_sessions.length.should == 2
music_sessions[0].should == music_session_1
# grab just the 1st
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, offset:0, limit:1)
music_sessions.length.should == 1
music_sessions[0].should == music_session_1
# then the second
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, offset:1, limit:2)
music_sessions.length.should == 1
music_sessions[0].should == music_session_2
end
it "genre" do
# verify we can get all 2 sessions
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id)
music_sessions.length.should == 2
# get only african
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, genre: 'african')
music_sessions.length.should == 1
music_sessions[0].genre.should == Genre.find('african')
# get only ambient
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, genre: 'ambient')
music_sessions.length.should == 1
music_sessions[0].genre.should == Genre.find('ambient')
end
it "language" do
# verify we can get all 2 sessions
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id)
music_sessions.length.should == 2
# get only english
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, lang: 'eng')
music_sessions.length.should == 1
music_sessions[0].language.should == 'eng'
# get only ambient
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, lang: 'spa')
music_sessions.length.should == 1
music_sessions[0].language.should == 'spa'
end
it "keyword" do
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, keyword: 'Jump')
music_sessions.length.should == 1
music_sessions[0].should == music_session_1
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, keyword: 'Bunny')
music_sessions.length.should == 2
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, keyword: 'play')
music_sessions.length.should == 1
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, keyword: 'bun')
music_sessions.length.should == 2
end
it "date" do
music_session_1.scheduled_start = 1.days.ago
music_session_1.save!
# if no day/timezone_offset specified, then the 15 minute slush rule will still kick in, nixing music_session_1
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id)
music_sessions.length.should == 1
music_sessions[0].should == music_session_2
# find today's session
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, day: Date.today.to_s, timezone_offset: DateTime.now.offset.numerator)
music_sessions.length.should == 1
music_sessions[0].should == music_session_2
# find yesterday's session... oh wait, you can't find a session for yesterday, because the 15 minute slush rule will still kick in.
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, day: (Date.today - 1).to_s, timezone_offset: DateTime.now.offset.numerator)
music_sessions.length.should == 0
# but let's make it tomorrow, so we can test in that direction
music_session_1.scheduled_start = 1.day.from_now
music_session_1.save!
music_sessions, user_search = sms(searcher_1, client_id: searcher_conn_1.client_id, day: (Date.today + 1).to_s, timezone_offset: DateTime.now.offset.numerator)
music_sessions.length.should == 1
music_sessions[0].should == music_session_1
end
end
describe "scheduled_rsvp" do
let(:creator_1) { FactoryGirl.create(:user) }
let!(:music_session_1) { FactoryGirl.create(:music_session, :creator => creator_1, description: "Bunny Jumps", :create_type => MusicSession::CREATE_TYPE_IMMEDIATE ) }
it "lists one" do
MusicSession.scheduled_rsvp(creator_1).should == [music_session_1]
end
it "does not list canceled" do
music_session_1.canceled = true
music_session_1.save!
MusicSession.scheduled_rsvp(creator_1).should == []
end
it "does not list old" do
music_session_1.scheduled_start = 5.hours.ago
music_session_1.save!
MusicSession.scheduled_rsvp(creator_1).should == []
end
it "only show approved sessions" do
MusicSession.scheduled_rsvp(creator_1, true).should == [music_session_1]
end
it "does not show unchosen" do
music_session_1.rsvp_slots[0].rsvp_requests_rsvp_slots[0].chosen = false
music_session_1.rsvp_slots[0].rsvp_requests_rsvp_slots[0].save!
MusicSession.scheduled_rsvp(creator_1, true).should == []
end
it "create_type = nil will still return RSVPs" do
music_session_1.create_type = nil
music_session_1.save!
MusicSession.scheduled_rsvp(creator_1, true).should == [music_session_1]
end
end
end
describe "scheduled session rescheduled logic" do
it "detect change to scheduling info" do
music_session1.description = "Hey!"
music_session1.save!
music_session1.scheduling_info_changed.should be_false
music_session1.scheduled_start = Time.now - 1.days
music_session1.save!
music_session1.scheduling_info_changed.should be_true
end
end
describe "html_sanitize" do
it "sanitizes" do
music_session1.name = '<b>dog</b>'
music_session1.description = '<html>cat</html>'
music_session1.save!
music_session1.name.should == 'dog'
music_session1.description.should == 'cat'
end
end
describe "purgeable sessions " do
it 'selects unscheduled sessions past due date' do
interval = MusicSession::UNSTARTED_INTERVAL_DAYS_PURGE
dd = Time.now - (interval.to_i + 1).days
Timecop.travel(dd)
msess1 = FactoryGirl.create(:music_session)
msess2 = FactoryGirl.create(:music_session)
purging = MusicSession.purgeable_sessions
expect(purging.size).to be(1)
expect(purging[0].id).to eq(msess1.id)
end
it 'selects recurring and non-recurring sessions past due date' do
[MusicSession::UNSTARTED_INTERVAL_DAYS_PURGE,
MusicSession::UNSTARTED_INTERVAL_DAYS_PURGE_RECUR].each do |interval|
dd = Time.now - (interval.to_i + 1).days
Timecop.travel(dd)
msess1 = FactoryGirl.create(:music_session, scheduled_start: Time.now)
msess2 = FactoryGirl.create(:music_session, scheduled_start: Time.now)
purging = MusicSession.purgeable_sessions
expect(purging.size).to be(1)
expect(purging[0].id).to eq(msess1.id)
MusicSession.delete_all
end
end
end
end