You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
842 lines
35 KiB
842 lines
35 KiB
<?php
|
|
/* For licensing terms, see /license.txt */
|
|
|
|
/**
|
|
* Class bbb
|
|
* This script initiates a video conference session, calling the BigBlueButton
|
|
* API
|
|
* @package chamilo.plugin.bigbluebutton
|
|
*
|
|
* BigBlueButton-Chamilo connector class
|
|
*/
|
|
//namespace Chamilo\Plugin\BBB;
|
|
|
|
/**
|
|
* Class bbb
|
|
* @package Chamilo\Plugin\BBB
|
|
*/
|
|
class bbb
|
|
{
|
|
public $url;
|
|
public $salt;
|
|
public $api;
|
|
public $user_complete_name = null;
|
|
public $protocol = 'http://';
|
|
public $debug = false;
|
|
public $logout_url = null;
|
|
public $plugin_enabled = false;
|
|
|
|
/**
|
|
*
|
|
* Constructor (generates a connection to the API and the Chamilo settings
|
|
* required for the connection to the video conference server)
|
|
* @param string $host
|
|
* @param string $salt
|
|
*/
|
|
public function __construct($host = null, $salt = null)
|
|
{
|
|
// Initialize video server settings from global settings
|
|
$plugin = BBBPlugin::create();
|
|
|
|
$bbb_plugin = $plugin->get('tool_enable');
|
|
|
|
if (empty($host)) {
|
|
$bbb_host = $plugin->get('host');
|
|
} else {
|
|
$bbb_host = $host;
|
|
}
|
|
if (empty($salt)) {
|
|
$bbb_salt = $plugin->get('salt');
|
|
} else {
|
|
$bbb_salt = $salt;
|
|
}
|
|
|
|
$this->logout_url = api_get_path(WEB_PLUGIN_PATH).'bbb/listing.php?'.api_get_cidreq();
|
|
$this->table = Database::get_main_table('plugin_bbb_meeting');
|
|
|
|
if ($bbb_plugin == true) {
|
|
$userInfo = api_get_user_info();
|
|
$this->user_complete_name = $userInfo['complete_name'];
|
|
$this->salt = $bbb_salt;
|
|
$info = parse_url($bbb_host);
|
|
$this->url = $bbb_host.'/bigbluebutton/';
|
|
if (isset($info['scheme'])) {
|
|
$this->protocol = $info['scheme'].'://';
|
|
$this->url = str_replace($this->protocol, '', $this->url);
|
|
}
|
|
|
|
// Setting BBB api
|
|
define('CONFIG_SECURITY_SALT', $this->salt);
|
|
define('CONFIG_SERVER_BASE_URL', $this->url);
|
|
|
|
$this->api = new BigBlueButtonBN();
|
|
$this->plugin_enabled = true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Checks whether a user is teacher in the current course
|
|
* @return bool True if the user can be considered a teacher in this course, false otherwise
|
|
*/
|
|
public function isTeacher()
|
|
{
|
|
return api_is_course_admin() || api_is_coach() || api_is_platform_admin();
|
|
}
|
|
|
|
/**
|
|
* See this file in you BBB to set up default values
|
|
* @param array $params Array of parameters that will be completed if not containing all expected variables
|
|
|
|
/var/lib/tomcat6/webapps/bigbluebutton/WEB-INF/classes/bigbluebutton.properties
|
|
*
|
|
More record information:
|
|
http://code.google.com/p/bigbluebutton/wiki/RecordPlaybackSpecification
|
|
|
|
# Default maximum number of users a meeting can have.
|
|
# Doesn't get enforced yet but is the default value when the create
|
|
# API doesn't pass a value.
|
|
defaultMaxUsers=20
|
|
|
|
# Default duration of the meeting in minutes.
|
|
# Current default is 0 (meeting doesn't end).
|
|
defaultMeetingDuration=0
|
|
|
|
# Remove the meeting from memory when the end API is called.
|
|
# This allows 3rd-party apps to recycle the meeting right-away
|
|
# instead of waiting for the meeting to expire (see below).
|
|
removeMeetingWhenEnded=false
|
|
|
|
# The number of minutes before the system removes the meeting from memory.
|
|
defaultMeetingExpireDuration=1
|
|
|
|
# The number of minutes the system waits when a meeting is created and when
|
|
# a user joins. If after this period, a user hasn't joined, the meeting is
|
|
# removed from memory.
|
|
defaultMeetingCreateJoinDuration=5
|
|
*
|
|
*/
|
|
public function createMeeting($params)
|
|
{
|
|
$params['c_id'] = api_get_course_int_id();
|
|
$courseCode = api_get_course_id();
|
|
$params['session_id'] = api_get_session_id();
|
|
|
|
$params['attendee_pw'] = isset($params['moderator_pw']) ? $params['moderator_pw'] : api_get_course_id();
|
|
$attendeePassword = $params['attendee_pw'];
|
|
$params['moderator_pw'] = isset($params['moderator_pw']) ? $params['moderator_pw'] : $this->getModMeetingPassword();
|
|
$moderatorPassword = $params['moderator_pw'];
|
|
|
|
$params['record'] = api_get_course_setting('big_blue_button_record_and_store', $courseCode) == 1 ? true : false;
|
|
$max = api_get_course_setting('big_blue_button_max_students_allowed', $courseCode);
|
|
$max = isset($max) ? $max : -1;
|
|
$params['status'] = 1;
|
|
// Generate a pseudo-global-unique-id to avoid clash of conferences on
|
|
// the same BBB server with several Chamilo portals
|
|
$params['remote_id'] = uniqid(true, true);
|
|
|
|
if ($this->debug) {
|
|
error_log("enter create_meeting ".print_r($params, 1));
|
|
}
|
|
|
|
$params['created_at'] = api_get_utc_datetime();
|
|
$id = Database::insert($this->table, $params);
|
|
|
|
if ($id) {
|
|
if ($this->debug) {
|
|
error_log("create_meeting: $id ");
|
|
}
|
|
|
|
$meetingName = isset($params['meeting_name']) ? $params['meeting_name'] : api_get_course_id().'-'.api_get_session_id();
|
|
$welcomeMessage = isset($params['welcome_msg']) ? $params['welcome_msg'] : null;
|
|
$record = isset($params['record']) && $params['record'] ? 'true' : 'false';
|
|
$duration = isset($params['duration']) ? intval($params['duration']) : 0;
|
|
// This setting currently limits the maximum conference duration,
|
|
// to avoid lingering sessions on the video-conference server #6261
|
|
$duration = 300;
|
|
|
|
$bbbParams = array(
|
|
'meetingId' => $params['remote_id'], // REQUIRED
|
|
'meetingName' => $meetingName, // REQUIRED
|
|
'attendeePw' => $attendeePassword, // Match this value in getJoinMeetingURL() to join as attendee.
|
|
'moderatorPw' => $moderatorPassword, // Match this value in getJoinMeetingURL() to join as moderator.
|
|
'welcomeMsg' => $welcomeMessage, // ''= use default. Change to customize.
|
|
'dialNumber' => '', // The main number to call into. Optional.
|
|
'voiceBridge' => '12345', // PIN to join voice. Required.
|
|
'webVoice' => '', // Alphanumeric to join voice. Optional.
|
|
'logoutUrl' => $this->logout_url,
|
|
'maxParticipants' => $max, // Optional. -1 = unlimitted. Not supported in BBB. [number]
|
|
'record' => $record, // New. 'true' will tell BBB to record the meeting.
|
|
'duration' => $duration, // Default = 0 which means no set duration in minutes. [number]
|
|
//'meta_category' => '', // Use to pass additional info to BBB server. See API docs.
|
|
);
|
|
|
|
if ($this->debug) {
|
|
error_log("create_meeting params: ".print_r($bbbParams,1));
|
|
}
|
|
|
|
$status = false;
|
|
$meeting = null;
|
|
|
|
while ($status == false) {
|
|
$result = $this->api->createMeetingWithXmlResponseArray(
|
|
$bbbParams
|
|
);
|
|
if (isset($result) && strval($result['returncode']) == 'SUCCESS'
|
|
) {
|
|
if ($this->debug) {
|
|
error_log(
|
|
"create_meeting result: " . print_r($result, 1)
|
|
);
|
|
}
|
|
$meeting = $this->joinMeeting($meetingName, true);
|
|
|
|
return $meeting;
|
|
}
|
|
}
|
|
return $this->logout;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Tells whether the given meeting exists and is running
|
|
* (using course code as name)
|
|
* @param string $meetingName Meeting name (usually the course code)
|
|
*
|
|
* @return bool True if meeting exists, false otherwise
|
|
* @assert ('') === false
|
|
* @assert ('abcdefghijklmnopqrstuvwxyzabcdefghijklmno') === false
|
|
*/
|
|
public function meetingExists($meetingName)
|
|
{
|
|
if (empty($meetingName)) {
|
|
return false;
|
|
}
|
|
$courseId = api_get_course_int_id();
|
|
$sessionId = api_get_session_id();
|
|
$meetingData = Database::select(
|
|
'*',
|
|
$this->table,
|
|
array(
|
|
'where' => array(
|
|
'c_id = ? AND session_id = ? AND meeting_name = ? AND status = 1 ' =>
|
|
array($courseId, $sessionId, $meetingName)
|
|
)
|
|
),
|
|
'first'
|
|
);
|
|
if ($this->debug) {
|
|
error_log("meeting_exists ".print_r($meetingData, 1));
|
|
}
|
|
if (empty($meetingData)) {
|
|
return false;
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns a meeting "join" URL
|
|
* @param string The name of the meeting (usually the course code)
|
|
* @return mixed The URL to join the meeting, or false on error
|
|
* @todo implement moderator pass
|
|
* @assert ('') === false
|
|
* @assert ('abcdefghijklmnopqrstuvwxyzabcdefghijklmno') === false
|
|
*/
|
|
public function joinMeeting($meetingName, $loop = false)
|
|
{
|
|
if (empty($meetingName)) {
|
|
return false;
|
|
}
|
|
|
|
$pass = $this->getUserMeetingPassword();
|
|
|
|
$meetingData = Database::select(
|
|
'*',
|
|
$this->table,
|
|
array('where' => array('meeting_name = ? AND status = 1 ' => $meetingName)),
|
|
'first'
|
|
);
|
|
|
|
if (empty($meetingData) || !is_array($meetingData)) {
|
|
if ($this->debug) {
|
|
error_log("meeting does not exist: $meetingName");
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
$params = array(
|
|
'meetingId' => $meetingData['remote_id'],
|
|
// -- REQUIRED - The unique id for the meeting
|
|
'password' => $this->getModMeetingPassword()
|
|
// -- REQUIRED - The moderator password for the meeting
|
|
);
|
|
|
|
$status = false;
|
|
$meetingInfoExists = false;
|
|
while ($status == false) {
|
|
|
|
$meetingIsRunningInfo = $this->getMeetingInfo($params);
|
|
if ($meetingIsRunningInfo === false) {
|
|
//checking with the remote_id didn't work, so just in case and
|
|
// to provide backwards support, check with the id
|
|
$params = array(
|
|
'meetingId' => $meetingData['id'],
|
|
// -- REQUIRED - The unique id for the meeting
|
|
'password' => $this->getModMeetingPassword()
|
|
// -- REQUIRED - The moderator password for the meeting
|
|
);
|
|
$meetingIsRunningInfo = $this->getMeetingInfo($params);
|
|
}
|
|
|
|
if ($this->debug) {
|
|
error_log(print_r($meetingIsRunningInfo, 1));
|
|
}
|
|
|
|
if (strval($meetingIsRunningInfo['returncode']) == 'SUCCESS' &&
|
|
isset($meetingIsRunningInfo['meetingName']) &&
|
|
!empty($meetingIsRunningInfo['meetingName'])
|
|
//strval($meetingIsRunningInfo['running']) == 'true'
|
|
) {
|
|
$meetingInfoExists = true;
|
|
}
|
|
|
|
if ($this->debug) {
|
|
error_log(
|
|
"meeting is running: " . intval($meetingInfoExists)
|
|
);
|
|
}
|
|
|
|
if ($meetingInfoExists) {
|
|
$status = true;
|
|
}
|
|
|
|
if ($loop) {
|
|
continue;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ($meetingInfoExists) {
|
|
$joinParams = array(
|
|
'meetingId' => $meetingData['remote_id'], // -- REQUIRED - A unique id for the meeting
|
|
'username' => $this->user_complete_name, //-- REQUIRED - The name that will display for the user in the meeting
|
|
'password' => $pass, //-- REQUIRED - The attendee or moderator password, depending on what's passed here
|
|
//'createTime' => api_get_utc_datetime(), //-- OPTIONAL - string. Leave blank ('') unless you set this correctly.
|
|
'userID' => api_get_user_id(), //-- OPTIONAL - string
|
|
'webVoiceConf' => '' // -- OPTIONAL - string
|
|
);
|
|
$url = $this->api->getJoinMeetingURL($joinParams);
|
|
$url = $this->protocol.$url;
|
|
} else {
|
|
$url = $this->logout_url;
|
|
}
|
|
if ($this->debug) {
|
|
error_log("return url :" . $url);
|
|
}
|
|
return $url;
|
|
}
|
|
|
|
/**
|
|
* Get information about the given meeting
|
|
* @param array ...?
|
|
* @return mixed Array of information on success, false on error
|
|
* @assert (array()) === false
|
|
*/
|
|
public function getMeetingInfo($params)
|
|
{
|
|
try {
|
|
$result = $this->api->getMeetingInfoWithXmlResponseArray($params);
|
|
if ($result == null) {
|
|
if ($this->debug) {
|
|
error_log("Failed to get any response. Maybe we can't contact the BBB server.");
|
|
}
|
|
} else {
|
|
return $result;
|
|
}
|
|
} catch (Exception $e) {
|
|
if ($this->debug) {
|
|
error_log('Caught exception: ', $e->getMessage(), "\n");
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Gets all the course meetings saved in the plugin_bbb_meeting table
|
|
* @return array Array of current open meeting rooms
|
|
*/
|
|
public function getCourseMeetings()
|
|
{
|
|
$pass = $this->getUserMeetingPassword();
|
|
$meetingList = Database::select('*', $this->table, array('where' => array('c_id = ? AND session_id = ? ' => array(api_get_course_int_id(), api_get_session_id()))));
|
|
$newMeetingList = array();
|
|
|
|
$item = array();
|
|
|
|
foreach ($meetingList as $meetingDB) {
|
|
$meetingBBB = $this->getMeetingInfo(array('meetingId' => $meetingDB['remote_id'], 'password' => $pass));
|
|
if ($meetingBBB === false) {
|
|
//checking with the remote_id didn't work, so just in case and
|
|
// to provide backwards support, check with the id
|
|
$params = array(
|
|
'meetingId' => $meetingDB['id'],
|
|
// -- REQUIRED - The unique id for the meeting
|
|
'password' => $pass
|
|
// -- REQUIRED - The moderator password for the meeting
|
|
);
|
|
$meetingBBB = $this->getMeetingInfo($params);
|
|
}
|
|
|
|
if ($meetingDB['visibility'] == 0 and $this->isTeacher() == false) {
|
|
continue;
|
|
}
|
|
$meetingBBB['end_url'] = api_get_self().'?'.api_get_cidreq().'&action=end&id='.$meetingDB['id'];
|
|
|
|
if ((string)$meetingBBB['returncode'] == 'FAILED') {
|
|
if ($meetingDB['status'] == 1 && $this->isTeacher()) {
|
|
$this->endMeeting($meetingDB['id']);
|
|
}
|
|
} else {
|
|
$meetingBBB['add_to_calendar_url'] = api_get_self().'?'.api_get_cidreq().'&action=add_to_calendar&id='.$meetingDB['id'].'&start='.api_strtotime($meetingDB['created_at']);
|
|
}
|
|
|
|
$recordArray = array();
|
|
$actionLinksArray = array();
|
|
|
|
if ($meetingDB['record'] == 1) {
|
|
// backwards compatibility (when there was no remote ID)
|
|
$mId = $meetingDB['remote_id'];
|
|
if (empty($mId)) {
|
|
$mId = $meetingDB['id'];
|
|
}
|
|
if (empty($mId)) {
|
|
// if the id is still empty (should *never* occur as 'id' is
|
|
// the table's primary key), skip this conference
|
|
continue;
|
|
}
|
|
$recordingParams = array(
|
|
'meetingId' => $mId, //-- OPTIONAL - comma separate if multiple ids
|
|
);
|
|
|
|
//To see the recording list in your BBB server do: bbb-record --list
|
|
$records = $this->api->getRecordingsWithXmlResponseArray($recordingParams);
|
|
if (!empty($records)) {
|
|
$count = 1;
|
|
if (isset($records['message']) && !empty($records['message'])) {
|
|
if ($records['messageKey'] == 'noRecordings') {
|
|
$recordArray[] = get_lang('NoRecording');
|
|
if ($meetingDB['visibility'] == 0) {
|
|
$actionLinksArray[] = Display::url(
|
|
Display::return_icon(
|
|
'invisible.png',
|
|
get_lang('MakeVisible'),
|
|
array(),
|
|
ICON_SIZE_MEDIUM
|
|
),
|
|
api_get_self().'?'.
|
|
api_get_cidreq().
|
|
'&action=publish&id='.$meetingDB['id']
|
|
);
|
|
} else {
|
|
$actionLinksArray[] = Display::url(
|
|
Display::return_icon(
|
|
'visible.png',
|
|
get_lang('MakeInvisible'),
|
|
array(),
|
|
ICON_SIZE_MEDIUM
|
|
),
|
|
api_get_self().'?'.
|
|
api_get_cidreq().
|
|
'&action=unpublish&id='.$meetingDB['id']
|
|
);
|
|
}
|
|
} else {
|
|
//$recordArray[] = $records['message'];
|
|
}
|
|
} else {
|
|
foreach ($records as $record) {
|
|
//if you get several recordings here and you used a
|
|
// previous version of Chamilo, you might want to
|
|
// only keep the last result for each chamilo conf
|
|
// (see show_links after the end of this loop)
|
|
if (is_array($record) && isset($record['recordId'])) {
|
|
$url = Display::url(
|
|
get_lang('ViewRecord')." [~".$record['playbackFormatLength']."']",
|
|
$record['playbackFormatUrl'],
|
|
array('target' => '_blank')
|
|
);
|
|
$actionLinks = '';
|
|
if ($this->isTeacher()) {
|
|
$actionLinks .= Display::url(
|
|
Display::return_icon(
|
|
'link.gif',
|
|
get_lang('CopyToLinkTool')
|
|
),
|
|
api_get_self().'?'.
|
|
api_get_cidreq().
|
|
'&action=copy_record_to_link_tool&id='.$meetingDB['id']
|
|
);
|
|
$actionLinks .= Display::url(
|
|
Display::return_icon(
|
|
'agenda.png',
|
|
get_lang('AddToCalendar')
|
|
),
|
|
api_get_self().'?'.
|
|
api_get_cidreq().
|
|
'&action=add_to_calendar&id='.$meetingDB['id'].
|
|
'&start='.api_strtotime($meetingDB['created_at']).
|
|
'&url='.$record['playbackFormatUrl']
|
|
);
|
|
$actionLinks .= Display::url(
|
|
Display::return_icon(
|
|
'delete.png',
|
|
get_lang('Delete')
|
|
),
|
|
api_get_self().'?'.
|
|
api_get_cidreq().
|
|
'&action=delete_record&id='.$record['recordId']
|
|
);
|
|
if ($meetingDB['visibility'] == 0) {
|
|
$actionLinks .= Display::url(
|
|
Display::return_icon(
|
|
'invisible.png',
|
|
get_lang('MakeVisible'),
|
|
array(),
|
|
ICON_SIZE_MEDIUM
|
|
),
|
|
api_get_self().'?'.
|
|
api_get_cidreq().
|
|
'&action=publish&id='.$meetingDB['id']
|
|
);
|
|
} else {
|
|
$actionLinks .= Display::url(
|
|
Display::return_icon(
|
|
'visible.png',
|
|
get_lang('MakeInvisible'),
|
|
array(),
|
|
ICON_SIZE_MEDIUM
|
|
),
|
|
api_get_self().'?'.
|
|
api_get_cidreq().
|
|
'&action=unpublish&id='.$meetingDB['id']
|
|
);
|
|
}
|
|
}
|
|
//$url .= api_get_self().'?action=publish&id='.$record['recordID'];
|
|
$count++;
|
|
$recordArray[] = $url;
|
|
$actionLinksArray[] = $actionLinks;
|
|
} else {
|
|
/*if (is_array($record) && isset($record['recordID']) && isset($record['playbacks'])) {
|
|
|
|
//Fix the bbb timestamp
|
|
//$record['startTime'] = substr($record['startTime'], 0, strlen($record['startTime']) -3);
|
|
//$record['endTime'] = substr($record['endTime'], 0, strlen($record['endTime']) -3);
|
|
//.' - '.api_convert_and_format_date($record['startTime']).' - '.api_convert_and_format_date($record['endTime'])
|
|
foreach($record['playbacks'] as $item) {
|
|
$url = Display::url(get_lang('ViewRecord'), $item['url'], array('target' => '_blank'));
|
|
//$url .= Display::url(get_lang('DeleteRecord'), api_get_self().'?action=delete_record&'.$record['recordID']);
|
|
if ($this->isTeacher()) {
|
|
$url .= Display::url(Display::return_icon('link.gif',get_lang('CopyToLinkTool')), api_get_self().'?action=copy_record_to_link_tool&id='.$meetingDB['id'].'&record_id='.$record['recordID']);
|
|
$url .= Display::url(Display::return_icon('agenda.png',get_lang('AddToCalendar')), api_get_self().'?action=add_to_calendar&id='.$meetingDB['id'].'&start='.api_strtotime($meetingDB['created_at']).'&url='.$item['url']);
|
|
$url .= Display::url(Display::return_icon('delete.png',get_lang('Delete')), api_get_self().'?action=delete_record&id='.$record['recordID']);
|
|
}
|
|
//$url .= api_get_self().'?action=publish&id='.$record['recordID'];
|
|
$count++;
|
|
$recordArray[] = $url;
|
|
}
|
|
}*/
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
$actionLinks = '';
|
|
if ($this->isTeacher()) {
|
|
if ($meetingDB['visibility'] == 0) {
|
|
$actionLinks .= Display::url(
|
|
Display::return_icon(
|
|
'invisible.png',
|
|
get_lang('MakeVisible'),
|
|
array(),
|
|
ICON_SIZE_MEDIUM
|
|
),
|
|
api_get_self().'?'.
|
|
api_get_cidreq().
|
|
'&action=publish&id='.$meetingDB['id']
|
|
);
|
|
} else {
|
|
$actionLinks .= Display::url(
|
|
Display::return_icon(
|
|
'visible.png',
|
|
get_lang('MakeInvisible'),
|
|
array(),
|
|
ICON_SIZE_MEDIUM
|
|
),
|
|
api_get_self().'?'.
|
|
api_get_cidreq().
|
|
'&action=unpublish&id='.$meetingDB['id']
|
|
);
|
|
}
|
|
}
|
|
$actionLinksArray[] = $actionLinks;
|
|
$item['action_links'] = implode('<br />', $actionLinksArray);
|
|
}
|
|
//var_dump($recordArray);
|
|
$item['show_links'] = implode('<br />', $recordArray);
|
|
$item['action_links'] = implode('<br />', $actionLinksArray);
|
|
}
|
|
|
|
$item['created_at'] = api_convert_and_format_date($meetingDB['created_at']);
|
|
//created_at
|
|
$meetingDB['created_at'] = $item['created_at']; //avoid overwrite in array_merge() below
|
|
|
|
$item['publish_url'] = api_get_self().'?'.api_get_cidreq().'&action=publish&id='.$meetingDB['id'];
|
|
$item['unpublish_url'] = api_get_self().'?'.api_get_cidreq().'&action=unpublish&id='.$meetingDB['id'];
|
|
|
|
if ($meetingDB['status'] == 1) {
|
|
$joinParams = array(
|
|
'meetingId' => $meetingDB['remote_id'], //-- REQUIRED - A unique id for the meeting
|
|
'username' => $this->user_complete_name, //-- REQUIRED - The name that will display for the user in the meeting
|
|
'password' => $pass, //-- REQUIRED - The attendee or moderator password, depending on what's passed here
|
|
'createTime' => '', //-- OPTIONAL - string. Leave blank ('') unless you set this correctly.
|
|
'userID' => '', // -- OPTIONAL - string
|
|
'webVoiceConf' => '' // -- OPTIONAL - string
|
|
);
|
|
$item['go_url'] = $this->protocol.$this->api->getJoinMeetingURL($joinParams);
|
|
}
|
|
$item = array_merge($item, $meetingDB, $meetingBBB);
|
|
$newMeetingList[] = $item;
|
|
}
|
|
return $newMeetingList;
|
|
}
|
|
|
|
/**
|
|
* Function disabled
|
|
*/
|
|
public function publishMeeting($id)
|
|
{
|
|
//return BigBlueButtonBN::setPublishRecordings($id, 'true', $this->url, $this->salt);
|
|
if (empty($id)) {
|
|
return false;
|
|
}
|
|
$id = intval($id);
|
|
Database::update($this->table, array('visibility' => 1), array('id = ? ' => $id));
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Function disabled
|
|
*/
|
|
public function unpublishMeeting($id)
|
|
{
|
|
//return BigBlueButtonBN::setPublishRecordings($id, 'false', $this->url, $this->salt);
|
|
if (empty($id)) {
|
|
return false;
|
|
}
|
|
$id = intval($id);
|
|
Database::update($this->table, array('visibility' => 0), array('id = ? ' => $id));
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Closes a meeting (usually when the user click on the close button from
|
|
* the conferences listing.
|
|
* @param string The internal ID of the meeting (id field for this meeting)
|
|
* @return void
|
|
* @assert (0) === false
|
|
*/
|
|
public function endMeeting($id)
|
|
{
|
|
if (empty($id)) {
|
|
return false;
|
|
}
|
|
$meetingData = Database::select('*', $this->table, array('where' => array('id = ?' => array($id))), 'first');
|
|
$pass = $this->getUserMeetingPassword();
|
|
|
|
$endParams = array(
|
|
'meetingId' => $meetingData['remote_id'], // REQUIRED - We have to know which meeting to end.
|
|
'password' => $pass, // REQUIRED - Must match moderator pass for meeting.
|
|
);
|
|
$this->api->endMeetingWithXmlResponseArray($endParams);
|
|
Database::update($this->table, array('status' => 0, 'closed_at' => api_get_utc_datetime()), array('id = ? ' => $id));
|
|
}
|
|
|
|
/**
|
|
* Gets the password for a specific meeting for the current user
|
|
* @return string A moderator password if user is teacher, or the course code otherwise
|
|
*/
|
|
public function getUserMeetingPassword()
|
|
{
|
|
if ($this->isTeacher()) {
|
|
return $this->getModMeetingPassword();
|
|
} else {
|
|
return api_get_course_id();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Generated a moderator password for the meeting
|
|
* @return string A password for the moderation of the videoconference
|
|
*/
|
|
public function getModMeetingPassword()
|
|
{
|
|
return api_get_course_id().'mod';
|
|
}
|
|
|
|
/**
|
|
* Get users online in the current course room
|
|
* @return int The number of users currently connected to the videoconference
|
|
* @assert () > -1
|
|
*/
|
|
public function getUsersOnlineInCurrentRoom()
|
|
{
|
|
$courseId = api_get_course_int_id();
|
|
$sessionId = api_get_session_id();
|
|
$meetingData = Database::select('*', $this->table, array('where' => array('c_id = ? AND session_id = ? AND status = 1 ' => array($courseId, $sessionId))), 'first');
|
|
if (empty($meetingData)) {
|
|
return 0;
|
|
}
|
|
$pass = $this->getModMeetingPassword();
|
|
$info = $this->getMeetingInfo(array('meetingId' => $meetingData['remote_id'], 'password' => $pass));
|
|
if ($info === false) {
|
|
//checking with the remote_id didn't work, so just in case and
|
|
// to provide backwards support, check with the id
|
|
$params = array(
|
|
'meetingId' => $meetingData['id'],
|
|
// -- REQUIRED - The unique id for the meeting
|
|
'password' => $pass
|
|
// -- REQUIRED - The moderator password for the meeting
|
|
);
|
|
$info = $this->getMeetingInfo($params);
|
|
}
|
|
|
|
if (!empty($info) && isset($info['participantCount'])) {
|
|
return $info['participantCount'];
|
|
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Deletes a previous recording of a meeting
|
|
* @param int integral ID of the recording
|
|
* @return array ?
|
|
* @assert () === false
|
|
* @todo Also delete links and agenda items created from this recording
|
|
*/
|
|
public function deleteRecord($id)
|
|
{
|
|
if (empty($id) or $id != intval($id)) {
|
|
return false;
|
|
}
|
|
$meetingData = Database::select('*', $this->table, array('where' => array('id = ?' => array($id))), 'first');
|
|
$recordingParams = array(
|
|
/*
|
|
* NOTE: Set the recordId below to a valid id after you have
|
|
* created a recorded meeting, and received a real recordID
|
|
* back from your BBB server using the
|
|
* getRecordingsWithXmlResponseArray method.
|
|
*/
|
|
|
|
// REQUIRED - We have to know which recording:
|
|
'recordId' => $meetingData['remote_id'],
|
|
);
|
|
return $this->api->deleteRecordingsWithXmlResponseArray($recordingParams);
|
|
}
|
|
|
|
/**
|
|
* Creates a link in the links tool from the given videoconference recording
|
|
* @param int ID of the item in the plugin_bbb_meeting table
|
|
* @param string Hash identifying the recording, as provided by the API
|
|
* @return mixed ID of the newly created link, or false on error
|
|
* @assert (null, null) === false
|
|
* @assert (1, null) === false
|
|
* @assert (null, 'abcdefabcdefabcdefabcdef') === false
|
|
*/
|
|
public function copyRecordToLinkTool($id)
|
|
{
|
|
if (empty($id)) {
|
|
return false;
|
|
}
|
|
require_once api_get_path(LIBRARY_PATH).'link.lib.php';
|
|
//$records = BigBlueButtonBN::getRecordingsUrl($id);
|
|
$meetingData = Database::select('*', $this->table, array('where' => array('id = ?' => array($id))), 'first');
|
|
|
|
$records = $this->api->getRecordingsWithXmlResponseArray(array('meetingId' => $meetingData['remote_id']));
|
|
|
|
if (!empty($records)) {
|
|
$count = 1;
|
|
if (isset($records['message']) && !empty($records['message'])) {
|
|
if ($records['messageKey'] == 'noRecordings') {
|
|
$recordArray[] = get_lang('NoRecording');
|
|
} else {
|
|
//$recordArray[] = $records['message'];
|
|
}
|
|
return false;
|
|
} else {
|
|
$record = $records[0];
|
|
if (is_array($record) && isset($record['recordId'])) {
|
|
$url = $record['playbackFormatUrl'];
|
|
$link = new Link();
|
|
$params['url'] = $url;
|
|
$params['title'] = $meetingData['meeting_name'];
|
|
$id = $link->save($params);
|
|
return $id;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Checks if the videoconference server is running.
|
|
* Function currently disabled (always returns 1)
|
|
* @return bool True if server is running, false otherwise
|
|
* @assert () === false
|
|
*/
|
|
public function isServerRunning()
|
|
{
|
|
return true;
|
|
//return BigBlueButtonBN::isServerRunning($this->protocol.$this->url);
|
|
}
|
|
|
|
/**
|
|
* Get active session in the all platform
|
|
*/
|
|
public function getActiveSessionsCount()
|
|
{
|
|
$meetingList = Database::select(
|
|
'count(id) as count',
|
|
$this->table,
|
|
array('where' => array('status = ?' => array(1))),
|
|
'first'
|
|
);
|
|
|
|
return $meetingList['count'];
|
|
}
|
|
|
|
/**
|
|
* @param string $url
|
|
*/
|
|
public function redirectToBBB($url)
|
|
{
|
|
if (file_exists(__DIR__ . '/../config.vm.php')) {
|
|
// Using VM
|
|
echo Display::url(get_lang('ClickToContinue'), $url);
|
|
exit;
|
|
} else {
|
|
// Classic
|
|
header("Location: $url");
|
|
exit;
|
|
}
|
|
|
|
// js
|
|
/*echo '<script>';
|
|
echo 'window.location = "'.$url.'"';
|
|
echo '</script>';
|
|
exit;*/
|
|
}
|
|
}
|
|
|