mirror of
https://github.com/fusionpbx/fusionpbx.git
synced 2025-12-30 00:53:50 +00:00
* remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove closing tag * remove invalid method params * remove closing tag * remove closing tag * Update auto_loader to load each class file in the project Update the auto_loader class to use an include statement on each file in the project to load the class within the file. This will allow mismatched names within the file to be loaded and mapped according to the declaration instead of the filename. The class is then checked against the parsed classes from the PHP engine so that namespaces are available and mapped to the file they were declared in. An update was also made to the search algorithm used to find a file that was not already loaded by collapsing the array to have only valid matches to increase performance on a cache miss. Logging within the auto_loader has been moved to a function. Multiple files were modified to allow the include statement. When the class has the `if(class_exists())` statement, the auto_loader is called to check for the class. This caused an infinite loop scenario so all wrappers have been removed. The auto_loader will now break the loop by directly modifying the internal classes array instead of trying to restart with the 'reload_classes' method. - APCu is used to cache classes so any loading of the classes is done only once. To clear the APCu cache, restart php-fpm or call the auto_loader::clear_cache() function. - Cache file is used when APCu is not available. To clear the cache remove it from the tmp folder or call the auto_loader::clear_cache() function. - All classes must no longer have a class_exists wrapper to benefit from the performance boost. - Classes should not be directly included when the auto_loader is used. * remove include statement of class file * Update destinations.php
710 lines
29 KiB
PHP
710 lines
29 KiB
PHP
<?php
|
|
/*
|
|
FusionPBX
|
|
Version: MPL 1.1
|
|
|
|
The contents of this file are subject to the Mozilla Public License Version
|
|
1.1 (the "License"); you may not use this file except in compliance with
|
|
the License. You may obtain a copy of the License at
|
|
http://www.mozilla.org/MPL/
|
|
|
|
Software distributed under the License is distributed on an "AS IS" basis,
|
|
WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
for the specific language governing rights and limitations under the
|
|
License.
|
|
|
|
The Original Code is FusionPBX
|
|
|
|
The Initial Developer of the Original Code is
|
|
Mark J Crane <markjcrane@fusionpbx.com>
|
|
Portions created by the Initial Developer are Copyright (C) 2008-2024
|
|
the Initial Developer. All Rights Reserved.
|
|
|
|
Contributor(s):
|
|
Mark J Crane <markjcrane@fusionpbx.com>
|
|
James Rose <james.o.rose@gmail.com>
|
|
*/
|
|
|
|
//includes files
|
|
require_once dirname(__DIR__, 2) . "/resources/require.php";
|
|
require_once "resources/check_auth.php";
|
|
|
|
//check permissions
|
|
if (permission_exists('music_on_hold_view')) {
|
|
//access granted
|
|
}
|
|
else {
|
|
echo "access denied";
|
|
exit;
|
|
}
|
|
|
|
//add multi-lingual support
|
|
$language = new text;
|
|
$text = $language->get();
|
|
|
|
//add additional variables
|
|
$search = $_GET["search"] ?? '';
|
|
$show = $_GET['show'] ?? '';
|
|
|
|
//get the music_on_hold array
|
|
$sql = "select * from v_music_on_hold ";
|
|
$sql .= "where true ";
|
|
if ($show != "all" || !permission_exists('music_on_hold_all')) {
|
|
$sql .= "and (domain_uuid = :domain_uuid or domain_uuid is null) ";
|
|
$parameters['domain_uuid'] = $_SESSION['domain_uuid'];
|
|
}
|
|
if (permission_exists('music_on_hold_domain')) {
|
|
$sql .= "or domain_uuid is null ";
|
|
}
|
|
$sql .= "order by domain_uuid desc, music_on_hold_name asc, music_on_hold_rate asc";
|
|
$database = new database;
|
|
$streams = $database->select($sql, $parameters ?? null, 'all');
|
|
unset($sql, $parameters);
|
|
|
|
//get the http post data
|
|
if (!empty($_POST['moh'])) {
|
|
$action = $_POST['action'];
|
|
$moh = $_POST['moh'];
|
|
}
|
|
|
|
//process the http post data by action
|
|
if (!empty($action) && !empty($moh)) {
|
|
switch ($action) {
|
|
case 'delete':
|
|
if (permission_exists('music_on_hold_delete')) {
|
|
$obj = new switch_music_on_hold;
|
|
$obj->delete($moh);
|
|
}
|
|
break;
|
|
}
|
|
|
|
header('Location: music_on_hold.php');
|
|
exit;
|
|
}
|
|
|
|
//get order and order by and sanitize the values
|
|
$order_by = $_GET["order_by"] ?? '';
|
|
$order = $_GET["order"] ?? '';
|
|
|
|
//download music on hold file
|
|
if (!empty($_GET['action'])
|
|
&& $_GET['action'] == "download"
|
|
&& is_uuid($_GET['id'])
|
|
&& !empty($streams)) {
|
|
|
|
//get the uuid
|
|
$stream_uuid = $_GET['id'];
|
|
|
|
//get the record
|
|
foreach($streams as $row) {
|
|
if ($stream_uuid == $row['music_on_hold_uuid']) {
|
|
$stream_domain_uuid = $row['domain_uuid'];
|
|
$stream_name = $row['music_on_hold_name'];
|
|
$stream_path = $row['music_on_hold_path'];
|
|
break;
|
|
}
|
|
}
|
|
|
|
//replace the sounds_dir variable in the path
|
|
$stream_path = str_replace('$${sounds_dir}', $_SESSION['switch']['sounds']['dir'], $stream_path);
|
|
$stream_path = str_replace('..', '', $stream_path);
|
|
|
|
//get the file and sanitize it
|
|
$stream_file = basename($_GET['file']);
|
|
$search = array('..', '/', ':');
|
|
$stream_file = str_replace($search, '', $stream_file);
|
|
|
|
//join the path and file name
|
|
$stream_full_path = path_join($stream_path, $stream_file);
|
|
|
|
//download the file
|
|
if (file_exists($stream_full_path)) {
|
|
|
|
$fd = fopen($stream_full_path, "rb");
|
|
if (!empty($_GET['t']) && $_GET['t'] == "bin") {
|
|
header("Content-Type: application/force-download");
|
|
header("Content-Type: application/octet-stream");
|
|
header("Content-Type: application/download");
|
|
header("Content-Description: File Transfer");
|
|
}
|
|
else {
|
|
$stream_file_ext = pathinfo($stream_file, PATHINFO_EXTENSION);
|
|
switch ($stream_file_ext) {
|
|
case "wav" : header("Content-Type: audio/x-wav"); break;
|
|
case "mp3" : header("Content-Type: audio/mpeg"); break;
|
|
case "ogg" : header("Content-Type: audio/ogg"); break;
|
|
}
|
|
}
|
|
header('Content-Disposition: attachment; filename="'.$stream_file.'"');
|
|
header("Cache-Control: no-cache, must-revalidate"); // HTTP/1.1
|
|
header("Expires: Sat, 26 Jul 1997 05:00:00 GMT"); // Date in the past
|
|
if (!empty($_GET['t']) && $_GET['t'] == "bin") {
|
|
header("Content-Length: ".filesize($stream_full_path));
|
|
}
|
|
ob_clean();
|
|
|
|
//content-range
|
|
if (isset($_SERVER['HTTP_RANGE']) && (empty($_GET['t']) || $_GET['t'] != "bin")) {
|
|
range_download($stream_full_path);
|
|
}
|
|
|
|
fpassthru($fd);
|
|
}
|
|
exit;
|
|
}
|
|
|
|
//upload music on hold file
|
|
if (!empty($_POST['action']) && $_POST['action'] == 'upload'
|
|
&& !empty($_FILES)
|
|
&& is_uploaded_file($_FILES['file']['tmp_name'])
|
|
) {
|
|
|
|
//validate the token
|
|
$token = new token;
|
|
if (!$token->validate($_SERVER['PHP_SELF'])) {
|
|
message::add($text['message-invalid_token'],'negative');
|
|
header('Location: music_on_hold.php');
|
|
exit;
|
|
}
|
|
|
|
//determine name
|
|
if (!empty($_POST['name_new'])) {
|
|
//set the action
|
|
$action = 'add';
|
|
//get the stream_name
|
|
$stream_name = $_POST['name_new'];
|
|
//get the rate
|
|
$stream_rate = is_numeric($_POST['rate']) ? $_POST['rate'] : '';
|
|
}
|
|
else {
|
|
//get the stream uuid
|
|
$stream_uuid = $_POST['name'];
|
|
//find the matching stream
|
|
if (!empty($streams) && @sizeof($streams) != 0) {
|
|
foreach ($streams as $row) {
|
|
if ($stream_uuid == $row['music_on_hold_uuid']) {
|
|
//set the action
|
|
$action = 'update';
|
|
//set the variables
|
|
$stream_domain_uuid = $row['domain_uuid'];
|
|
$stream_name = $row['music_on_hold_name'];
|
|
$stream_path = $row['music_on_hold_path'];
|
|
$stream_rate = $row['music_on_hold_rate'];
|
|
$stream_shuffle = $row['music_on_hold_shuffle'];
|
|
$stream_channels = $row['music_on_hold_channels'];
|
|
$stream_internal = $row['music_on_hold_interval'];
|
|
$stream_timer_name = $row['music_on_hold_timer_name'];
|
|
$stream_chime_list = $row['music_on_hold_chime_list'];
|
|
$stream_chime_freq = $row['music_on_hold_chime_freq'];
|
|
$stream_chime_max = $row['music_on_hold_chime_max'];
|
|
$stream_rate = $row['music_on_hold_rate'];
|
|
//end the loop
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//get remaining values
|
|
$stream_file_name_temp = $_FILES['file']['tmp_name'];
|
|
$stream_file_name = $_FILES['file']['name'];
|
|
$stream_file_ext = strtolower(pathinfo($stream_file_name, PATHINFO_EXTENSION));
|
|
|
|
//check file type
|
|
$valid_file_type = ($stream_file_ext == 'wav' || $stream_file_ext == 'mp3' || $stream_file_ext == 'ogg') ? true : false;
|
|
|
|
//proceed for valid file type
|
|
if ($stream_file_ext == 'wav' || $stream_file_ext == 'mp3' || $stream_file_ext == 'ogg') {
|
|
|
|
//strip slashes, replace spaces
|
|
$slashes = ["/","\\"];
|
|
$stream_file_name = str_replace($slashes, '', $stream_file_name);
|
|
$stream_file_name = str_replace(' ', '-', $stream_file_name);
|
|
if ($action == "add") {
|
|
$stream_name = str_replace($slashes, '', $stream_name);
|
|
$stream_name = str_replace(' ', '_', $stream_name);
|
|
}
|
|
|
|
//detect auto rate
|
|
if ($stream_rate == '') {
|
|
$path_rate = '48000';
|
|
$stream_rate_auto = true;
|
|
}
|
|
else {
|
|
$path_rate = $stream_rate;
|
|
$stream_rate_auto = false;
|
|
}
|
|
|
|
//define default path
|
|
if ($action == "add") {
|
|
$stream_path = path_join($_SESSION['switch']['sounds']['dir'], 'music', $_SESSION['domain_name'], $stream_name, $path_rate);
|
|
$stream_path = str_replace('.loc', '._loc', $stream_path); // 14.03.22 freeswitch bug
|
|
}
|
|
|
|
//find whether the path already exists
|
|
$stream_new_name = true;
|
|
if (!empty($streams) && @sizeof($streams) != 0) {
|
|
foreach ($streams as $row) {
|
|
$alternate_path = str_replace('$${sounds_dir}', $_SESSION['switch']['sounds']['dir'], $row['music_on_hold_path']);
|
|
if ($stream_path == $row['music_on_hold_path'] || $stream_path == $alternate_path) {
|
|
$stream_new_name = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//set the variables
|
|
$stream_path = str_replace('$${sounds_dir}', $_SESSION['switch']['sounds']['dir'], $stream_path);
|
|
|
|
//add new path
|
|
if ($stream_new_name) {
|
|
$stream_uuid = uuid();
|
|
$array['music_on_hold'][0]['music_on_hold_uuid'] = $stream_uuid;
|
|
$array['music_on_hold'][0]['domain_uuid'] = $domain_uuid;
|
|
$array['music_on_hold'][0]['music_on_hold_name'] = $stream_name;
|
|
$array['music_on_hold'][0]['music_on_hold_path'] = $stream_path;
|
|
$array['music_on_hold'][0]['music_on_hold_rate'] = strlen($stream_rate) != 0 ? $stream_rate : null;
|
|
$array['music_on_hold'][0]['music_on_hold_shuffle'] = 'false';
|
|
$array['music_on_hold'][0]['music_on_hold_channels'] = 1;
|
|
$array['music_on_hold'][0]['music_on_hold_interval'] = 20;
|
|
$array['music_on_hold'][0]['music_on_hold_timer_name'] = 'soft';
|
|
$array['music_on_hold'][0]['music_on_hold_chime_list'] = null;
|
|
$array['music_on_hold'][0]['music_on_hold_chime_freq'] = null;
|
|
$array['music_on_hold'][0]['music_on_hold_chime_max'] = null;
|
|
|
|
$p = permissions::new();
|
|
$p->add('music_on_hold_add', 'temp');
|
|
|
|
$database = new database;
|
|
$database->app_name = 'music_on_hold';
|
|
$database->app_uuid = '1dafe0f8-c08a-289b-0312-15baf4f20f81';
|
|
$database->save($array);
|
|
unset($array);
|
|
|
|
$p->delete('music_on_hold_add', 'temp');
|
|
}
|
|
|
|
//check target folder, move uploaded file
|
|
if (!is_dir($stream_path)) {
|
|
mkdir($stream_path, 0770, true);
|
|
|
|
// 14.03.22 freeswitch bug - shouldn't be needed with freeswitch 1.10.8
|
|
if (preg_match('|^(/usr/share/freeswitch/sounds/music/(.*?\._loc.*?))/|', $stream_path, $m)) {
|
|
$fs_bug_target = $m[2];
|
|
$fs_bug_link = str_replace('._loc', '.loc', $m[1]);
|
|
symlink($fs_bug_target, $fs_bug_link);
|
|
}
|
|
}
|
|
if (is_dir($stream_path)) {
|
|
if (copy($stream_file_name_temp, $stream_path.'/'.$stream_file_name)) {
|
|
@unlink($stream_file_name_temp);
|
|
}
|
|
}
|
|
|
|
//set message
|
|
message::add($text['message-upload_completed']);
|
|
|
|
//clear the cache
|
|
$cache = new cache;
|
|
$cache->delete("configuration:local_stream.conf");
|
|
|
|
$music = new switch_music_on_hold;
|
|
$music->reload();
|
|
|
|
}
|
|
//set message for unsupported file type
|
|
else {
|
|
message::add($text['message-unsupported_file_type']);
|
|
}
|
|
|
|
//redirect
|
|
header("Location: music_on_hold.php");
|
|
exit;
|
|
}
|
|
|
|
//create token
|
|
$object = new token;
|
|
$token = $object->create($_SERVER['PHP_SELF']);
|
|
|
|
//include the header
|
|
$document['title'] = $text['title-music_on_hold'];
|
|
require_once "resources/header.php";
|
|
|
|
//script
|
|
echo "<script language='JavaScript' type='text/javascript'>\n";
|
|
|
|
//file type check
|
|
echo " function check_file_type(file_input) {\n";
|
|
echo " file_ext = file_input.value.substr((~-file_input.value.lastIndexOf('.') >>> 0) + 2);\n";
|
|
echo " if (file_ext != 'mp3' && file_ext != 'wav' && file_ext != 'ogg' && file_ext != '') {\n";
|
|
echo " display_message(\"".$text['message-unsupported_file_type']."\", 'negative', '2750');\n";
|
|
echo " }\n";
|
|
echo " }\n";
|
|
|
|
//custom name (category)
|
|
echo " function name_mode(mode) {\n";
|
|
echo " if (mode == 'new') {\n";
|
|
echo " document.getElementById('name_select').style.display='none';\n";
|
|
echo " document.getElementById('btn_new').style.display='none';\n";
|
|
echo " document.getElementById('name_new').style.display='';\n";
|
|
echo " document.getElementById('btn_select').style.display='';\n";
|
|
echo " document.getElementById('rate').style.display='';\n";
|
|
echo " document.getElementById('name_new').focus();\n";
|
|
echo " }\n";
|
|
echo " else if (mode == 'select') {\n";
|
|
echo " document.getElementById('name_new').style.display='none';\n";
|
|
echo " document.getElementById('name_new').value = '';\n";
|
|
echo " document.getElementById('rate').style.display='none';\n";
|
|
echo " document.getElementById('btn_select').style.display='none';\n";
|
|
echo " document.getElementById('name_select').selectedIndex = 0;\n";
|
|
echo " document.getElementById('name_select').style.display='';\n";
|
|
echo " document.getElementById('btn_new').style.display='';\n";
|
|
echo " }\n";
|
|
echo " }\n";
|
|
|
|
echo "</script>";
|
|
|
|
//show the content
|
|
echo "<div class='action_bar' id='action_bar'>\n";
|
|
echo " <div class='heading'><b>".$text['title-music_on_hold']."</b></div>\n";
|
|
echo " <div class='actions'>\n";
|
|
if (permission_exists('music_on_hold_add')) {
|
|
$modify_add_action = empty($streams) || @sizeof($streams) == 0 ? "name_mode('new'); $('#btn_select').hide();" : null; //hide categories select box when none exist
|
|
echo "<form id='form_upload' class='inline' method='post' enctype='multipart/form-data'>\n";
|
|
echo "<input name='action' type='hidden' value='upload'>\n";
|
|
echo "<input type='hidden' name='".$token['name']."' value='".$token['hash']."'>\n";
|
|
echo button::create(['type'=>'button','label'=>$text['button-add'],'icon'=>$_SESSION['theme']['button_icon_add'],'id'=>'btn_add','onclick'=>"$(this).fadeOut(250, function(){ ".$modify_add_action." $('span#form_upload').fadeIn(250); });"]);
|
|
echo "<span id='form_upload' style='display: none;'>";
|
|
echo button::create(['label'=>$text['button-cancel'],'icon'=>$_SESSION['theme']['button_icon_cancel'],'type'=>'button','id'=>'btn_upload_cancel','onclick'=>"$('span#form_upload').fadeOut(250, function(){ name_mode('select'); document.getElementById('form_upload').reset(); $('#btn_add').fadeIn(250) });"]);
|
|
//name (category)
|
|
echo "<select name='name' id='name_select' class='formfld' style='width: auto; margin: 0;'>\n";
|
|
echo " <option value='' selected='selected' disabled='disabled'>".$text['label-category']."</option>\n";
|
|
|
|
if (permission_exists('music_on_hold_domain')) {
|
|
echo " <optgroup label='".$text['option-global']."'>\n";
|
|
if (!empty($streams) && @sizeof($streams) != 0) {
|
|
foreach ($streams as $row) {
|
|
if (empty($row['domain_uuid'])) {
|
|
if (empty($row['music_on_hold_rate'])) { $option_name = $row['music_on_hold_name']; }
|
|
if (!empty($row['music_on_hold_rate'])) { $option_name = $row['music_on_hold_name'] .'/'.$row['music_on_hold_rate']; }
|
|
echo " <option value='".escape($row['music_on_hold_uuid'])."'>".escape($option_name)."</option>\n";
|
|
}
|
|
}
|
|
}
|
|
echo " </optgroup>\n";
|
|
}
|
|
$local_found = false;
|
|
if (!empty($streams) && @sizeof($streams) != 0) {
|
|
foreach ($streams as $row) {
|
|
if (is_uuid($row['domain_uuid'])) {
|
|
$local_found = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if ($local_found) {
|
|
if (permission_exists('music_on_hold_domain')) {
|
|
echo " <optgroup label='".$text['option-local']."'>\n";
|
|
}
|
|
if (!empty($streams) && @sizeof($streams) != 0) {
|
|
foreach ($streams as $row) {
|
|
if (!empty($row['domain_uuid'])) {
|
|
if (empty($row['music_on_hold_rate'])) { $option_name = $row['music_on_hold_name']; }
|
|
if (!empty($row['music_on_hold_rate'])) { $option_name = $row['music_on_hold_name'] .'/'.$row['music_on_hold_rate']; }
|
|
echo " <option value='".escape($row['music_on_hold_uuid'])."'>".escape($option_name)."</option>\n";
|
|
}
|
|
}
|
|
}
|
|
if (permission_exists('music_on_hold_domain')) {
|
|
echo " </optgroup>\n";
|
|
}
|
|
}
|
|
echo " </select>";
|
|
echo "<input class='formfld' style='width: 100px; margin: 0; display: none;' type='text' name='name_new' id='name_new' maxlength='255' placeholder=\"".$text['label-category']."\" value=''>";
|
|
//rate
|
|
echo "<select id='rate' name='rate' class='formfld' style='display: none; width: auto; margin: 0;'>\n";
|
|
echo " <option value=''>".$text['option-default']."</option>\n";
|
|
echo " <option value='8000'>8 kHz</option>\n";
|
|
echo " <option value='16000'>16 kHz</option>\n";
|
|
echo " <option value='32000'>32 kHz</option>\n";
|
|
echo " <option value='48000'>48 kHz</option>\n";
|
|
echo "</select>";
|
|
echo button::create(['type'=>'button','title'=>!empty($text['label-new']),'icon'=>$_SESSION['theme']['button_icon_add'],'id'=>'btn_new','onclick'=>"name_mode('new');"]);
|
|
echo button::create(['type'=>'button','title'=>$text['label-select'],'icon'=>'list','id'=>'btn_select','style'=>'display: none;','onclick'=>"name_mode('select');"]);
|
|
//file
|
|
echo "<input type='text' class='txt' style='width: 100px; cursor: pointer; margin: 0;' id='filename' placeholder='Select...' onclick=\"document.getElementById('file').click(); this.blur();\" onfocus='this.blur();'>";
|
|
echo "<input type='file' id='file' name='file' style='display: none;' accept='.wav,.mp3,.ogg' onchange=\"document.getElementById('filename').value = this.files.item(0).name; check_file_type(this);\">";
|
|
//submit
|
|
$margin_right = permission_exists('music_on_hold_delete') ? 'margin-right: 15px;' : null;
|
|
echo button::create(['type'=>'submit','label'=>$text['button-upload'],'style'=>$margin_right,'icon'=>$_SESSION['theme']['button_icon_upload']]);
|
|
echo "</span>\n";
|
|
echo "</form>";
|
|
}
|
|
if (permission_exists('music_on_hold_all')) {
|
|
if ($show == 'all') {
|
|
echo " <input type='hidden' name='show' value='all'>";
|
|
}
|
|
else {
|
|
echo button::create(['type'=>'button','label'=>$text['button-show_all'],'icon'=>$_SESSION['theme']['button_icon_all'],'link'=>'?type=&show=all'.(!empty($search) ? "&search=".urlencode($search) : null)]);
|
|
}
|
|
}
|
|
if (permission_exists('music_on_hold_delete') && $streams) {
|
|
echo button::create(['type'=>'button','label'=>$text['button-delete'],'icon'=>$_SESSION['theme']['button_icon_delete'],'id'=>'btn_delete','name'=>'btn_delete','style'=>'display: none;','onclick'=>"modal_open('modal-delete','btn_delete');"]);
|
|
}
|
|
echo " </div>\n";
|
|
echo " <div style='clear: both;'></div>\n";
|
|
echo "</div>\n";
|
|
|
|
if (permission_exists('music_on_hold_delete') && $streams) {
|
|
echo modal::create(['id'=>'modal-delete','type'=>'delete','actions'=>button::create(['type'=>'button','label'=>$text['button-continue'],'icon'=>'check','id'=>'btn_delete','style'=>'float: right; margin-left: 15px;','collapse'=>'never','onclick'=>"modal_close(); list_action_set('delete'); list_form_submit('form_list');"])]);
|
|
}
|
|
|
|
echo $text['title_description-music_on_hold']."\n";
|
|
echo "<br /><br />\n";
|
|
|
|
echo "<form id='form_list' method='post'>\n";
|
|
echo "<input type='hidden' id='action' name='action' value=''>\n";
|
|
|
|
//show the array of data
|
|
if (!empty($streams) && @sizeof($streams) != 0) {
|
|
$previous_name = '';
|
|
|
|
//loop through the array
|
|
$x = 0;
|
|
foreach ($streams as $row) {
|
|
|
|
//set the variables
|
|
$music_on_hold_name = $row['music_on_hold_name'];
|
|
$music_on_hold_rate = $row['music_on_hold_rate'];
|
|
|
|
//add the name (category)
|
|
if ($previous_name != $music_on_hold_name) {
|
|
echo "<b><i>".escape($music_on_hold_name)."</i></b>".(!is_uuid($row['domain_uuid']) ? ' ('.$text['label-global'].')' : null)."<br />\n";
|
|
}
|
|
|
|
//determine if rate was set to auto or not
|
|
$auto_rate = empty($music_on_hold_rate) ? true : false;
|
|
|
|
//determine icons to show
|
|
$stream_icons = array();
|
|
$i = 0;
|
|
if (permission_exists('music_on_hold_path')) {
|
|
$stream_icons[$i]['icon'] = 'fa-folder-open';
|
|
$stream_icons[$i]['title'] = $row['music_on_hold_name'];
|
|
$i++;
|
|
}
|
|
if ($row['music_on_hold_shuffle'] == 'true') {
|
|
$stream_icons[$i]['icon'] = 'fa-random';
|
|
$stream_icons[$i]['title'] = $text['label-shuffle'];
|
|
$i++;
|
|
}
|
|
if (!empty($row['music_on_hold_chime_list'])) {
|
|
$stream_icons[$i]['icon'] = 'fa-bell';
|
|
$stream_icons[$i]['title'] = $text['label-chime_list'].': '.$row['music_on_hold_chime_list'];
|
|
$i++;
|
|
}
|
|
if ($row['music_on_hold_channels'] == '2') {
|
|
$stream_icons[$i]['icon'] = 'fa-headphones';
|
|
$stream_icons[$i]['title'] = $text['label-stereo'];
|
|
$stream_icons[$i]['margin'] = 6;
|
|
$i++;
|
|
}
|
|
if (!empty($stream_icons)) {
|
|
$icons = '';
|
|
foreach ($stream_icons as $stream_icon) {
|
|
$icons .= "<span class='fas ".$stream_icon['icon']." icon_body' title='".escape($stream_icon['title'])."' style='width: 12px; height: 12px; margin-left: ".(!empty($stream_icon['margin']) ? $stream_icon['margin'] : 8)."px; vertical-align: text-top; cursor: help;'></span>";
|
|
}
|
|
}
|
|
|
|
//set the rate label
|
|
$stream_rate = $auto_rate ? $text['option-default'] : ($music_on_hold_rate/1000).' kHz';
|
|
if (permission_exists('music_on_hold_edit')) {
|
|
$stream_details = "<a href='music_on_hold_edit.php?id=".urlencode($row['music_on_hold_uuid'])."' class='default-color'>".$stream_rate.'</a> '.$icons;
|
|
}
|
|
else {
|
|
$stream_details = $stream_rate.' '.$icons;
|
|
}
|
|
|
|
//get the music on hold path and files
|
|
$stream_path = str_replace("\$\${sounds_dir}",$_SESSION['switch']['sounds']['dir'] ?? '', $row['music_on_hold_path']);
|
|
if (file_exists($stream_path)) {
|
|
$stream_files = array_merge(glob($stream_path.'/*.wav'), glob($stream_path.'/*.mp3'), glob($stream_path.'/*.ogg'));
|
|
}
|
|
|
|
//start the table
|
|
echo "<div class='card'>\n";
|
|
echo " <table class='list'>\n";
|
|
echo " <tr class='list-header'>\n";
|
|
if (permission_exists('music_on_hold_delete')) {
|
|
echo " <th class='checkbox'>\n";
|
|
echo " <input type='checkbox' id='checkbox_all_".$row['music_on_hold_uuid']."' name='checkbox_all' onclick=\"list_all_toggle('".$row['music_on_hold_uuid']."'); document.getElementById('checkbox_all_".$row['music_on_hold_uuid']."_hidden').value = this.checked ? 'true' : ''; checkbox_on_change(this);\">\n";
|
|
echo " <input type='hidden' id='checkbox_all_".$row['music_on_hold_uuid']."_hidden' name='moh[".$row['music_on_hold_uuid']."][checked]'>\n";
|
|
echo " </th>\n";
|
|
}
|
|
if ($show == "all" && permission_exists('music_on_hold_all')) {
|
|
echo th_order_by('domain_name', $text['label-domain'], $order_by, $order, $param ?? null, "class='shrink'");
|
|
}
|
|
echo " <th class='pct-50'>".$stream_details."</th>\n";
|
|
echo " <th class='center shrink'>".$text['label-tools']."</th>\n";
|
|
echo " <th class='right hide-xs no-wrap pct-20'>".$text['label-file-size']."</th>\n";
|
|
echo " <th class='right hide-sm-dn pct-30'>".$text['label-uploaded']."</th>\n";
|
|
echo " </tr>";
|
|
unset($stream_icons, $icons);
|
|
|
|
//list the stream files
|
|
if (!empty($stream_files)) {
|
|
foreach ($stream_files as $stream_file_path) {
|
|
$row_uuid = uuid();
|
|
$stream_file = pathinfo($stream_file_path, PATHINFO_BASENAME);
|
|
$stream_file_size = byte_convert(filesize($stream_file_path));
|
|
$stream_file_date = date("M d, Y H:i:s", filemtime($stream_file_path));
|
|
$stream_file_ext = pathinfo($stream_file, PATHINFO_EXTENSION);
|
|
switch ($stream_file_ext) {
|
|
case "wav" : $stream_file_type = "audio/wav"; break;
|
|
case "mp3" : $stream_file_type = "audio/mpeg"; break;
|
|
case "ogg" : $stream_file_type = "audio/ogg"; break;
|
|
}
|
|
//playback progress bar
|
|
echo "<tr class='list-row' id='recording_progress_bar_".$row_uuid."' style='display: none;' onclick=\"recording_seek(event,'".escape($row_uuid)."')\"><td id='playback_progress_bar_background_".escape($row_uuid)."' class='playback_progress_bar_background' colspan='5'><span class='playback_progress_bar' id='recording_progress_".$row_uuid."'></span></td></tr>\n";
|
|
echo "<tr class='list-row' style='display: none;'><td></td></tr>\n"; // dummy row to maintain alternating background color
|
|
$list_row_link = "javascript:recording_play('".$row_uuid."','".urlencode($stream_file)."');";
|
|
echo "<tr class='list-row' href=\"".$list_row_link."\">\n";
|
|
if (permission_exists('music_on_hold_delete')) {
|
|
echo " <td class='checkbox'>\n";
|
|
echo " <input type='checkbox' name='moh[".$row['music_on_hold_uuid']."][$x][checked]' id='checkbox_".$x."' class='checkbox_".$row['music_on_hold_uuid']."' value='true' onclick=\"checkbox_on_change(this); if (!this.checked) { document.getElementById('checkbox_all_".$row['music_on_hold_uuid']."').checked = false; }\">\n";
|
|
echo " <input type='hidden' name='moh[".$row['music_on_hold_uuid']."][$x][file_name]' value=\"".escape($stream_file)."\" />\n";
|
|
echo " </td>\n";
|
|
}
|
|
if ($show == "all" && permission_exists('music_on_hold_all')) {
|
|
if (!empty($_SESSION['domains'][$row['domain_uuid']]['domain_name'])) {
|
|
$domain = $_SESSION['domains'][$row['domain_uuid']]['domain_name'];
|
|
}
|
|
else {
|
|
$domain = $text['label-global'];
|
|
}
|
|
echo " <td>".escape($domain)."</td>\n";
|
|
}
|
|
echo " <td class='overflow'>".escape($stream_file)."</td>\n";
|
|
echo " <td class='button center no-link no-wrap'>";
|
|
echo "<audio id='recording_audio_".$row_uuid."' style='display: none;' preload='none' ontimeupdate=\"update_progress('".$row_uuid."')\" onended=\"recording_reset('".$row_uuid."');\" src='music_on_hold.php?action=download&id=".escape($row['music_on_hold_uuid'])."&file=".urlencode($stream_file)."' type='".$stream_file_type."'></audio>";
|
|
echo button::create(['type'=>'button','title'=>$text['label-play'].' / '.$text['label-pause'],'icon'=>$_SESSION['theme']['button_icon_play'],'id'=>'recording_button_'.$row_uuid,'onclick'=>"recording_play('".$row_uuid."','".urlencode($stream_file)."');"]);
|
|
echo button::create(['type'=>'button','title'=>$text['label-download'],'icon'=>$_SESSION['theme']['button_icon_download'],'link'=>"?action=download&id=".urlencode($row['music_on_hold_uuid'])."&file=".urlencode($stream_file)]);
|
|
echo " </td>\n";
|
|
echo " <td class='right no-wrap hide-xs'>".escape($stream_file_size)."</td>\n";
|
|
echo " <td class='right no-wrap hide-sm-dn'>".escape($stream_file_date)."</td>\n";
|
|
echo "</tr>\n";
|
|
$x++;
|
|
}
|
|
}
|
|
|
|
echo " </table>\n";
|
|
echo "</div>\n";
|
|
echo "<br />\n";
|
|
|
|
//set the previous music_on_hold_name
|
|
$previous_name = $music_on_hold_name;
|
|
|
|
}
|
|
unset($streams, $row);
|
|
|
|
}
|
|
|
|
echo "<input type='hidden' name='".$token['name']."' value='".$token['hash']."'>\n";
|
|
echo "</form>\n";
|
|
|
|
//include the footer
|
|
require_once "resources/footer.php";
|
|
|
|
//define the download function (helps safari play audio sources)
|
|
function range_download($file) {
|
|
$fp = @fopen($file, 'rb');
|
|
|
|
$size = filesize($file); // File size
|
|
$length = $size; // Content length
|
|
$start = 0; // Start byte
|
|
$end = $size - 1; // End byte
|
|
// Now that we've gotten so far without errors we send the accept range header
|
|
/* At the moment we only support single ranges.
|
|
* Multiple ranges requires some more work to ensure it works correctly
|
|
* and comply with the spesifications: http://www.w3.org/Protocols/rfc2616/rfc2616-sec19.html#sec19.2
|
|
*
|
|
* Multirange support annouces itself with:
|
|
* header('Accept-Ranges: bytes');
|
|
*
|
|
* Multirange content must be sent with multipart/byteranges mediatype,
|
|
* (mediatype = mimetype)
|
|
* as well as a boundry header to indicate the various chunks of data.
|
|
*/
|
|
header("Accept-Ranges: 0-$length");
|
|
// header('Accept-Ranges: bytes');
|
|
// multipart/byteranges
|
|
// http://www.w3.org/Protocols/rfc2616/rfc2616-sec19.html#sec19.2
|
|
if (isset($_SERVER['HTTP_RANGE'])) {
|
|
|
|
$c_start = $start;
|
|
$c_end = $end;
|
|
// Extract the range string
|
|
list(, $range) = explode('=', $_SERVER['HTTP_RANGE'], 2);
|
|
// Make sure the client hasn't sent us a multibyte range
|
|
if (strpos($range, ',') !== false) {
|
|
// (?) Shoud this be issued here, or should the first
|
|
// range be used? Or should the header be ignored and
|
|
// we output the whole content?
|
|
header('HTTP/1.1 416 Requested Range Not Satisfiable');
|
|
header("Content-Range: bytes $start-$end/$size");
|
|
// (?) Echo some info to the client?
|
|
exit;
|
|
}
|
|
// If the range starts with an '-' we start from the beginning
|
|
// If not, we forward the file pointer
|
|
// And make sure to get the end byte if spesified
|
|
if (!empty($range0) && $range0 == '-') {
|
|
// The n-number of the last bytes is requested
|
|
$c_start = $size - substr($range, 1);
|
|
}
|
|
else {
|
|
$range = explode('-', $range);
|
|
$c_start = $range[0];
|
|
$c_end = (isset($range[1]) && is_numeric($range[1])) ? $range[1] : $size;
|
|
}
|
|
/* Check the range and make sure it's treated according to the specs.
|
|
* http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html
|
|
*/
|
|
// End bytes can not be larger than $end.
|
|
$c_end = ($c_end > $end) ? $end : $c_end;
|
|
// Validate the requested range and return an error if it's not correct.
|
|
if ($c_start > $c_end || $c_start > $size - 1 || $c_end >= $size) {
|
|
|
|
header('HTTP/1.1 416 Requested Range Not Satisfiable');
|
|
header("Content-Range: bytes $start-$end/$size");
|
|
// (?) Echo some info to the client?
|
|
exit;
|
|
}
|
|
$start = $c_start;
|
|
$end = $c_end;
|
|
$length = $end - $start + 1; // Calculate new content length
|
|
fseek($fp, $start);
|
|
header('HTTP/1.1 206 Partial Content');
|
|
}
|
|
// Notify the client the byte range we'll be outputting
|
|
header("Content-Range: bytes $start-$end/$size");
|
|
header("Content-Length: $length");
|
|
|
|
// Start buffered download
|
|
$buffer = 1024 * 8;
|
|
while(!feof($fp) && ($p = ftell($fp)) <= $end) {
|
|
if ($p + $buffer > $end) {
|
|
// In case we're only outputtin a chunk, make sure we don't
|
|
// read past the length
|
|
$buffer = $end - $p + 1;
|
|
}
|
|
set_time_limit(0); // Reset time limit for big files
|
|
echo fread($fp, $buffer);
|
|
flush(); // Free up memory. Otherwise large files will trigger PHP's memory limit.
|
|
}
|
|
|
|
fclose($fp);
|
|
}
|
|
|
|
?>
|