class Room < ApplicationRecord before_create :setup validates :name, presence: true belongs_to :owner, class_name: 'User', foreign_key: :user_id RETURNCODE_SUCCESS = "SUCCESS" META_LISTED = "gl-listed" UID_LENGTH = 6 # Determines if a user owns a room. def owned_by?(user) return false if user.nil? user.rooms.include?(self) end # Checks if a room is running on the BigBlueButton server. def is_running? bbb.is_meeting_running?(bbb_id) end # Determines the invite URL for the room. def invite_path "#{Rails.configuration.relative_url_root}/#{uid}" end # Creates a meeting on the BigBlueButton server. def start_session(options = {}) create_options = { record: options[:meeting_recorded].to_s, logoutURL: options[:meeting_logout_url] || '', moderatorPW: random_password(12), attendeePW: random_password(12), moderatorOnlyMessage: options[:moderator_message], "meta_#{META_LISTED}": false } # Update session info. self.update_attributes(sessions: sessions + 1, last_session: DateTime.now) #meeting_options.merge!( #{ "meta_room-id": options[:room_owner], # "meta_meeting-name": options[:meeting_name]} #) if options[:room_owner] # Send the create request. begin bbb.create_meeting(name, bbb_id, create_options) rescue BigBlueButton::BigBlueButtonException => exc puts "BigBlueButton failed on create: #{exc.key}: #{exc.message}" end end # Returns a URL to join a user into a meeting. def join_path(name, options = {}, uid = nil) # Create the meeting if it isn't running. start_session(options) unless is_running? # Set meeting options. options[:meeting_logout_url] ||= nil options[:moderator_message] ||= '' options[:user_is_moderator] ||= false options[:meeting_recorded] ||= false return call_invalid_res if !bbb # Get the meeting info. meeting_info = bbb.get_meeting_info(bbb_id, nil) # Determine the password to use when joining. password = if options[:user_is_moderator] meeting_info[:moderatorPW] else meeting_info[:attendeePW] end # Generate the join URL. join_opts = {} join_opts.merge!({userID: uid}) if uid join_opts.merge!({joinViaHtml5: true}) if Rails.configuration.html5_enabled bbb.join_meeting_url(bbb_id, name, password, join_opts) end # Notify waiting users that a meeting has started. def notify_waiting ActionCable.server.broadcast("#{uid}_waiting_channel", { action: "started" }) end # Retrieves all the users in a room. def participants begin res = bbb.get_meeting_info(bbb_id, nil) res[:attendees].map do |att| User.find_by(uid: att[:userID], name: att[:fullName]) end rescue BigBlueButton::BigBlueButtonException => exc # The meeting is most likely not running. [] end end # Fetches all recordings for a room. def recordings res = bbb.get_recordings(meetingID: bbb_id) # Format playbacks in a more pleasant way. res[:recordings].each do |r| next if r.key?(:error) r[:playbacks] = if !r[:playback] || !r[:playback][:format] [] elsif r[:playback][:format].is_a?(Array) r[:playback][:format] else [r[:playback][:format]] end r.delete(:playback) end res[:recordings] end # Fetches a rooms public recordings. def public_recordings recordings.select do |r| r[:metadata][:"gl-listed"] == "true" end end def update_recording(record_id, meta) meta.merge!({recordID: record_id}) bbb.send_api_request("updateRecordings", meta) end # Deletes a recording from a room. def delete_recording(record_id) bbb.delete_recordings(record_id) end private def bbb_endpoint Rails.configuration.bigbluebutton_endpoint end def bbb_secret Rails.configuration.bigbluebutton_secret end # Sets a BigBlueButtonApi object for interacting with the API. def bbb @bbb ||= BigBlueButton::BigBlueButtonApi.new(remove_slash(bbb_endpoint), bbb_secret, "0.8") #@bbb ||= if Rails.configuration.loadbalanced_configuration # lb_user = retrieve_loadbalanced_credentials(self.room.owner.provider) # BigBlueButton::BigBlueButtonApi.new(remove_slash(lb_user["apiURL"]), lb_user["secret"], "0.8") #else # BigBlueButton::BigBlueButtonApi.new(remove_slash(bbb_endpoint), bbb_secret, "0.8") #end end # Generates a uid for the room and BigBlueButton. def setup self.uid = random_room_uid self.bbb_id = Digest::SHA1.hexdigest(Rails.application.secrets[:secret_key_base] + Time.now.to_i.to_s).to_s end # Generates a random room uid that uses the users name. def random_room_uid charset = %w{ 2 3 4 6 7 9 a c d e f g h j k m n p q r t v w x y z} owner.firstname.downcase + "-" + (0...UID_LENGTH).map{ charset.to_a[rand(charset.size)] }.join.insert(UID_LENGTH / 2, "-") end # Rereives the loadbalanced BigBlueButton credentials for a user. def retrieve_loadbalanced_credentials(provider) # Include Omniauth accounts under the Greenlight provider. provider = "greenlight" if Rails.configuration.providers.include?(provider.to_sym) # Build the URI. uri = encode_bbb_url( Rails.configuration.loadbalancer_endpoint, Rails.configuration.loadbalancer_secret, {name: provider} ) # Make the request. http = Net::HTTP.new(uri.host, uri.port) http.use_ssl = (uri.scheme == 'https') response = http.get(uri.request_uri) unless response.kind_of?(Net::HTTPSuccess) raise "Error retrieving provider credentials: #{response.code} #{response.message}" end # Parse XML. doc = XmlSimple.xml_in(response.body, 'ForceArray' => false) # Return the user credentials if the request succeeded on the loadbalancer. return doc['user'] if doc['returncode'] == RETURNCODE_SUCCESS raise "User with provider #{provider} does not exist." if doc['messageKey'] == "noSuchUser" raise "API call #{url} failed with #{doc['messageKey']}." end # Builds a request to retrieve credentials from the load balancer. def encode_bbb_url(base_url, secret, params) encoded_params = OAuth::Helper.normalize(params) string = "getUser" + encoded_params + secret checksum = OpenSSL::Digest.digest('sha1', string).unpack("H*").first URI.parse("#{base_url}?#{encoded_params}&checksum=#{checksum}") end # Removes trailing forward slash from a URL. def remove_slash(s) s.nil? ? nil : s.chomp("/") end # Generates a random password for a meeting. def random_password(length) o = ([('a'..'z'), ('A'..'Z')].map do |i| i.to_a end).flatten ((0...length).map do o[rand(o.length)] end).join end end