Compare commits
18 Commits
d5fa504526
...
dev
| Author | SHA1 | Date | |
|---|---|---|---|
| 190c848aa9 | |||
| fd9239c2bb | |||
| 3d880b6699 | |||
| 69c5a88c50 | |||
| 598af2c954 | |||
| bd9f4c074d | |||
| 8ebdcb7d1d | |||
| 9e7ec4fc52 | |||
| d98e31faca | |||
| f96abcbb4e | |||
| c89e1da002 | |||
| d8ff7cc641 | |||
| ef7dc8ff09 | |||
| 6948df2357 | |||
| c9f46413fb | |||
| 9b89c3b649 | |||
| 8421f5d374 | |||
| 792998334f |
46
config.json
Normal file
46
config.json
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
{
|
||||||
|
"target_extensions": ["334", "338", "227", "228"],
|
||||||
|
"list_filter_type": "blacklist",
|
||||||
|
"extension_filter_list": [
|
||||||
|
"209",
|
||||||
|
"214",
|
||||||
|
"220",
|
||||||
|
"234",
|
||||||
|
"254",
|
||||||
|
"333",
|
||||||
|
"344",
|
||||||
|
"348",
|
||||||
|
"355",
|
||||||
|
"365",
|
||||||
|
"370",
|
||||||
|
"371",
|
||||||
|
"372",
|
||||||
|
"373",
|
||||||
|
"374",
|
||||||
|
"375",
|
||||||
|
"377",
|
||||||
|
"378",
|
||||||
|
"379",
|
||||||
|
"383",
|
||||||
|
"384",
|
||||||
|
"390",
|
||||||
|
"391",
|
||||||
|
"393",
|
||||||
|
"397",
|
||||||
|
"398",
|
||||||
|
"529"
|
||||||
|
],
|
||||||
|
"blacklisted_terms": ["vesibule", "inpatient"],
|
||||||
|
|
||||||
|
"prepend_extensions": [
|
||||||
|
{"name": "Night Hours", "extension": "*271"},
|
||||||
|
{"name": "Overhead Page", "extension": "900"},
|
||||||
|
{"name": "All page", "extension": "300"},
|
||||||
|
{"name": "Park 71", "extension": "71"},
|
||||||
|
{"name": "Park 72", "extension": "72"},
|
||||||
|
{"name": "OR", "extension": "356"},
|
||||||
|
{"name": "Lab", "extension": "340"},
|
||||||
|
{"name": "Business", "extension": "249"},
|
||||||
|
{"name": "OP Nurse", "extension": "336"}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -1 +0,0 @@
|
|||||||
extensions = 334,338
|
|
||||||
@@ -1,32 +1,49 @@
|
|||||||
#!/usr/bin/php -q
|
#!/usr/bin/php -q
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
|
$PROVISION_DIR = '/tftpboot';
|
||||||
|
|
||||||
$bootstrap_settings['freepbx_auth'] = false;
|
$bootstrap_settings['freepbx_auth'] = false;
|
||||||
require_once('/etc/freepbx.conf');
|
require_once('/etc/freepbx.conf');
|
||||||
|
|
||||||
function main(): void {
|
$DB = FreePBX::Database();
|
||||||
|
|
||||||
|
$CONFIG_PATH = "./config.json";
|
||||||
|
$CONFIG = read_config_json_file($CONFIG_PATH);
|
||||||
|
|
||||||
|
$TARGET_EXTENSIONS = read_config_json("target_extensions");
|
||||||
|
$LIST_FILTER_TYPE = read_config_json_string("list_filter_type");
|
||||||
|
$EXTENSION_FILTER_LIST = read_config_json("extension_filter_list");
|
||||||
|
$BLACKLISTED_TERMS = read_config_json("blacklisted_terms");
|
||||||
|
$PREPEND_EXTENSIONS = read_config_json_object_list("prepend_extensions");
|
||||||
|
|
||||||
|
function main(): void {
|
||||||
|
global $TARGET_EXTENSIONS;
|
||||||
|
global $PROVISION_DIR;
|
||||||
|
|
||||||
$provision_dir = '/tftpboot';
|
|
||||||
$argv = $_SERVER['argv'] ?? [];
|
$argv = $_SERVER['argv'] ?? [];
|
||||||
$do_notify = in_array('--notify', $argv, true);
|
$do_notify = in_array('--notify', $argv, true);
|
||||||
|
|
||||||
|
$pbdb = pull_db();
|
||||||
|
|
||||||
|
blacklist_terms($pbdb);
|
||||||
|
filter_extensions($pbdb);
|
||||||
|
prepend_contact_list($pbdb);
|
||||||
|
|
||||||
$pbdb = trim_db(pull_db());
|
|
||||||
|
|
||||||
$ext_list = read_extension_ini();
|
|
||||||
$mac_list = pull_mac_list();
|
$mac_list = pull_mac_list();
|
||||||
|
|
||||||
foreach ($ext_list as $ext) {
|
foreach ($TARGET_EXTENSIONS as $ext) {
|
||||||
$mac = strtolower($mac_list[$ext]) ?? null;
|
$mac = strtolower($mac_list[$ext]) ?? null;
|
||||||
|
|
||||||
if (!$mac) { echo "Mac for $ext not found\n"; continue; }
|
if (!$mac) { echo "Mac for $ext not found\n"; continue; }
|
||||||
|
|
||||||
$file = $provision_dir . '/' . $mac . '-features.cfg';
|
$file = $PROVISION_DIR . '/' . $mac . '-features.cfg';
|
||||||
|
|
||||||
$xml = pull_xml_file($file);
|
$xml = pull_xml_file($file);
|
||||||
$attendant = remove_attendants($xml);
|
$attendants = remove_attendants($xml);
|
||||||
write_attendants($attendant, $pbdb);
|
write_attendants($attendants, $pbdb);
|
||||||
write_to_file($provision_dir, $file, $xml);
|
write_to_file($file, $xml);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($do_notify) {
|
if ($do_notify) {
|
||||||
@@ -34,13 +51,12 @@ function main(): void {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function pull_db(): array {
|
function pull_db(): array {
|
||||||
$db = FreePBX::Database();
|
global $DB;
|
||||||
$pbdb = [];
|
$pbdb = [];
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$stmt = $db->prepare("
|
$stmt = $DB->prepare("
|
||||||
SELECT name, extension
|
SELECT name, extension
|
||||||
FROM users
|
FROM users
|
||||||
WHERE extension REGEXP '^[0-9]+$'
|
WHERE extension REGEXP '^[0-9]+$'
|
||||||
@@ -59,49 +75,88 @@ function pull_db(): array {
|
|||||||
usort($pbdb, function ($a, $b) {
|
usort($pbdb, function ($a, $b) {
|
||||||
return strcasecmp($a['name'], $b['name']);
|
return strcasecmp($a['name'], $b['name']);
|
||||||
});
|
});
|
||||||
|
|
||||||
return $pbdb;
|
return $pbdb;
|
||||||
}
|
}
|
||||||
|
|
||||||
function trim_db($pbdb): array {
|
function filter_extensions(array &$pbdb) {
|
||||||
$pbdb_prepend = [
|
global $LIST_FILTER_TYPE;
|
||||||
[ 'name' => 'Night Hours', 'extension' => '*271', ],
|
|
||||||
[ 'name' => 'Overhead Page', 'extension' => '900', ],
|
|
||||||
[ 'name' => 'All page', 'extension' => '300'],
|
|
||||||
[ 'name' => 'Park 71', 'extension' => '71'],
|
|
||||||
[ 'name' => 'Park 72', 'extension' => '72'],
|
|
||||||
];
|
|
||||||
|
|
||||||
array_unshift($pbdb, ...$pbdb_prepend);
|
if ($LIST_FILTER_TYPE == "whitelist") {
|
||||||
|
whitelist_extension_filter($pbdb);
|
||||||
|
} elseif ($LIST_FILTER_TYPE == "blacklist") {
|
||||||
|
blacklist_extension_filter($pbdb);
|
||||||
|
} else {
|
||||||
|
fwrite(STDERR, "Filter type invalid: $LIST_FILTER_TYPE");
|
||||||
|
fwrite(STDERR, "Use either whitelist or blacklist");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$remove_names = [
|
function blacklist_terms(array &$pbdb) {
|
||||||
'inpatient',
|
global $BLACKLISTED_TERMS;
|
||||||
'vestibule'
|
|
||||||
];
|
$pbdb = array_values(array_filter($pbdb, function ($item) use ($BLACKLISTED_TERMS) {
|
||||||
|
|
||||||
$filtered = array_filter($pbdb, function ($item) use ($remove_names) {
|
|
||||||
if (!is_array($item) || !isset($item['name'])) {
|
if (!is_array($item) || !isset($item['name'])) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
$name = ltrim($item['name']);
|
$name = ltrim($item['name']);
|
||||||
|
|
||||||
foreach ($remove_names as $remove_name) {
|
foreach ($BLACKLISTED_TERMS as $term) {
|
||||||
if (stripos($name, $remove_name) === 0) {
|
if (stripos($name, $term) === 0) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
});
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
return $filtered;
|
function whitelist_extension_filter(array &$pbdb) {
|
||||||
|
global $EXTENSION_FILTER_LIST;
|
||||||
|
|
||||||
|
$allowed = array_fill_keys(
|
||||||
|
array_map('trim', array_map('strval', $EXTENSION_FILTER_LIST)),
|
||||||
|
true
|
||||||
|
);
|
||||||
|
|
||||||
|
$pbdb = array_values(array_filter($pbdb, function ($item) use ($allowed) {
|
||||||
|
if (!is_array($item) || !isset($item['extension'])) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$ext = trim((string)$item['extension']);
|
||||||
|
return isset($allowed[$ext]);
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
function blacklist_extension_filter(array &$pbdb) {
|
||||||
|
global $EXTENSION_FILTER_LIST;
|
||||||
|
|
||||||
|
$blocked = array_fill_keys(
|
||||||
|
array_map('trim', array_map('strval', $EXTENSION_FILTER_LIST)),
|
||||||
|
true
|
||||||
|
);
|
||||||
|
|
||||||
|
$pbdb = array_values(array_filter($pbdb, function ($item) use ($blocked) {
|
||||||
|
if (!is_array($item) || !isset($item['extension'])) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
$ext = trim((string)$item['extension']);
|
||||||
|
return !isset($blocked[$ext]);
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
function prepend_contact_list(array &$pbdb) {
|
||||||
|
global $PREPEND_EXTENSIONS;
|
||||||
|
array_unshift($pbdb, ...$PREPEND_EXTENSIONS);
|
||||||
}
|
}
|
||||||
|
|
||||||
function pull_mac_list(): array {
|
function pull_mac_list(): array {
|
||||||
$db = FreePBX::Database();
|
global $DB;
|
||||||
$mac_db = [];
|
$mac_db = [];
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$stmt = $db->prepare("
|
$stmt = $DB->prepare("
|
||||||
SELECT mac, SUBSTRING_INDEX(ext, '-', 1) AS ext
|
SELECT mac, SUBSTRING_INDEX(ext, '-', 1) AS ext
|
||||||
FROM endpoint_extensions
|
FROM endpoint_extensions
|
||||||
ORDER BY CAST(ext AS UNSIGNED)
|
ORDER BY CAST(ext AS UNSIGNED)
|
||||||
@@ -114,14 +169,64 @@ function pull_mac_list(): array {
|
|||||||
return array_column($mac_db, 'mac', 'ext');
|
return array_column($mac_db, 'mac', 'ext');
|
||||||
}
|
}
|
||||||
|
|
||||||
function read_extension_ini(): array {
|
|
||||||
$extensions = parse_ini_file('extensions.ini');
|
|
||||||
$ext_list = array_map('intval', explode(',', $extensions['extensions']));
|
|
||||||
|
|
||||||
return $ext_list;
|
function read_config_json_file(string $path): array {
|
||||||
|
$json = file_get_contents($path);
|
||||||
|
$data = json_decode($json, true);
|
||||||
|
|
||||||
|
if ($data === null && json_last_error() !== JSON_ERROR_NONE) {
|
||||||
|
throw new RuntimeException("JSON decode error: " . json_last_error_msg());
|
||||||
|
}
|
||||||
|
return $data;
|
||||||
}
|
}
|
||||||
|
|
||||||
function pull_xml_file($file): DOMDocument {
|
function read_config_json(string $key): array {
|
||||||
|
global $CONFIG;
|
||||||
|
$value = $CONFIG[$key] ?? [];
|
||||||
|
|
||||||
|
if (is_string($value)) {
|
||||||
|
$items = preg_split('/\s*,\s*/', $value, -1, PREG_SPLIT_NO_EMPTY);
|
||||||
|
} elseif (is_array($value)) {
|
||||||
|
$items = $value;
|
||||||
|
} else {
|
||||||
|
$items = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$items = array_values(array_filter($items, fn($v) => is_scalar($v)));
|
||||||
|
|
||||||
|
$items = array_map(
|
||||||
|
fn($v) => strtolower(trim((string)$v)),
|
||||||
|
$items
|
||||||
|
);
|
||||||
|
|
||||||
|
return array_values(array_filter($items, fn($v) => $v !== ''));
|
||||||
|
}
|
||||||
|
|
||||||
|
function read_config_json_string(string $key): string {
|
||||||
|
global $CONFIG;
|
||||||
|
$value = $CONFIG[$key] ?? '';
|
||||||
|
return strtolower(trim((string)$value));
|
||||||
|
}
|
||||||
|
|
||||||
|
function read_config_json_object_list(string $key): array {
|
||||||
|
global $CONFIG;
|
||||||
|
$value = $CONFIG[$key] ?? [];
|
||||||
|
|
||||||
|
if (!is_array($value)) return [];
|
||||||
|
|
||||||
|
$out = [];
|
||||||
|
foreach ($value as $row) {
|
||||||
|
if (is_array($row) && isset($row['name'], $row['extension'])) {
|
||||||
|
$out[] = [
|
||||||
|
'name' => (string)$row['name'],
|
||||||
|
'extension' => (string)$row['extension'],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
function pull_xml_file(string $file): DOMDocument {
|
||||||
if (!file_exists($file)) {
|
if (!file_exists($file)) {
|
||||||
$file = '/tftpboot/000000000000-features.cfg';
|
$file = '/tftpboot/000000000000-features.cfg';
|
||||||
}
|
}
|
||||||
@@ -156,7 +261,7 @@ function pull_xml_file($file): DOMDocument {
|
|||||||
return $xml;
|
return $xml;
|
||||||
}
|
}
|
||||||
|
|
||||||
function remove_attendants($xml): DOMElement {
|
function remove_attendants(DOMDocument $xml): DOMElement {
|
||||||
$xpath = new DOMXPath($xml);
|
$xpath = new DOMXPath($xml);
|
||||||
$attendantNodes = $xpath->query('/polycomConfig/attendant');
|
$attendantNodes = $xpath->query('/polycomConfig/attendant');
|
||||||
if ($attendantNodes->length === 0) {
|
if ($attendantNodes->length === 0) {
|
||||||
@@ -177,24 +282,25 @@ function remove_attendants($xml): DOMElement {
|
|||||||
return $attendant;
|
return $attendant;
|
||||||
}
|
}
|
||||||
|
|
||||||
function write_attendants($attendant, $pbdb): void {
|
function write_attendants(DOMElement $attendant, array $pbdb): void {
|
||||||
$index = 1;
|
$index = 1;
|
||||||
foreach ($pbdb as $r) {
|
foreach ($pbdb as $r) {
|
||||||
$label = trim((string)($r['name'] ?? ''));
|
$label = trim((string)($r['name'] ?? ''));
|
||||||
$address = trim((string)($r['extension'] ?? ''));
|
$address = trim((string)($r['extension'] ?? ''));
|
||||||
$type = trim((string)("normal"));
|
$type = trim((string)("normal"));
|
||||||
|
|
||||||
$attendant->setAttribute("attendant.resourceList.{$index}.address", $address);
|
$attendant->setAttribute("attendant.resourceList.{$index}.address", $address);
|
||||||
$attendant->setAttribute("attendant.resourceList.{$index}.label", $label);
|
$attendant->setAttribute("attendant.resourceList.{$index}.label", $label);
|
||||||
$attendant->setAttribute("attendant.resourceList.{$index}.type", $type);
|
$attendant->setAttribute("attendant.resourceList.{$index}.type", $type);
|
||||||
|
|
||||||
$index++;
|
$index++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function write_to_file($provision_dir, $file, $xml): void {
|
function write_to_file(string $file, DOMDocument $xml): void {
|
||||||
if (!is_dir($provision_dir)) {
|
global $PROVISION_DIR;
|
||||||
fwrite(STDERR, "Provisioning directory not found: $provision_dir\n");
|
if (!is_dir($PROVISION_DIR)) {
|
||||||
|
fwrite(STDERR, "Provisioning directory not found: $PROVISION_DIR\n");
|
||||||
exit(2);
|
exit(2);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -219,7 +325,7 @@ function write_to_file($provision_dir, $file, $xml): void {
|
|||||||
echo "Wrote $file \n";
|
echo "Wrote $file \n";
|
||||||
}
|
}
|
||||||
|
|
||||||
function notify($pbdb): void {
|
function notify(array $pbdb): void {
|
||||||
$notified = 0;
|
$notified = 0;
|
||||||
foreach ($pbdb as $r) {
|
foreach ($pbdb as $r) {
|
||||||
$ext = trim((string)$r['extension']);
|
$ext = trim((string)$r['extension']);
|
||||||
|
|||||||
59
readme.md
Normal file
59
readme.md
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
## Gen PC Sidecar Script
|
||||||
|
The files are stored typically inside of `/var/lib/asterisk/PC-Contact-Sync` and the scripts should be run every 6 hours by a cron job.
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Quickstart
|
||||||
|
#### Config Fields
|
||||||
|
`config.json` has several fields. The config file needs to be next to the script in order to be read.
|
||||||
|
|
||||||
|
<div style="border-left: 6px solid #FFA500; padding: 10px;">
|
||||||
|
<strong>⚠️ Warning:</strong> Make sure the config follows proper json syntax.
|
||||||
|
</div>
|
||||||
|
<br>
|
||||||
|
|
||||||
|
```
|
||||||
|
target_extensions
|
||||||
|
list_filter_type
|
||||||
|
extension_filter_list
|
||||||
|
blacklisted_terms
|
||||||
|
prepend_extensions
|
||||||
|
```
|
||||||
|
|
||||||
|
1. Add the phones that need a config pushed out for to the `target_extensions` list.
|
||||||
|
2. Add the type of filter you want to use to filter out extensions from the contact list to `list_filter_type`.
|
||||||
|
3. Add Extensions you want either black/white listed to the `extension_filter_list`.
|
||||||
|
4. Add words that are in contacts you want removed to `blacklisted_terms`.
|
||||||
|
5. Add extra contacts you want added to the phones in `prepend_extensions`.
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Process Walkthrough
|
||||||
|
|
||||||
|
#### 1. Pull Contact List
|
||||||
|
Initially the script formats together a contact list to be pushed to the phones. This php script will pull every contact from asterisk database which contains a **name** and **extension**.
|
||||||
|
|
||||||
|
#### 2. Contact List Formatting
|
||||||
|
The script will now filter the contact list before writing to the phones. It has 3 processes to format it:
|
||||||
|
|
||||||
|
1. Filter extensions by black/white list
|
||||||
|
2. Remove blacklisted terms
|
||||||
|
3. Prepend a set of configured contacts listed
|
||||||
|
|
||||||
|
The filter list will apply either a black/white list to the `extension_filter_list`. If it is a white list `extension_filter_list` is kept and anything not in it is removed. If it is a blacklist anything inside of `extension_filter_list` is removed.
|
||||||
|
|
||||||
|
The script will remove any contact name that has words contained inside of `blacklisted_terms`.
|
||||||
|
|
||||||
|
Finally contacts specified in `prepend_extensions` are added to the beginning of the contact list.
|
||||||
|
|
||||||
|
#### 3. Write to phones
|
||||||
|
Polycom phones read their extension list to display on the phone from a file in `/tftboot` that is served by PBXact. The phone looks for settings including its contact list in `/tftboot/phone_mac_address-features.cfg` replacing mac_address with the actual mac of the phone without any : in the number. The default if it cant find its mac is to use the file with 0s.
|
||||||
|
|
||||||
|
##### Example:
|
||||||
|
```
|
||||||
|
/tftboot/000000000-features.cfg
|
||||||
|
```
|
||||||
|
|
||||||
|
The script will go down the list and read the .cfg for each phone in `target_extensions` and write in the contact list that was created earlier. It wites to a config file that matches the extensions mac address.
|
||||||
|
|
||||||
|
Phones might need to be told to reconfigure to pull the updated cfg.
|
||||||
Reference in New Issue
Block a user