canvas-lms/app/models/media_object.rb

319 lines
11 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'
has_many :media_tracks, :dependent => :destroy, :order => 'locale'
validates_presence_of :media_id, :context_id, :context_type
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 }
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|
files << {
:name => attachment.display_name,
:url => attachment.cacheable_s3_download_url,
:media_type => (attachment.content_type || "").match(/\Avideo/) ? 'video' : 'audio',
:id => attachment.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}"
while !res[:ready]
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)
FasterCSV.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 = Attachment.find_by_id(entry[:originalId]) if entry[:originalId].present?
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(false)
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.update_all({:media_entry_id => nil}, "id IN (#{attachment_ids.join(",")}) OR root_attachment_id IN (#{attachment_ids.join(",")})") 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 media_sources
Kaltura::ClientV3.new.media_sources(self.media_id)
end
def retrieve_details_later
send_later(:retrieve_details_ensure_codecs)
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
named_scope :active, lambda{
{:conditions => ['media_objects.workflow_state != ?', 'deleted'] }
}
named_scope :by_media_id, lambda { |media_id|
{ :conditions => [ 'media_objects.media_id = ? OR media_objects.old_media_id = ?', media_id, media_id ] }
}
named_scope :by_media_type, lambda { |media_type|
{ :conditions => [ 'media_objects.media_type = ?', media_type ]}
}
workflow do
state :active
state :deleted
end
end