719ef1377a
Conflicts: akka-bbb-apps/src/main/scala/org/bigbluebutton/core/MeetingActor.scala akka-bbb-apps/src/main/scala/org/bigbluebutton/core/api/ValueObjects.scala akka-bbb-apps/src/main/scala/org/bigbluebutton/core/apps/UsersApp.scala bbb-video/src/main/java/org/bigbluebutton/app/video/VideoApplication.java bbb-video/src/main/java/org/bigbluebutton/app/video/converter/VideoRotator.java bigbluebutton-client/locale/en_US/bbbResources.properties bigbluebutton-client/locale/pt_BR/bbbResources.properties bigbluebutton-client/resources/prod/MconfLive.html bigbluebutton-client/src/org/bigbluebutton/common/Images.as bigbluebutton-client/src/org/bigbluebutton/modules/deskshare/utils/BrowserCheck.as bigbluebutton-client/src/org/bigbluebutton/modules/deskshare/utils/JavaCheck.as bigbluebutton-client/src/org/bigbluebutton/modules/deskshare/view/components/DesktopPublishWindow.mxml bigbluebutton-client/src/org/bigbluebutton/modules/present/ui/views/PresentationWindow.mxml bigbluebutton-client/src/org/bigbluebutton/modules/users/views/UsersWindow.mxml bigbluebutton-config/bin/bbb-conf record-and-playback/core/lib/recordandplayback/generators/audio_processor.rb record-and-playback/core/lib/recordandplayback/generators/video.rb record-and-playback/core/scripts/rap-worker.rb record-and-playback/presentation/playback/presentation/0.9.0/lib/writing.js record-and-playback/presentation/scripts/process/presentation.rb record-and-playback/presentation/scripts/publish/presentation.rb
211 lines
5.9 KiB
Ruby
Executable File
211 lines
5.9 KiB
Ruby
Executable File
# Set encoding to utf-8
|
|
# encoding: UTF-8
|
|
|
|
#
|
|
# BigBlueButton open source conferencing system - http://www.bigbluebutton.org/
|
|
#
|
|
# Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below).
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify it under the
|
|
# terms of the GNU Lesser General Public License as published by the Free Software
|
|
# Foundation; either version 3.0 of the License, or (at your option) any later
|
|
# version.
|
|
#
|
|
# BigBlueButton 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 Lesser General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU Lesser General Public License along
|
|
# with BigBlueButton; if not, see <http://www.gnu.org/licenses/>.
|
|
#
|
|
|
|
|
|
path = File.expand_path(File.join(File.dirname(__FILE__), '../lib'))
|
|
$LOAD_PATH << path
|
|
|
|
require 'recordandplayback/audio_archiver'
|
|
require 'recordandplayback/events_archiver'
|
|
require 'recordandplayback/video_archiver'
|
|
require 'recordandplayback/presentation_archiver'
|
|
require 'recordandplayback/deskshare_archiver'
|
|
require 'recordandplayback/generators/events'
|
|
require 'recordandplayback/generators/audio'
|
|
require 'recordandplayback/generators/video'
|
|
require 'recordandplayback/generators/audio_processor'
|
|
require 'recordandplayback/generators/presentation'
|
|
require 'custom_hash'
|
|
require 'open4'
|
|
require 'pp'
|
|
require 'absolute_time'
|
|
require 'logger'
|
|
require 'find'
|
|
require 'rubygems'
|
|
|
|
module BigBlueButton
|
|
class MissingDirectoryException < RuntimeError
|
|
end
|
|
|
|
class FileNotFoundException < RuntimeError
|
|
end
|
|
|
|
class ExecutionStatus
|
|
def initialize
|
|
@output = []
|
|
@errors = []
|
|
@detailedStatus = nil
|
|
end
|
|
|
|
attr_accessor :output
|
|
attr_accessor :errors
|
|
attr_accessor :detailedStatus
|
|
|
|
def success?
|
|
@detailedStatus.success?
|
|
end
|
|
|
|
def exited?
|
|
@detailedStatus.exited?
|
|
end
|
|
|
|
def exitstatus
|
|
@detailedStatus.exitstatus
|
|
end
|
|
end
|
|
|
|
# BigBlueButton logs information about its progress.
|
|
# Replace with your own logger if you desire.
|
|
#
|
|
# @param [Logger] log your own logger
|
|
# @return [Logger] the logger you set
|
|
def self.logger=(log)
|
|
@logger = log
|
|
end
|
|
|
|
# Get BigBlueButton logger.
|
|
#
|
|
# @return [Logger]
|
|
def self.logger
|
|
return @logger if @logger
|
|
logger = Logger.new(STDOUT)
|
|
logger.level = Logger::INFO
|
|
@logger = logger
|
|
end
|
|
|
|
def self.redis_publisher=(publisher)
|
|
@redis_publisher = publisher
|
|
end
|
|
|
|
def self.redis_publisher
|
|
return @redis_publisher
|
|
end
|
|
|
|
def self.dir_exists?(dir)
|
|
FileTest.directory?(dir)
|
|
end
|
|
|
|
def self.execute(command, fail_on_error=true)
|
|
status = ExecutionStatus.new
|
|
status.detailedStatus = Open4::popen4(command) do | pid, stdin, stdout, stderr|
|
|
BigBlueButton.logger.info("Executing: #{command}")
|
|
|
|
status.output = stdout.readlines
|
|
BigBlueButton.logger.info( "Output: #{Array(status.output).join()} ") unless status.output.empty?
|
|
|
|
status.errors = stderr.readlines
|
|
unless status.errors.empty?
|
|
BigBlueButton.logger.error( "Error: stderr: #{Array(status.errors).join()}")
|
|
end
|
|
end
|
|
BigBlueButton.logger.info("Success?: #{status.success?}")
|
|
BigBlueButton.logger.info("Process exited? #{status.exited?}")
|
|
BigBlueButton.logger.info("Exit status: #{status.exitstatus}")
|
|
if status.success? == false and fail_on_error
|
|
raise "Execution failed"
|
|
end
|
|
status
|
|
end
|
|
|
|
def self.exec_ret(*command)
|
|
BigBlueButton.logger.info "Executing: #{command.join(' ')}"
|
|
IO.popen([*command, :err => [:child, :out]]) do |io|
|
|
io.each_line do |line|
|
|
BigBlueButton.logger.info line.chomp
|
|
end
|
|
end
|
|
BigBlueButton.logger.info "Exit status: #{$?.exitstatus}"
|
|
return $?.exitstatus
|
|
end
|
|
|
|
def self.exec_redirect_ret(outio, *command)
|
|
BigBlueButton.logger.info "Executing: #{command.join(' ')}"
|
|
BigBlueButton.logger.info "Sending output to #{outio}"
|
|
IO.pipe do |r, w|
|
|
pid = spawn(*command, :out => outio, :err => w)
|
|
w.close
|
|
r.each_line do |line|
|
|
BigBlueButton.logger.info line.chomp
|
|
end
|
|
Process.waitpid(pid)
|
|
BigBlueButton.logger.info "Exit status: #{$?.exitstatus}"
|
|
return $?.exitstatus
|
|
end
|
|
end
|
|
|
|
def self.hash_to_str(hash)
|
|
return PP.pp(hash, "")
|
|
end
|
|
|
|
def self.monotonic_clock()
|
|
return (AbsoluteTime.now * 1000).to_i
|
|
end
|
|
|
|
def self.get_dir_size(dir_name)
|
|
size = 0
|
|
if FileTest.directory?(dir_name)
|
|
Find.find(dir_name) { |f| size += File.size(f) }
|
|
end
|
|
size.to_s
|
|
end
|
|
|
|
def self.add_tag_to_xml(xml_filename, parent_xpath, tag, content)
|
|
if File.exist? xml_filename
|
|
doc = Nokogiri::XML(File.open(xml_filename)) { |x| x.noblanks }
|
|
|
|
node = doc.at_xpath("#{parent_xpath}/#{tag}")
|
|
node.remove if not node.nil?
|
|
|
|
node = Nokogiri::XML::Node.new tag, doc
|
|
node.content = content
|
|
|
|
doc.at(parent_xpath) << node
|
|
|
|
xml_file = File.new(xml_filename, "w")
|
|
xml_file.write(doc.to_xml(:indent => 2))
|
|
xml_file.close
|
|
end
|
|
end
|
|
|
|
def self.add_raw_size_to_metadata(dir_name, raw_dir_name)
|
|
size = BigBlueButton.get_dir_size(raw_dir_name)
|
|
BigBlueButton.add_tag_to_xml("#{dir_name}/metadata.xml", "//recording", "raw_size", size)
|
|
end
|
|
|
|
def self.add_playback_size_to_metadata(dir_name)
|
|
size = BigBlueButton.get_dir_size(dir_name)
|
|
BigBlueButton.add_tag_to_xml("#{dir_name}/metadata.xml", "//recording/playback", "size", size)
|
|
end
|
|
|
|
def self.add_download_size_to_metadata(dir_name)
|
|
size = BigBlueButton.get_dir_size(dir_name)
|
|
BigBlueButton.add_tag_to_xml("#{dir_name}/metadata.xml", "//recording/download", "size", size)
|
|
end
|
|
|
|
def self.record_id_to_timestamp(r)
|
|
r.split("-")[1].to_i / 1000
|
|
end
|
|
|
|
def self.done_to_timestamp(r)
|
|
BigBlueButton.record_id_to_timestamp(File.basename(r, ".done"))
|
|
end
|
|
end
|