Chamilo is a learning management system focused on ease of use and accessibility
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.
 
 
 
 
 
 
chamilo-lms/plugin/bbb/lib/bbb.lib.php

1940 lines
62 KiB

<?php
/* For licensing terms, see /license.txt */
/**
* Class bbb
* This script initiates a video conference session, calling the BigBlueButton
* API BigBlueButton-Chamilo connector class
*/
class bbb
{
public $url;
public $salt;
public $api;
public $userCompleteName = '';
public $protocol = 'http://';
public $debug = false;
public $logoutUrl = '';
public $pluginEnabled = false;
public $enableGlobalConference = false;
public $enableGlobalConferencePerUser = false;
public $isGlobalConference = false;
public $groupSupport = false;
public $userSupport = false;
public $accessUrl = 1;
public $userId = 0;
public $plugin;
private $courseCode;
private $courseId;
private $sessionId;
private $groupId;
private $maxUsersLimit;
/**
* 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
* @param bool $isGlobalConference
* @param int $isGlobalPerUser
*/
public function __construct(
$host = '',
$salt = '',
$isGlobalConference = false,
$isGlobalPerUser = 0
) {
$this->courseCode = api_get_course_id();
$this->courseId = api_get_course_int_id();
$this->sessionId = api_get_session_id();
$this->groupId = api_get_group_id();
// Initialize video server settings from global settings
$this->plugin = BBBPlugin::create();
$bbbPluginEnabled = $this->plugin->get('tool_enable');
$bbb_host = !empty($host) ? $host : $this->plugin->get('host');
$bbb_salt = !empty($salt) ? $salt : $this->plugin->get('salt');
$this->table = Database::get_main_table('plugin_bbb_meeting');
$this->enableGlobalConference = $this->plugin->get('enable_global_conference') === 'true';
$this->isGlobalConference = (bool) $isGlobalConference;
$columns = Database::listTableColumns($this->table);
$this->groupSupport = isset($columns['group_id']) ? true : false;
$this->userSupport = isset($columns['user_id']) ? true : false;
$this->accessUrl = api_get_current_access_url_id();
$this->enableGlobalConferencePerUser = false;
if ($this->userSupport && !empty($isGlobalPerUser)) {
$this->enableGlobalConferencePerUser = $this->plugin->get('enable_global_conference_per_user') === 'true';
$this->userId = $isGlobalPerUser;
}
if ($this->groupSupport) {
// Plugin check
$this->groupSupport = $this->plugin->get('enable_conference_in_course_groups') === 'true' ? true : false;
if ($this->groupSupport) {
// Platform check
$bbbSetting = api_get_setting('bbb_enable_conference_in_course_groups');
$bbbSetting = isset($bbbSetting['bbb']) ? $bbbSetting['bbb'] === 'true' : false;
if ($bbbSetting) {
// Course check
$courseInfo = api_get_course_info();
if ($courseInfo) {
$this->groupSupport = api_get_course_plugin_setting(
'bbb',
'bbb_enable_conference_in_groups',
$courseInfo
) === '1';
}
}
}
}
$this->maxUsersLimit = $this->plugin->get('max_users_limit');
if ($bbbPluginEnabled === 'true') {
$userInfo = api_get_user_info();
if (empty($userInfo) && !empty($isGlobalPerUser)) {
// If we are following a link to a global "per user" conference
// then generate a random guest name to join the conference
// because there is no part of the process where we give a name
$this->userCompleteName = 'Guest'.rand(1000, 9999);
} else {
$this->userCompleteName = $userInfo['complete_name'];
}
$this->salt = $bbb_salt;
if (!empty($bbb_host)) {
if (substr($bbb_host, -1, 1) !== '/') {
$bbb_host .= '/';
}
if (!preg_match('#/bigbluebutton/$#', $bbb_host)) {
$bbb_host .= 'bigbluebutton/';
}
}
$info = parse_url($bbb_host);
$this->url = $bbb_host;
if (isset($info['scheme'])) {
$this->protocol = $info['scheme'].'://';
$this->url = str_replace($this->protocol, '', $this->url);
$urlWithProtocol = $bbb_host;
} else {
// We assume it's an http, if user wants to use https, the host *must* include the protocol.
$urlWithProtocol = 'http://'.$bbb_host;
}
// Setting BBB api
define('CONFIG_SECURITY_SALT', $this->salt);
define('CONFIG_SERVER_URL_WITH_PROTOCOL', $urlWithProtocol);
define('CONFIG_SERVER_BASE_URL', $this->url);
$this->api = new BigBlueButtonBN();
$this->pluginEnabled = true;
$this->logoutUrl = $this->getListingUrl();
}
}
/**
* @param int $courseId Optional. Course ID.
* @param int $sessionId Optional. Session ID.
* @param int $groupId Optional. Group ID.
*
* @return string
*/
public function getListingUrl($courseId = 0, $sessionId = 0, $groupId = 0)
{
return api_get_path(WEB_PLUGIN_PATH).'bbb/listing.php?'
.$this->getUrlParams($courseId, $sessionId, $groupId);
}
/**
* @param int $courseId Optional. Course ID.
* @param int $sessionId Optional. Session ID.
* @param int $groupId Optional. Group ID.
*
* @return string
*/
public function getUrlParams($courseId = 0, $sessionId = 0, $groupId = 0)
{
if (empty($this->courseCode) && !$courseId) {
if ($this->isGlobalConferencePerUserEnabled()) {
return 'global=1&user_id='.$this->userId;
}
if ($this->isGlobalConference()) {
return 'global=1';
}
return '';
}
return http_build_query(
[
'cidReq' => $courseId ? api_get_course_entity($courseId)->getCode() : $this->courseCode,
'id_session' => $sessionId ?: $this->sessionId,
'gidReq' => $groupId ?: $this->groupId,
]
);
}
/**
* @return bool
*/
public function isGlobalConferencePerUserEnabled()
{
return $this->enableGlobalConferencePerUser;
}
/**
* @return bool
*/
public function isGlobalConference()
{
if ($this->isGlobalConferenceEnabled() === false) {
return false;
}
return (bool) $this->isGlobalConference;
}
/**
* @return bool
*/
public function isGlobalConferenceEnabled()
{
return $this->enableGlobalConference;
}
/**
* @param array $userInfo
*
* @return bool
*/
public static function showGlobalConferenceLink($userInfo)
{
if (empty($userInfo)) {
return false;
}
$setting = api_get_plugin_setting('bbb', 'enable_global_conference');
$settingLink = api_get_plugin_setting('bbb', 'enable_global_conference_link');
if ($setting === 'true' && $settingLink === 'true') {
//$content = Display::url(get_lang('LaunchVideoConferenceRoom'), $url);
$allowedRoles = api_get_plugin_setting(
'bbb',
'global_conference_allow_roles'
);
if (api_is_platform_admin()) {
$userInfo['status'] = PLATFORM_ADMIN;
}
$showGlobalLink = true;
if (!empty($allowedRoles)) {
if (!in_array($userInfo['status'], $allowedRoles)) {
$showGlobalLink = false;
}
}
return $showGlobalLink;
}
}
/**
* Gets the global limit of users in a video-conference room.
* This value can be overridden by course-specific values
* @return int Maximum number of users set globally
*/
public function getMaxUsersLimit()
{
$limit = $this->maxUsersLimit;
if ($limit <= 0) {
$limit = 0;
}
$courseLimit = 0;
$sessionLimit = 0;
// Check the extra fields for this course and session
// Session limit takes priority over course limit
// Course limit takes priority over global limit
if (!empty($this->courseId)) {
$extraField = new ExtraField('course');
$fieldId = $extraField->get_all(
array('variable = ?' => 'plugin_bbb_course_users_limit')
);
$extraValue = new ExtraFieldValue('course');
$value = $extraValue->get_values_by_handler_and_field_id($this->courseId, $fieldId[0]['id']);
if (!empty($value['value'])) {
$courseLimit = (int) $value['value'];
}
}
if (!empty($this->sessionId)) {
$extraField = new ExtraField('session');
$fieldId = $extraField->get_all(
array('variable = ?' => 'plugin_bbb_session_users_limit')
);
$extraValue = new ExtraFieldValue('session');
$value = $extraValue->get_values_by_handler_and_field_id($this->sessionId, $fieldId[0]['id']);
if (!empty($value['value'])) {
$sessionLimit = (int) $value['value'];
}
}
if (!empty($sessionLimit)) {
return $sessionLimit;
} elseif (!empty($courseLimit)) {
return $courseLimit;
}
return (int) $limit;
}
/**
* Sets the global limit of users in a video-conference room.
*
* @param int Maximum number of users (globally)
*/
public function setMaxUsersLimit($max)
{
if ($max < 0) {
$max = 0;
}
$this->maxUsersLimit = (int) $max;
}
/**
* 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
*
* @return mixed
*/
public function createMeeting($params)
{
$params['c_id'] = api_get_course_int_id();
$params['session_id'] = api_get_session_id();
if ($this->hasGroupSupport()) {
$params['group_id'] = api_get_group_id();
}
if ($this->isGlobalConferencePerUserEnabled() && !empty($this->userId)) {
$params['user_id'] = (int) $this->userId;
}
$params['attendee_pw'] = isset($params['attendee_pw']) ? $params['attendee_pw'] : $this->getUserMeetingPassword();
$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_plugin_setting('bbb', 'big_blue_button_record_and_store') == 1 ? true : false;
$max = api_get_course_plugin_setting('bbb', 'big_blue_button_max_students_allowed');
$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);
// Each simultaneous conference room needs to have a different
// voice_bridge composed of a 5 digits number, so generating a random one
$params['voice_bridge'] = rand(10000, 99999);
$params['created_at'] = api_get_utc_datetime();
$params['access_url'] = $this->accessUrl;
// Check interface feature is installed
$interfaceFeature = $this->plugin->get('interface');
if ($interfaceFeature === false) {
if (isset($params['interface'])) {
unset($params['interface']);
}
}
$id = Database::insert($this->table, $params);
if ($id) {
Event::addEvent(
'bbb_create_meeting',
'meeting_id',
(int) $id,
null,
api_get_user_id(),
api_get_course_int_id(),
api_get_session_id()
);
$meetingName = isset($params['meeting_name']) ? $params['meeting_name'] : $this->getCurrentVideoConferenceName(
);
$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' => $params['voice_bridge'], // PIN to join voice. Required.
'webVoice' => '', // Alphanumeric to join voice. Optional.
'logoutUrl' => $this->logoutUrl.'&action=logout&remote_id='.$params['remote_id'],
'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.
);
$status = false;
$meeting = null;
while ($status === false) {
$result = $this->api->createMeetingWithXmlResponseArray($bbbParams);
if (isset($result) && strval($result['returncode']) == 'SUCCESS') {
if ($this->plugin->get('allow_regenerate_recording') === 'true') {
$internalId = Database::escape_string($result['internalMeetingID']);
$sql = "UPDATE $this->table SET internal_meeting_id = '".$internalId."'
WHERE id = $id";
Database::query($sql);
}
$meeting = $this->joinMeeting($meetingName, true);
return $meeting;
}
}
}
return false;
}
/**
* @return bool
*/
public function hasGroupSupport()
{
return $this->groupSupport;
}
/**
* Gets the password for a specific meeting for the current user
*
* @param string $courseCode
*
* @return string A moderator password if user is teacher, or the course code otherwise
*
*/
public function getUserMeetingPassword($courseCode = null)
{
if ($this->isGlobalConferencePerUserEnabled()) {
return 'url_'.$this->userId.'_'.api_get_current_access_url_id();
}
if ($this->isGlobalConference()) {
return 'url_'.api_get_current_access_url_id();
}
$courseCode = empty($courseCode) ? api_get_course_id() : $courseCode;
return $courseCode;
}
/**
* Generated a moderator password for the meeting.
*
* @param string $courseCode
*
* @return string A password for the moderation of the videoconference
*/
public function getModMeetingPassword($courseCode = null)
{
if ($this->isGlobalConferencePerUserEnabled()) {
return 'url_'.$this->userId.'_'.api_get_current_access_url_id().'_mod';
}
if ($this->isGlobalConference()) {
return 'url_'.api_get_current_access_url_id().'_mod';
}
$courseCode = empty($courseCode) ? api_get_course_id() : $courseCode;
return $courseCode.'mod';
}
/**
* @return string
*/
public function getCurrentVideoConferenceName()
{
if ($this->isGlobalConferencePerUserEnabled()) {
return 'url_'.$this->userId.'_'.api_get_current_access_url_id();
}
if ($this->isGlobalConference()) {
return 'url_'.api_get_current_access_url_id();
}
if ($this->hasGroupSupport()) {
return api_get_course_id().'-'.api_get_session_id().'-'.api_get_group_id();
}
return api_get_course_id().'-'.api_get_session_id();
}
/**
* 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)
{
if ($this->debug) {
error_log("joinMeeting: $meetingName");
}
if (empty($meetingName)) {
return false;
}
$manager = $this->isConferenceManager();
if ($manager) {
$pass = $this->getModMeetingPassword();
} else {
$pass = $this->getUserMeetingPassword();
}
$meetingData = Database::select(
'*',
$this->table,
array(
'where' => array(
'meeting_name = ? AND status = 1 AND access_url = ?' => array(
$meetingName,
$this->accessUrl,
),
),
),
'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
);
$meetingInfoExists = false;
$meetingIsRunningInfo = $this->getMeetingInfo($params);
if ($this->debug) {
error_log('Searching meeting with params:');
error_log(print_r($params, 1));
error_log('Result:');
error_log(print_r($meetingIsRunningInfo, 1));
}
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('Searching meetingId with params:');
error_log(print_r($params, 1));
error_log('Result:');
error_log(print_r($meetingIsRunningInfo, 1));
}
}
if (strval($meetingIsRunningInfo['returncode']) === 'SUCCESS' &&
isset($meetingIsRunningInfo['meetingName']) &&
!empty($meetingIsRunningInfo['meetingName'])
) {
$meetingInfoExists = true;
}
if ($this->debug) {
error_log(
"meeting is running: ".intval($meetingInfoExists)
);
}
$url = false;
if ($meetingInfoExists) {
$joinParams = [
'meetingId' => $meetingData['remote_id'],
// -- REQUIRED - A unique id for the meeting
'username' => $this->userCompleteName,
//-- 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
'interface' => $this->checkInterface($meetingData),
];
$url = $this->api->getJoinMeetingURL($joinParams);
$url = $this->protocol.$url;
}
if ($this->debug) {
error_log("return url :".$url);
}
return $url;
}
/**
* 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 isConferenceManager()
{
if (api_is_coach() || api_is_platform_admin()) {
return true;
}
if ($this->isGlobalConferencePerUserEnabled()) {
$currentUserId = api_get_user_id();
if ($this->userId === $currentUserId) {
return true;
} else {
return false;
}
}
$courseInfo = api_get_course_info();
if (!empty($courseInfo)) {
return api_is_course_admin();
}
return false;
}
/**
* 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.");
}
}
return $result;
} catch (Exception $e) {
if ($this->debug) {
error_log('Caught exception: ', $e->getMessage(), "\n");
}
}
return false;
}
/**
* @param $meetingInfo
*
* @return int
*/
public function checkInterface($meetingInfo)
{
$interface = BBBPlugin::LAUNCH_TYPE_DEFAULT;
$type = $this->plugin->get('launch_type');
switch ($type) {
case BBBPlugin::LAUNCH_TYPE_DEFAULT:
$interface = $this->plugin->get('interface');
break;
case BBBPlugin::LAUNCH_TYPE_SET_BY_TEACHER:
if (isset($meetingInfo['interface'])) {
$interface = $meetingInfo['interface'];
}
break;
case BBBPlugin::LAUNCH_TYPE_SET_BY_STUDENT:
if (isset($meetingInfo['id'])) {
$roomInfo = $this->getMeetingParticipantInfo($meetingInfo['id'], api_get_user_id());
if (!empty($roomInfo) && isset($roomInfo['interface'])) {
$interface = $roomInfo['interface'];
} else {
if (isset($_REQUEST['interface'])) {
$interface = isset($_REQUEST['interface']) ? (int) $_REQUEST['interface'] : 0;
}
}
}
break;
}
return $interface;
}
/**
* @param int $meetingId
* @param int $userId
*
* @return array
*/
public function getMeetingParticipantInfo($meetingId, $userId)
{
$meetingData = Database::select(
'*',
'plugin_bbb_room',
array('where' => array('meeting_id = ? AND participant_id = ?' => [$meetingId, $userId])),
'first'
);
if ($meetingData) {
return $meetingData;
}
return [];
}
/**
* Save a participant in a meeting room
*
* @param int $meetingId
* @param int $participantId
* @param int $interface
*
* @return false|int The last inserted ID. Otherwise return false
*/
public function saveParticipant($meetingId, $participantId, $interface = 0)
{
$params = [
'meeting_id' => $meetingId,
'participant_id' => $participantId,
'in_at' => api_get_utc_datetime(),
'out_at' => api_get_utc_datetime(),
];
if ($this->plugin->get('interface') !== false) {
$params['interface'] = $interface;
}
return Database::insert(
'plugin_bbb_room',
$params
);
}
/**
* 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)
{
$meetingData = $this->getMeetingByName($meetingName);
return !empty($meetingData);
}
/**
* @param string $meetingName
*
* @return array
*/
public function getMeetingByName($meetingName)
{
if (empty($meetingName)) {
return [];
}
$courseId = api_get_course_int_id();
$sessionId = api_get_session_id();
$conditions = array(
'where' => array(
'c_id = ? AND session_id = ? AND meeting_name = ? AND status = 1 AND access_url = ?' =>
array($courseId, $sessionId, $meetingName, $this->accessUrl),
),
);
if ($this->hasGroupSupport()) {
$groupId = api_get_group_id();
$conditions = array(
'where' => array(
'c_id = ? AND session_id = ? AND meeting_name = ? AND group_id = ? AND status = 1 AND access_url = ?' =>
array(
$courseId,
$sessionId,
$meetingName,
$groupId,
$this->accessUrl,
),
),
);
}
$meetingData = Database::select(
'*',
$this->table,
$conditions,
'first'
);
if ($this->debug) {
error_log('meeting_exists '.print_r($meetingData, 1));
}
return $meetingData;
}
/**
* Gets a list from the database of all meetings attached to a course with the given status
* @param int $courseId
* @param int $sessionId
* @param int $status 0 for closed meetings, 1 for open meetings
*
* @return array
*/
public function getAllMeetingsInCourse($courseId, $sessionId, $status)
{
$conditions = array(
'where' => array(
'status = ? AND c_id = ? AND session_id = ? ' => array(
$status,
$courseId,
$sessionId,
),
),
);
return Database::select(
'*',
$this->table,
$conditions
);
}
/**
* Gets all the course meetings saved in the plugin_bbb_meeting table and
* generate actionable links (join/close/delete/etc)
*
* @param int $courseId
* @param int $sessionId
* @param int $groupId
* @param bool $isAdminReport Optional. Set to true then the report is for admins
* @param array $dateRange Optional
*
* @return array Array of current open meeting rooms
* @throws Exception
*/
public function getMeetings(
$courseId = 0,
$sessionId = 0,
$groupId = 0,
$isAdminReport = false,
$dateRange = []
) {
$em = Database::getManager();
$manager = $this->isConferenceManager();
$conditions = [];
if ($courseId || $sessionId || $groupId) {
$conditions = array(
'where' => array(
'c_id = ? AND session_id = ? ' => array($courseId, $sessionId),
),
);
if ($this->hasGroupSupport()) {
$conditions = array(
'where' => array(
'c_id = ? AND session_id = ? AND group_id = ? ' => array(
$courseId,
$sessionId,
$groupId,
),
),
);
}
if ($this->isGlobalConferencePerUserEnabled()) {
$conditions = array(
'where' => array(
'c_id = ? AND session_id = ? AND user_id = ?' => array(
$courseId,
$sessionId,
$this->userId,
),
),
);
}
}
if (!empty($dateRange)) {
$dateStart = date_create($dateRange['search_meeting_start']);
$dateStart = date_format($dateStart, 'Y-m-d H:i:s');
$dateEnd = date_create($dateRange['search_meeting_end']);
$dateEnd = $dateEnd->add(new DateInterval('P1D'));
$dateEnd = date_format($dateEnd, 'Y-m-d H:i:s');
$conditions = array(
'where' => array(
'created_at BETWEEN ? AND ? ' => array($dateStart, $dateEnd),
),
);
}
$conditions['order'] = 'created_at ASC';
$meetingList = Database::select(
'*',
$this->table,
$conditions
);
$isGlobal = $this->isGlobalConference();
$newMeetingList = array();
foreach ($meetingList as $meetingDB) {
$item = array();
$courseId = $meetingDB['c_id'];
$courseInfo = api_get_course_info_by_id($courseId);
$courseCode = '';
if (!empty($courseInfo)) {
$courseCode = $courseInfo['code'];
}
if ($manager) {
$pass = $meetingDB['moderator_pw'];
} else {
$pass = $meetingDB['attendee_pw'];
}
$meetingBBB = $this->getMeetingInfo(
[
'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 && $this->isConferenceManager() === false) {
continue;
}
$meetingBBB['end_url'] = $this->endUrl($meetingDB);
if (isset($meetingBBB['returncode']) && (string) $meetingBBB['returncode'] === 'FAILED') {
if ($meetingDB['status'] == 1 && $this->isConferenceManager()) {
$this->endMeeting($meetingDB['id'], $courseCode);
}
} else {
$meetingBBB['add_to_calendar_url'] = $this->addToCalendarUrl($meetingDB);
}
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;
}
$record = [];
$recordingParams = ['meetingId' => $mId];
$records = $this->api->getRecordingsWithXmlResponseArray($recordingParams);
if (!empty($records)) {
if (!isset($records['messageKey']) || $records['messageKey'] !== 'noRecordings') {
$record = end($records);
if (!is_array($record) || !isset($record['recordId'])) {
continue;
}
if (!empty($record['playbackFormatUrl'])) {
$this->updateMeetingVideoUrl($meetingDB['id'], $record['playbackFormatUrl']);
}
}
}
if (isset($record['playbackFormatUrl']) && !empty($record['playbackFormatUrl'])) {
$recordLink = Display::url(
$this->plugin->get_lang('ViewRecord'),
$record['playbackFormatUrl'],
['target' => '_blank', 'class' => 'btn btn-default']
);
} else {
$recordLink = $this->plugin->get_lang('NoRecording');
}
if ($isAdminReport) {
$this->forceCIdReq(
$courseInfo['code'],
$meetingDB['session_id'],
$meetingDB['group_id']
);
}
$actionLinks = $this->getActionLinks(
$meetingDB,
$record,
$isGlobal,
$isAdminReport
);
$item['show_links'] = $recordLink;
} else {
$actionLinks = $this->getActionLinks(
$meetingDB,
[],
$isGlobal,
$isAdminReport
);
$item['show_links'] = $this->plugin->get_lang('NoRecording');
}
$item['action_links'] = implode(PHP_EOL, $actionLinks);
$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'] = $this->publishUrl($meetingDB);
$item['unpublish_url'] = $this->unPublishUrl($meetingBBB);
if ($meetingDB['status'] == 1) {
$joinParams = [
'meetingId' => $meetingDB['remote_id'],
//-- REQUIRED - A unique id for the meeting
'username' => $this->userCompleteName,
//-- 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
'interface' => $this->checkInterface($meetingDB),
];
$item['go_url'] = $this->protocol.$this->api->getJoinMeetingURL($joinParams);
}
$item = array_merge($item, $meetingDB, $meetingBBB);
$item['course'] = $em->find('ChamiloCoreBundle:Course', $item['c_id']);
$item['session'] = $em->find('ChamiloCoreBundle:Session', $item['session_id']);
$newMeetingList[] = $item;
}
return $newMeetingList;
}
/**
* @param array $meeting
*
* @return string
*/
public function endUrl($meeting)
{
if (!isset($meeting['id'])) {
return '';
}
return api_get_path(WEB_PLUGIN_PATH).'bbb/listing.php?'.$this->getUrlParams().'&action=end&id='.$meeting['id'];
}
/**
* 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)
* @param string $courseCode
*
* @return void
* @assert (0) === false
*/
public function endMeeting($id, $courseCode = null)
{
if (empty($id)) {
return false;
}
$meetingData = Database::select(
'*',
$this->table,
array('where' => array('id = ?' => array($id))),
'first'
);
$manager = $this->isConferenceManager();
if ($manager) {
$pass = $meetingData['moderator_pw'];
} else {
$pass = $meetingData['attendee_pw'];
}
Event::addEvent(
'bbb_end_meeting',
'meeting_id',
(int) $id,
null,
api_get_user_id(),
api_get_course_int_id(),
api_get_session_id()
);
$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)
);
}
/**
* @param array $meeting
* @param array $record
*
* @return string
*/
public function addToCalendarUrl($meeting, $record = [])
{
$url = isset($record['playbackFormatUrl']) ? $record['playbackFormatUrl'] : '';
return api_get_path(WEB_PLUGIN_PATH).'bbb/listing.php?'.$this->getUrlParams(
).'&action=add_to_calendar&id='.$meeting['id'].'&start='.api_strtotime($meeting['created_at']).'&url='.$url;
}
/**
* @param int $meetingId
* @param string $videoUrl
*
* @return bool|int
*/
public function updateMeetingVideoUrl($meetingId, $videoUrl)
{
return Database::update(
'plugin_bbb_meeting',
['video_url' => $videoUrl],
['id = ?' => intval($meetingId)]
);
}
/**
* Force the course, session and/or group IDs
*
* @param string $courseCode
* @param int $sessionId
* @param int $groupId
*/
public function forceCIdReq($courseCode, $sessionId = 0, $groupId = 0)
{
$this->courseCode = $courseCode;
$this->sessionId = (int) $sessionId;
$this->groupId = (int) $groupId;
}
/**
* @param array $meetingInfo
* @param array $recordInfo
* @param bool $isGlobal
* @param bool $isAdminReport
*
* @return array
*/
private function getActionLinks(
$meetingInfo,
$recordInfo,
$isGlobal = false,
$isAdminReport = false
) {
$isVisible = $meetingInfo['visibility'] != 0;
$linkVisibility = $isVisible
? Display::url(
Display::return_icon('visible.png', get_lang('MakeInvisible')),
$this->unPublishUrl($meetingInfo)
)
: Display::url(
Display::return_icon('invisible.png', get_lang('MakeVisible')),
$this->publishUrl($meetingInfo)
);
$links = [];
if ($this->plugin->get('allow_regenerate_recording') === 'true' && $meetingInfo['record'] == 1) {
if (!empty($recordInfo)) {
$links[] = Display::url(
Display::return_icon('reload.png', get_lang('RegenerateRecord')),
$this->regenerateRecordUrl($meetingInfo, $recordInfo)
);
} else {
$links[] = Display::url(
Display::return_icon('reload.png', get_lang('RegenerateRecord')),
$this->regenerateRecordUrlFromMeeting($meetingInfo)
);
}
}
if (empty($recordInfo)) {
if (!$isAdminReport) {
if ($meetingInfo['status'] == 0) {
$links[] = Display::url(
Display::return_icon('delete.png', get_lang('Delete')),
$this->deleteRecordUrl($meetingInfo)
);
$links[] = $linkVisibility;
}
return $links;
} else {
$links[] = Display::url(
Display::return_icon('course_home.png', get_lang('GoToCourse')),
$this->getListingUrl($meetingInfo['c_id'], $meetingInfo['session_id'], $meetingInfo['group_id'])
);
return $links;
}
}
if (!$isGlobal) {
$links[] = Display::url(
Display::return_icon('link.gif', get_lang('UrlMeetingToShare')),
$this->copyToRecordToLinkTool($meetingInfo)
);
$links[] = Display::url(
Display::return_icon('agenda.png', get_lang('AddToCalendar')),
$this->addToCalendarUrl($meetingInfo, $recordInfo)
);
}
$hide = $this->plugin->get('disable_download_conference_link') === 'true' ? true : false;
if ($hide == false) {
if ($meetingInfo['has_video_m4v']) {
$links[] = Display::url(
Display::return_icon('save.png', get_lang('DownloadFile')),
$recordInfo['playbackFormatUrl'].'/capture.m4v',
['target' => '_blank']
);
} else {
$links[] = Display::url(
Display::return_icon('save.png', get_lang('DownloadFile')),
'#',
[
'id' => "btn-check-meeting-video-{$meetingInfo['id']}",
'class' => 'check-meeting-video',
'data-id' => $meetingInfo['id'],
]
);
}
}
if (!$isAdminReport) {
$links[] = Display::url(
Display::return_icon('delete.png', get_lang('Delete')),
$this->deleteRecordUrl($meetingInfo)
);
$links[] = $linkVisibility;
} else {
$links[] = Display::url(
Display::return_icon('course_home.png', get_lang('GoToCourse')),
$this->getListingUrl($meetingInfo['c_id'], $meetingInfo['session_id'], $meetingInfo['group_id'])
);
}
return $links;
}
/**
* @param array $meeting
*
* @return string
*/
public function unPublishUrl($meeting)
{
if (!isset($meeting['id'])) {
return null;
}
return api_get_path(WEB_PLUGIN_PATH).'bbb/listing.php?'.$this->getUrlParams(
).'&action=unpublish&id='.$meeting['id'];
}
/**
* @param array $meeting
*
* @return string
*/
public function publishUrl($meeting)
{
if (!isset($meeting['id'])) {
return '';
}
return api_get_path(WEB_PLUGIN_PATH).'bbb/listing.php?'.$this->getUrlParams(
).'&action=publish&id='.$meeting['id'];
}
/**
* @param array $meeting
* @param array $recordInfo
*
* @return string
*/
public function regenerateRecordUrl($meeting, $recordInfo)
{
if ($this->plugin->get('allow_regenerate_recording') !== 'true') {
return '';
}
if (!isset($meeting['id'])) {
return '';
}
if (empty($recordInfo) || (!empty($recordInfo['recordId']) && !isset($recordInfo['recordId']))) {
return '';
}
return api_get_path(WEB_PLUGIN_PATH).'bbb/listing.php?'.$this->getUrlParams().
'&action=regenerate_record&id='.$meeting['id'].'&record_id='.$recordInfo['recordId'];
}
/**
* @param array $meeting
*
* @return string
*/
public function regenerateRecordUrlFromMeeting($meeting)
{
if ($this->plugin->get('allow_regenerate_recording') !== 'true') {
return '';
}
if (!isset($meeting['id'])) {
return '';
}
return api_get_path(WEB_PLUGIN_PATH).'bbb/listing.php?'.$this->getUrlParams().
'&action=regenerate_record&id='.$meeting['id'];
}
/**
* @param array $meeting
*
* @return string
*/
public function deleteRecordUrl($meeting)
{
if (!isset($meeting['id'])) {
return '';
}
return api_get_path(WEB_PLUGIN_PATH).'bbb/listing.php?'.$this->getUrlParams(
).'&action=delete_record&id='.$meeting['id'];
}
/**
* @param array $meeting
*
* @return string
*/
public function copyToRecordToLinkTool($meeting)
{
if (!isset($meeting['id'])) {
return '';
}
return api_get_path(WEB_PLUGIN_PATH).
'bbb/listing.php?'.$this->getUrlParams().'&action=copy_record_to_link_tool&id='.$meeting['id'];
}
/**
* 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;
}
/**
* 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();
$conditions = array(
'where' => array(
'c_id = ? AND session_id = ? AND status = 1 AND access_url = ?' => array(
$courseId,
$sessionId,
$this->accessUrl,
),
),
);
if ($this->hasGroupSupport()) {
$groupId = api_get_group_id();
$conditions = array(
'where' => array(
'c_id = ? AND session_id = ? AND group_id = ? AND status = 1 AND access_url = ?' => array(
$courseId,
$sessionId,
$groupId,
$this->accessUrl,
),
),
);
}
if ($this->isGlobalConferencePerUserEnabled()) {
$conditions = array(
'where' => array(
'user_id = ? AND status = 1 AND access_url = ?' => array(
$this->userId,
$this->accessUrl,
),
),
);
}
$meetingData = Database::select(
'*',
$this->table,
$conditions,
'first'
);
if (empty($meetingData)) {
return 0;
}
$pass = $meetingData['moderator_pw'];
$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;
}
/**
* @param int $id
* @param string $recordId
*
* @return bool
*/
public function regenerateRecording($id, $recordId = '')
{
if ($this->plugin->get('allow_regenerate_recording') !== 'true') {
return false;
}
if (empty($id)) {
return false;
}
$meetingData = Database::select(
'*',
$this->table,
array('where' => array('id = ?' => array($id))),
'first'
);
Event::addEvent(
'bbb_regenerate_record',
'record_id',
(int) $recordId,
null,
api_get_user_id(),
api_get_course_int_id(),
api_get_session_id()
);
// Check if there are recordings for this meeting
$recordings = $this->api->getRecordings(['meetingId' => $meetingData['remote_id']]);
if (!empty($recordings) && isset($recordings['messageKey']) && $recordings['messageKey'] === 'noRecordings') {
// Regenerate the meeting id
if (!empty($meetingData['internal_meeting_id'])) {
return $this->api->generateRecording(['recordId' => $meetingData['internal_meeting_id']]);
}
/*$pass = $this->getModMeetingPassword();
$info = $this->getMeetingInfo(['meetingId' => $meetingData['remote_id'], 'password' => $pass]);
if (!empty($info) && isset($info['internalMeetingID'])) {
return $this->api->generateRecording(['recordId' => $meetingData['internal_meeting_id']]);
}*/
return false;
} else {
if (!empty($recordings['records'])) {
$recordExists = false;
foreach ($recordings['records'] as $record) {
if ($recordId == $record['recordId']) {
$recordExists = true;
break;
}
}
if ($recordExists) {
return $this->api->generateRecording(['recordId' => $recordId]);
}
}
}
return false;
}
/**
* Deletes a recording of a meeting
*
* @param int $id ID of the recording
*
* @return bool
*
* @assert () === false
* @todo Also delete links and agenda items created from this recording
*/
public function deleteRecording($id)
{
if (empty($id)) {
return false;
}
$meetingData = Database::select(
'*',
$this->table,
array('where' => array('id = ?' => array($id))),
'first'
);
Event::addEvent(
'bbb_delete_record',
'meeting_id',
$id,
null,
api_get_user_id(),
api_get_course_int_id(),
api_get_session_id()
);
$delete = false;
$recordings = [];
// Check if there are recordings for this meeting
if (!empty($meetingData['remote_id'])) {
Event::addEvent(
'bbb_delete_record',
'remote_id',
$meetingData['remote_id'],
null,
api_get_user_id(),
api_get_course_int_id(),
api_get_session_id()
);
$recordings = $this->api->getRecordings(['meetingId' => $meetingData['remote_id']]);
}
if (!empty($recordings) && isset($recordings['messageKey']) && $recordings['messageKey'] == 'noRecordings') {
$delete = true;
} else {
if (!empty($recordings['records'])) {
$recordsToDelete = [];
foreach ($recordings['records'] as $record) {
$recordsToDelete[] = $record['recordId'];
}
$delete = true;
if (!empty($recordsToDelete)) {
$recordingParams = ['recordId' => implode(',', $recordsToDelete)];
Event::addEvent(
'bbb_delete_record',
'record_id_list',
implode(',', $recordsToDelete),
null,
api_get_user_id(),
api_get_course_int_id(),
api_get_session_id()
);
$result = $this->api->deleteRecordingsWithXmlResponseArray($recordingParams);
if (!empty($result) && isset($result['deleted']) && $result['deleted'] === 'true') {
$delete = true;
}
}
}
}
if ($delete) {
Database::delete(
'plugin_bbb_room',
array('meeting_id = ?' => array($id))
);
Database::delete(
$this->table,
array('id = ?' => array($id))
);
}
return $delete;
}
/**
* Creates a link in the links tool from the given videoconference recording
*
* @param int $id 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 copyRecordingToLinkTool($id)
{
if (empty($id)) {
return false;
}
//$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)) {
if (isset($records['message']) && !empty($records['message'])) {
if ($records['messageKey'] == 'noRecordings') {
$recordArray[] = $this->plugin->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 video conference 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);
}
/**
* Checks if the video conference plugin is properly configured
* @return bool True if plugin has a host and a salt, false otherwise
* @assert () === false
*/
public function isServerConfigured()
{
$host = $this->plugin->get('host');
if (empty($host)) {
return false;
}
$salt = $this->plugin->get('salt');
if (empty($salt)) {
return false;
}
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 = ? AND access_url = ?' => array(1, $this->accessUrl))),
'first'
);
return $meetingList['count'];
}
/**
* @param string $url
*/
public function redirectToBBB($url)
{
if (file_exists(__DIR__.'/../config.vm.php')) {
// Using VM
echo Display::url($this->plugin->get_lang('ClickToContinue'), $url);
exit;
} else {
// Classic
header("Location: $url");
exit;
}
}
/**
* @return string
*/
public function getConferenceUrl()
{
return api_get_path(WEB_PLUGIN_PATH).'bbb/start.php?launch=1&'.$this->getUrlParams();
}
/**
* Get the meeting info from DB by its name
*
* @param string $name
*
* @return array
*/
public function findMeetingByName($name)
{
$meetingData = Database::select(
'*',
'plugin_bbb_meeting',
array('where' => array('meeting_name = ? AND status = 1 ' => $name)),
'first'
);
return $meetingData;
}
/**
* Get the meeting info from DB by its name
*
* @param int $id
*
* @return array
*/
public function getMeeting($id)
{
$meetingData = Database::select(
'*',
'plugin_bbb_meeting',
array('where' => array('id = ?' => $id)),
'first'
);
return $meetingData;
}
/**
* Get the meeting info.
*
* @param int $id
*
* @return array
*/
public function getMeetingByRemoteId($id)
{
$meetingData = Database::select(
'*',
'plugin_bbb_meeting',
array('where' => array('remote_id = ?' => $id)),
'first'
);
return $meetingData;
}
/**
* @param int $meetingId
*
* @return array
*/
public function findConnectedMeetingParticipants($meetingId)
{
$meetingData = Database::select(
'*',
'plugin_bbb_room',
array('where' => array('meeting_id = ? AND in_at IS NOT NULL' => $meetingId))
);
$participantIds = [];
$return = [];
foreach ($meetingData as $participantInfo) {
if (in_array($participantInfo['participant_id'], $participantIds)) {
continue;
}
$participantIds[] = $participantInfo['participant_id'];
$return[] = [
'id' => $participantInfo['id'],
'meeting_id' => $participantInfo['meeting_id'],
'participant' => api_get_user_entity($participantInfo['participant_id']),
'in_at' => $participantInfo['in_at'],
'out_at' => $participantInfo['out_at'],
];
}
return $return;
}
/**
* Check if the meeting has a capture.m4v video file. If exists then the has_video_m4v field is updated
*
* @param int $meetingId
*
* @return bool
*/
public function checkDirectMeetingVideoUrl($meetingId)
{
$meetingInfo = Database::select(
'*',
'plugin_bbb_meeting',
[
'where' => ['id = ?' => intval($meetingId)],
],
'first'
);
if (!isset($meetingInfo['video_url'])) {
return false;
}
$hasCapture = SocialManager::verifyUrl($meetingInfo['video_url'].'/capture.m4v');
if ($hasCapture) {
return Database::update(
'plugin_bbb_meeting',
['has_video_m4v' => true],
['id = ?' => intval($meetingId)]
);
}
return $hasCapture;
}
}