canvas-lms/app/models/media_object.rb

332 lines
12 KiB
Ruby

#
# Copyright (C) 2011 Instructure, Inc.
#
# This file is part of Canvas.
#
# Canvas is free software: you can redistribute it and/or modify it under
# the terms of the GNU Affero General Public License as published by the Free
# Software Foundation, version 3 of the License.
#
# Canvas is distributed in the hope that it will be useful, but WITHOUT ANY
# WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
# A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
# details.
#
# You should have received a copy of the GNU Affero General Public License along
# with this program. If not, see <http://www.gnu.org/licenses/>.
#
class MediaObject < ActiveRecord::Base
include Workflow
belongs_to :user
belongs_to :context, :polymorphic => true
belongs_to :attachment
belongs_to :root_account, :class_name => 'Account'
validates_presence_of :media_id, :workflow_state
has_many :media_tracks, :dependent => :destroy, :order => 'locale'
after_create :retrieve_details_later
after_save :update_title_on_kaltura_later
serialize :data
attr_accessible :media_id, :title, :context, :user
attr_accessor :podcast_associated_asset
def user_entered_title=(val)
@push_user_title = true
write_attribute(:user_entered_title, val)
end
def update_title_on_kaltura_later
send_later(:update_title_on_kaltura) if @push_user_title
@push_user_title = nil
end
def self.find_by_media_id(media_id)
unless Rails.env.production?
raise "Do not look up MediaObjects by media_id - use the scope by_media_id instead to support migrated content."
end
super
end
set_policy do
given { |user| (self.user && self.user == user) || (self.context && self.context.grants_right?(user, nil, :manage_content)) }
can :add_captions and can :delete_captions
end
# if wait_for_completion is true, this will wait SYNCHRONOUSLY for the bulk
# upload to complete. Wrap it in a timeout if you ever want it to give up
# waiting.
def self.add_media_files(attachments, wait_for_completion)
return unless Kaltura::ClientV3.config
attachments = Array(attachments)
client = Kaltura::ClientV3.new
client.startSession(Kaltura::SessionType::ADMIN)
files = []
root_account_id = attachments.map{|a| a.root_account_id }.compact.first
attachments.select{|a| !a.media_object }.each do |attachment|
pseudonym = attachment.user.sis_pseudonym_for(attachment.context) if attachment.user
sis_source_id, sis_user_id = "", ""
if Kaltura::ClientV3.config['kaltura_sis'].present? && Kaltura::ClientV3.config['kaltura_sis'] == "1"
sis_source_id = %Q[,"sis_source_id":"#{attachment.context.sis_source_id}"] if attachment.context.respond_to?('sis_source_id') && attachment.context.sis_source_id
sis_user_id = %Q[,"sis_user_id":"#{pseudonym ? pseudonym.sis_user_id : ''}"] if pseudonym
end
files << {
:name => attachment.display_name,
:url => attachment.cacheable_s3_download_url,
:media_type => (attachment.content_type || "").match(/\Avideo/) ? 'video' : 'audio',
:partner_data => %Q[{"attachment_id":"#{attachment.id}","context_source":"file_upload" #{sis_source_id} #{sis_user_id}}]
}
end
res = client.bulkUploadAdd(files)
if !res[:ready]
if wait_for_completion
bulk_upload_id = res[:id]
Rails.logger.debug "waiting for bulk upload id: #{bulk_upload_id}"
started_at = Time.now
timeout = Setting.get('media_bulk_upload_timeout', 30.minutes.to_s).to_i
while !res[:ready]
if Time.now > started_at + timeout
MediaObject.send_later_enqueue_args(:refresh_media_files, {:run_at => 1.minute.from_now, :priority => Delayed::LOW_PRIORITY}, res[:id], attachments.map(&:id), root_account_id)
break
end
sleep(1.minute.to_i)
res = client.bulkUploadGet(bulk_upload_id)
end
else
MediaObject.send_later_enqueue_args(:refresh_media_files, {:run_at => 1.minute.from_now, :priority => Delayed::LOW_PRIORITY}, res[:id], attachments.map(&:id), root_account_id)
end
end
if res[:ready]
build_media_objects(res, root_account_id)
end
res
end
def self.bulk_migration(csv, root_account_id)
client = Kaltura::ClientV3.new
client.startSession(Kaltura::SessionType::ADMIN)
res = client.bulkUploadCsv(csv)
if !res[:ready]
MediaObject.send_later_enqueue_args(:refresh_media_files, {:run_at => 1.minute.from_now, :priority => Delayed::LOW_PRIORITY}, res[:id], [], root_account_id)
else
build_media_objects(res, root_account_id)
end
res
end
def self.migration_csv(media_objects)
CSV.generate do |csv|
media_objects.each do |mo|
mo.retrieve_details unless mo.data[:download_url]
if mo.data[:download_url]
row = []
row << mo.title
row << ""
row << "old_id_#{mo.media_id}"
row << mo.data[:download_url]
row << mo.media_type.capitalize
csv << row
end
end
end
end
def self.build_media_objects(data, root_account_id)
root_account = Account.find_by_id(root_account_id)
data[:entries].each do |entry|
attachment_id = nil
if entry[:originalId].present? && (Integer(entry[:originalId]).is_a?(Integer) rescue false)
attachment_id = entry[:originalId]
elsif entry[:originalId].present? && entry[:originalId].length >= 2
partner_data = JSON.parse(entry[:originalId]).with_indifferent_access
attachment_id = partner_data[:attachment_id] if partner_data[:attachment_id].present?
end
attachment = Attachment.find_by_id(attachment_id) if attachment_id
mo = MediaObject.find_or_initialize_by_media_id(entry[:entryId])
mo.root_account ||= root_account || Account.default
mo.title ||= entry[:name]
if attachment
mo.user_id ||= attachment.user_id
mo.context = attachment.context
mo.attachment_id = attachment.id
attachment.update_attribute(:media_entry_id, entry[:entryId])
# check for attachments that were created temporarily, just to import a media object
if attachment.full_path.starts_with?(File.join(Folder::ROOT_FOLDER_NAME, CC::CCHelper::MEDIA_OBJECTS_FOLDER) + '/')
attachment.destroy
end
end
mo.context ||= mo.root_account
mo.save
end
end
def self.refresh_media_files(bulk_upload_id, attachment_ids, root_account_id, attempt=0)
client = Kaltura::ClientV3.new
client.startSession(Kaltura::SessionType::ADMIN)
res = client.bulkUploadGet(bulk_upload_id)
if !res[:ready]
if attempt < Setting.get('media_object_bulk_refresh_max_attempts', '5').to_i
wait_period = Setting.get('media_object_bulk_refresh_wait_period', '30').to_i
MediaObject.send_later_enqueue_args(:refresh_media_files, {:run_at => wait_period.minutes.from_now, :priority => Delayed::LOW_PRIORITY}, bulk_upload_id, attachment_ids, root_account_id, attempt + 1)
else
# if it fails, then the attachment should no longer consider itself kalturable
Attachment.where("id IN (?) OR root_attachment_id IN (?)", attachment_ids, attachment_ids).update_all(:media_entry_id => nil) unless attachment_ids.empty?
end
res
else
build_media_objects(res, root_account_id)
end
end
def self.media_id_exists?(media_id)
client = Kaltura::ClientV3.new
client.startSession(Kaltura::SessionType::ADMIN)
info = client.mediaGet(media_id)
return !!info[:id]
end
def self.ensure_media_object(media_id, create_opts = {})
if !by_media_id(media_id).any?
self.send_later_enqueue_args(:create_if_id_exists, { :priority => Delayed::LOW_PRIORITY }, media_id, create_opts)
end
end
# typically call this in a delayed job, since it has to contact kaltura
def self.create_if_id_exists(media_id, create_opts = {})
if media_id_exists?(media_id) && !by_media_id(media_id).any?
create!(create_opts.merge(:media_id => media_id))
end
end
def update_title_on_kaltura
client = Kaltura::ClientV3.new
client.startSession(Kaltura::SessionType::ADMIN)
res = client.mediaUpdate(self.media_id, :name => self.user_entered_title)
if !res[:error]
self.title = self.user_entered_title
self.save
end
res
end
def retrieve_details_later
send_later(:retrieve_details_ensure_codecs)
end
def media_sources
Kaltura::ClientV3.new.media_sources(self.media_id)
end
def retrieve_details_ensure_codecs(attempt=0)
retrieve_details
if (!self.data || !self.data[:extensions] || !self.data[:extensions][:flv]) && self.created_at > 6.hours.ago
if(attempt < 10)
send_at((5 * attempt).minutes.from_now, :retrieve_details_ensure_codecs, attempt + 1)
else
ErrorReport.log_error(:default, {
:message => "Kaltura flavor retrieval failed",
:object => self.inspect.to_s,
})
end
end
end
def name
self.title
end
def retrieve_details
return unless self.media_id
# From Kaltura, retrieve the title (if it's not already set)
# and the list of valid flavors along with their id's.
# Might as well confirm the media type while you're at it.
client = Kaltura::ClientV3.new
client.startSession(Kaltura::SessionType::ADMIN)
self.data ||= {}
entry = client.mediaGet(self.media_id)
if entry
self.title = entry[:name]
self.media_type = client.mediaTypeToSymbol(entry[:mediaType]).to_s
self.duration = entry[:duration].to_i
self.data[:plays] = entry[:plays].to_i
self.data[:download_url] = entry[:downloadUrl]
tags = (entry[:tags] || "").split(/,/).map(&:strip)
old_id = tags.detect{|t| t.match(/old_id_/) }
self.old_media_id = old_id.sub(/old_id_/, '') if old_id
end
assets = client.flavorAssetGetByEntryId(self.media_id)
self.data[:extensions] ||= {}
assets.each do |asset|
asset[:fileExt] = "none" if asset[:fileExt].blank?
self.data[:extensions][asset[:fileExt].to_sym] = asset #.slice(:width, :height, :id, :entryId, :status, :containerFormat, :fileExt, :size
if asset[:size]
self.max_size = [self.max_size || 0, asset[:size].to_i].max
end
end
self.total_size = [self.max_size || 0, assets.map{|a| (a[:size] || 0).to_i }.sum].max
self.save
self.data
end
def podcast_format_details
data = self.data && self.data[:extensions] && self.data[:extensions][:mp3]
data ||= self.data && self.data[:extensions] && self.data[:extensions][:mp4]
if !data
self.retrieve_details
data ||= self.data && self.data[:extensions] && self.data[:extensions][:mp3]
data ||= self.data && self.data[:extensions] && self.data[:extensions][:mp4]
end
data
end
def delete_from_remote
return unless self.media_id
client = Kaltura::ClientV3.new
client.startSession(Kaltura::SessionType::ADMIN)
client.mediaDelete(self.media_id)
end
alias_method :destroy!, :destroy
def destroy
self.workflow_state = 'deleted'
self.attachment.destroy if self.attachment
save!
end
def data
self.read_attribute(:data) || self.write_attribute(:data, {})
end
def viewed!
send_later(:updated_viewed_at_and_retrieve_details, Time.now) if !self.data[:last_viewed_at] || self.data[:last_viewed_at] > 1.hour.ago
true
end
def updated_viewed_at_and_retrieve_details(time)
self.data[:last_viewed_at] = [time, self.data[:last_viewed_at]].compact.max
self.retrieve_details
end
def destroy_without_destroying_attachment
self.workflow_state = 'deleted'
self.attachment_id = nil
save!
end
scope :active, where("media_objects.workflow_state<>'deleted'")
scope :by_media_id, lambda { |media_id| where("media_objects.media_id=? OR media_objects.old_media_id=?", media_id, media_id) }
scope :by_media_type, lambda { |media_type| where(:media_type => media_type) }
workflow do
state :active
state :deleted
end
end