-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathfunctions.inc.php
252 lines (224 loc) · 9.45 KB
/
functions.inc.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
<?php
if (!defined('FREEPBX_IS_AUTH')) { die('No direct script access allowed'); }
// License for all code of this FreePBX module can be found in the license file inside the module directory
// Portions Copyright (C) 2011 Igor Okunev
// Portions Copyright (C) 2011 Mikael Carlsson
// Copyright 2013 Schmooze Com Inc.
//
function cdr_get_config($engine) {
global $core_conf, $amp_conf, $version;
if (isset($core_conf) && is_a($core_conf, "core_conf")) {
$section = 'asteriskcdrdb';
$core_conf->addResOdbc($section, ['enabled' => 'yes']);
$core_conf->addResOdbc($section, ['dsn' => 'MySQL-asteriskcdrdb']);
$core_conf->addResOdbc($section, ['pre-connect' => 'yes']);
if((version_compare($version, "14.0", "lt") && version_compare($version, "13.14.0", "ge")) || (version_compare($version, "14.0", "ge") && version_compare($version, "14.3.0", "ge"))) {
$core_conf->addResOdbc($section, ['max_connections' => '5']);
} else {
$core_conf->addResOdbc($section, ['pooling' => 'no']);
$core_conf->addResOdbc($section, ['limit' => '1']);
}
$core_conf->addResOdbc($section, ['username' => !empty($amp_conf['CDRDBUSER']) ? $amp_conf['CDRDBUSER'] : $amp_conf['AMPDBUSER']]);
$core_conf->addResOdbc($section, ['password' => !empty($amp_conf['CDRDBPASS']) ? $amp_conf['CDRDBPASS'] : $amp_conf['AMPDBPASS']]);
$core_conf->addResOdbc($section, ['database' => !empty($amp_conf['CDRDBNAME']) ? $amp_conf['CDRDBNAME'] : 'asteriskcdrdb']);
}
if (isset($amp_conf['CDRUSEGMT']) && file_exists($amp_conf['ASTETCDIR'] . '/cdr_adaptive_odbc.conf')) {
//Parse the existing file
$cdrConf = @parse_ini_file($amp_conf['ASTETCDIR'] . '/cdr_adaptive_odbc.conf', true);
//Modify the data
$content = "";
if (empty($cdrConf)) {
return;
}
foreach ($cdrConf as $section => $data) {
$content .= "[$section]\n";
if($amp_conf['CDRUSEGMT'] == 1){
$data['usegmtime'] = 'yes';
} else {
unset($data['usegmtime']);
}
foreach ($data as $key => $value) {
if ($key == 'alias start') {
$content .= $key . " =" . $value . "\n";
} else {
$content .= $key . "=" . $value . "\n";
}
}
$content .= "\n";
}
//Rewrite the file
\FreePBX::WriteConfig()->writeConfig('cdr_adaptive_odbc.conf', $content, false);
}
if(isset($amp_conf['CDR_BATCH_ENABLE'])) {
$enable = ($amp_conf['CDR_BATCH_ENABLE'] == 1) ? 'yes' : 'no';
$batch = ($amp_conf['CDR_BATCH'] == 1) ? 'yes' : 'no';
$schedule = ($amp_conf['CDR_BATCH_SCHEDULE_ONLY'] == 1) ? 'yes' : 'no';
$bshutdown = ($amp_conf['CDR_BATCH_SAFE_SHUT_DOWN'] == 1) ? 'yes' : 'no';
$generaladdtionalcontent = "enable=". $enable ."\n";
$generaladdtionalcontent .= "batch=". $batch ."\n";
$generaladdtionalcontent .= "size=". $amp_conf['CDR_BATCH_SIZE'] ."\n";
$generaladdtionalcontent .= "time=". $amp_conf['CDR_BATCH_TIME']."\n";
$generaladdtionalcontent .= "scheduleronly=". $schedule."\n";
$generaladdtionalcontent .= "safeshutdown=". $bshutdown."\n";
\FreePBX::WriteConfig()->writeConfig('cdr_general_additional.conf', $generaladdtionalcontent, true);
}
if(isset($amp_conf['CDR_BATCH_ENABLE'])) {
$enable = ($amp_conf['CDR_BATCH_ENABLE'] == 1) ? 'yes' : 'no';
$managerGeneralAddtionalContent = "enabled=". $enable ."\n";
$cdr_mappings = \FreePBX::Cdr()->getAll('cdr_mappings');
$managerMappingsAdditionalConf='';
foreach ($cdr_mappings as $map) {
$managerMappingsAdditionalConf.= key($map) . '=>' . $map[key($map)]." \n";
}
\FreePBX::WriteConfig()->writeConfig('cdr_manager_general_additional.conf', $managerGeneralAddtionalContent, true);
\FreePBX::WriteConfig()->writeConfig('cdr_manager_mapping_additional.conf', $managerMappingsAdditionalConf, true);
}
}
// NOTE: This function should probably be in a FreePBX library
// php function empty() treats 0 as empty, that is why I need the function below
// to be able to search for any number starting with 0
function is_blank($value) {
return empty($value) && !is_numeric($value);
}
/* Asterisk RegExp parser */
function cdr_asteriskregexp2sqllike( $source_data, $user_num ) {
$number = $user_num;
if ( strlen((string) $number) < 1 ) {
$number = $_POST[$source_data];
}
if ( str_starts_with((string) $number, '__') ) {
$number = substr((string) $number,1);
} elseif ( str_starts_with((string) $number, '_') ) {
$number_chars = preg_split('//', substr((string) $number,1), -1, PREG_SPLIT_NO_EMPTY);
$number = '';
foreach ($number_chars as $chr) {
if ( $chr == 'X' ) {
$number .= '[0-9]';
} elseif ( $chr == 'Z' ) {
$number .= '[1-9]';
} elseif ( $chr == 'N' ) {
$number .= '[2-9]';
} elseif ( $chr == '.' ) {
$number .= '.+';
} elseif ( $chr == '!' ) {
$_POST[ $source_data .'_neg' ] = 'true';
} else {
$number .= $chr;
}
}
$_POST[ $source_data .'_mod' ] = 'asterisk-regexp';
}
return $number;
}
function cdr_get_cel($uid, $cel_table = 'asteriskcdrdb.cel') {
global $dbcdr;
// common query components
//
$sql_base = "SELECT *, unix_timestamp(eventtime) as `event_timestamp` FROM $cel_table WHERE ";
$sql_order = " ORDER BY eventtime, id";
// get first set of CEL records
//
$sql_start = $sql_base . "uniqueid = '$uid' OR linkedid = '$uid'" . $sql_order;
$pass = $dbcdr->getAll($sql_start,DB_FETCHMODE_ASSOC);
if(DB::IsError($pass)) {
die_freepbx($pass->getDebugInfo() . "SQL - <br /> $sql_start" );
}
$last_criteria = [];
$next =[];
$done = false;
// continue querying all records based on the uniqueid and linkedid fields associated
// with the first set we queried until we have found all of them. This usually results
// in one or two more queries prior to the last one being identical indicating we have
// found all the records
//
while (!$done) {
unset($next);
$next = [];
foreach ($pass as $set) {
$next[] = $set['uniqueid'];
$next[] = $set['linkedid'];
}
$next = array_unique($next);
sort($next);
// if our criteria is now the same then we have found everything
//
if ($next == $last_criteria) {
$done = true;
continue;
}
unset($pass);
$set = "('" . implode("','", $next) . "')";
$sql_next = $sql_base . "uniqueid IN $set OR linkedid IN $set" . $sql_order;
$last_criteria = $next;
$next = [];
$pass = $dbcdr->getAll($sql_next,DB_FETCHMODE_ASSOC);
if(DB::IsError($pass)) {
die_freepbx($pass->getDebugInfo() . "SQL - <br /> $sql_next" );
}
}
return $pass;
}
function cdr_export_csv($csvdata) {
// Searching for more than 10,000 records take more than 30 seconds.
// php default timeout is 30 seconds, hard code it to 3000 seconds for now (which is WAY overkill).
// TODO: make this value a setting in Advanced Settings
set_time_limit(3000);
$fname = "cdr__" . (string) time() . $_SERVER["SERVER_NAME"] . ".csv";
$csv_header ="calldate,clid,src,dst,dcontext,channel,dstchannel,lastapp,lastdata,duration,billsec,disposition,amaflags,accountcode,uniqueid,userfield,did,cnum,cnam,outbound_cnum,outbound_cnam,dst_cnam,recordingfile,linkedid,peeraccount,sequence";
$mimetype = "application/octet-stream";
// Start sending headers
header("Pragma: public"); // required
header("Expires: 0");
header("Cache-Control: must-revalidate, post-check=0, pre-check=0");
header("Cache-Control: private",false); // required for certain browsers
header("Content-Transfer-Encoding: binary");
header("Content-Type: " . $mimetype);
header("Content-Disposition: attachment; filename=\"" . $fname . "\";" );
// Send data
$out = fopen('php://output', 'w');
fputcsv($out, explode(",",$csv_header));
foreach ($csvdata as $csv) {
$csv_line = [];
foreach(explode(",",$csv_header) as $k => $item) {
// if(preg_match("/\p{Hebrew}/u", utf8_decode((string) $csv[$item]))) {
if (preg_match("/\p{Hebrew}/u", mb_convert_encoding((string) $csv[$item], 'UTF-8', mb_detect_encoding((string) $csv[$item])))) {
/**
* Hebrew is read from right to the left.
* Need to change the order Num Name instead to Name and Num
* Otherwise, even if the csv format is correct, the result is messed up on Excel.
*/
preg_match('/<\d+>/', (string) $csv[$item], $_num);
preg_match('/".+"/', (string) $csv[$item], $_name);
// $name = str_replace('"','',utf8_decode($_name[0]));
$name = str_replace('"','',mb_convert_encoding($_name[0], 'UTF-8', mb_detect_encoding($_name[0])));
$csv[$item] = $_num[0].' "'.$name.'"';
}
$csv_line[$k] = $csv[$item];
}
fputcsv($out, $csv_line);
}
fclose($out);
die();
}
function writeCustomFiles($custConf){
$generalCustContent = '';
$nongeneralCustContent = '';
$nongenLines = false;
if($custConf) {
foreach ($custConf as $line) {
if(str_contains((string) $line, '[general]') || empty($line)){
continue;
}
if(str_contains((string) $line, '[csv]')){
$nongenLines = true;
}
if($nongenLines) {
$nongeneralCustContent .= str_replace(PHP_EOL, '', (string) $line)."\n";
} else {
$generalCustContent .= str_replace(PHP_EOL, '', (string) $line)."\n";
}
}
}
\FreePBX::WriteConfig()->writeConfig('cdr_general_custom.conf', $generalCustContent, false);
\FreePBX::WriteConfig()->writeConfig('cdr_non_general_custom.conf', $nongeneralCustContent, false);
}