plugins->registerEvent('server_insert', 'xmpp_plugin', 'insert'); $app->plugins->registerEvent('server_update', 'xmpp_plugin', 'update'); $app->plugins->registerEvent('xmpp_domain_insert', 'xmpp_plugin', 'ssl'); $app->plugins->registerEvent('xmpp_domain_update', 'xmpp_plugin', 'ssl'); $app->plugins->registerEvent('xmpp_domain_delete', 'xmpp_plugin', 'ssl'); $app->plugins->registerEvent('xmpp_domain_insert', 'xmpp_plugin', 'domainInsert'); $app->plugins->registerEvent('xmpp_domain_update', 'xmpp_plugin', 'domainUpdate'); $app->plugins->registerEvent('xmpp_domain_delete', 'xmpp_plugin', 'domainDelete'); $app->plugins->registerEvent('xmpp_user_insert', 'xmpp_plugin', 'userInsert'); $app->plugins->registerEvent('xmpp_user_update', 'xmpp_plugin', 'userUpdate'); $app->plugins->registerEvent('xmpp_user_delete', 'xmpp_plugin', 'userDelete'); } function insert($event_name, $data) { global $app, $conf; $this->update($event_name, $data); } // The purpose of this plugin is to rewrite the main.cf file function update($event_name, $data) { global $app, $conf; // get the config $app->uses("getconf,system,tpl"); $old_ini_data = $app->ini_parser->parse_ini_string($data['old']['config']); $xmpp_config = $app->getconf->get_server_config($conf['server_id'], 'xmpp'); // Global server config $tpl = new tpl(); $tpl->newTemplate('xmpp_metronome_conf_global.master'); $tpl->setVar('ipv6', $xmpp_config['xmpp_use_ipv6']=='y'?'true':'false'); $tpl->setVar('bosh_timeout', intval($xmpp_config['xmpp_bosh_max_inactivity'])); $tpl->setVar('port_http', intval($xmpp_config['xmpp_port_http'])); $tpl->setVar('port_https', intval($xmpp_config['xmpp_port_https'])); $tpl->setVar('port_pastebin', intval($xmpp_config['xmpp_port_pastebin'])); $tpl->setVar('port_bosh', intval($xmpp_config['xmpp_port_bosh'])); // Global server admins (for all hosted domains) $admins = ''; foreach(explode(',', $xmpp_config['xmpp_server_admins']) AS $a) $admins.= "\t\"".trim($a)."\",\n"; $tpl->setVar('server_admins', $admins); unset($admins); // enabled modules, so own modules or simmilar prosody-modules can easily be added $modules = ''; foreach(explode(',', $xmpp_config['xmpp_modules_enabled']) AS $m) $modules.= "\t\"".trim($m)."\",\n"; $tpl->setVar('modules_enabled', $modules); unset($modules); $app->system->file_put_contents($this->xmpp_config_dir.'/global.cfg.lua', $tpl->grab()); unset($tpl); $app->services->restartServiceDelayed('metronome', 'restart'); return; } function domainInsert($event_name, $data) { global $app, $conf; $this->domainUpdate($event_name, $data); } function domainUpdate($event_name, $data){ global $app, $conf; // get the config $app->uses("getconf,system,tpl"); // Collections $status_hosts = array($data['new']['domain']); $status_comps = array(); // Create main host file $tpl = new tpl(); $tpl->newTemplate('xmpp_metronome_conf_host.master'); $tpl->setVar('domain', $data['new']['domain']); $tpl->setVar('active', $data['new']['active'] == 'y' ? 'true' : 'false'); $tpl->setVar('public_registration', $data['new']['public_registration'] == 'y' ? 'true' : 'false'); // Domain admins $admins = array(); foreach(explode(',',$data['new']['domain_admins']) AS $adm){ $admins[] = trim($adm); } $tpl->setVar('domain_admins', "\t\t\"".implode("\",\n\t\t\"",$admins)."\"\n"); // Enable / Disable features if($data['new']['use_pubsub']=='y'){ $tpl->setVar('use_pubsub', 'true'); $status_comps[] = 'pubsub.'.$data['new']['domain']; }else{ $tpl->setVar('use_pubsub', 'false'); } if($data['new']['use_proxy']=='y'){ $tpl->setVar('use_proxy', 'true'); $status_comps[] = 'proxy.'.$data['new']['domain']; }else{ $tpl->setVar('use_proxy', 'false'); } if($data['new']['use_anon_host']=='y'){ $tpl->setVar('use_anon_host', 'true'); $status_hosts[] = 'anon.'.$data['new']['domain']; }else{ $tpl->setVar('use_anon_host', 'false'); } if($data['new']['use_vjud']=='y'){ $tpl->setVar('use_vjud', 'true'); $tpl->setVar('vjud_opt_mode', 'opt-'.$data['new']['vjud_opt_mode']); $status_comps[] = 'vjud.'.$data['new']['domain']; }else{ $tpl->setVar('use_vjud', 'false'); } $tpl->setVar('use_muc', $data['new']['use_muc_host']=='y'?'true':'false'); if($data['new']['use_muc_host'] == 'y'){ $status_comps[] = 'muc.'.$data['new']['domain']; $tpl->setVar('muc_restrict_room_creation', $data['new']['muc_restrict_room_creation']); $tpl->setVar('muc_name', strlen($data['new']['muc_name']) ? $data['new']['muc_name'] : $data['new']['domain'].' Chatrooms'); // Admins for MUC channels $admins = array(); foreach(explode(',',$data['new']['muc_admins']) AS $adm){ $admins[] = trim($adm); } $tpl->setVar('muc_admins', "\t\t\"".implode("\",\n\t\t\"",$admins)."\"\n"); $tpl->setVar('use_pastebin', $data['new']['use_pastebin']=='y'?'true':'false'); $tpl->setVar('pastebin_expire', intval($data['new']['pastebin_expire_after'])); $tpl->setVar('pastebin_trigger', $data['new']['pastebin_trigger']); $tpl->setVar('use_archive', $data['new']['use_http_archive']=='y'?'true':'false'); $tpl->setVar('archive_join', $data['new']['http_archive_show_join']=='y'?'true':'false'); $tpl->setVar('archive_status', $data['new']['http_archive_show_status']=='y'?'true':'false'); } // Check for SSL if(strlen($data['new']['ssl_cert']) && strlen($data['new']['ssl_key']) && !$this->ssl_certificate_deleted || $this->ssl_certificate_changed) $tpl->setVar('ssl_cert', true); $app->system->file_put_contents($this->xmpp_config_dir.'/hosts/'.$data['new']['domain'].'.cfg.lua', $tpl->grab()); unset($tpl); // Create status host file if($data['new']['use_status_host']=='y'){ $tpl = new tpl; $tpl->newTemplate('xmpp_metronome_conf_status.master'); $tpl->setVar('domain', $data['new']['domain']); $tpl->setVar('status_hosts', "\t\t\"".implode("\",\n\t\t\"",$status_hosts)."\"\n"); $tpl->setVar('status_comps', "\t\t\"".implode("\",\n\t\t\"",$status_comps)."\"\n"); $app->system->file_put_contents($this->xmpp_config_dir.'/status/'.$data['new']['domain'].'.cfg.lua', $tpl->grab()); unset($tpl); } $app->services->restartServiceDelayed('metronome', 'reload'); } function domainDelete($event_name, $data){ global $app, $conf; // get the config $app->uses("system"); $domain = $data['old']['domain']; $folder = str_replace('-', '%2d', str_replace('.', '%2e', $str = urlencode($domain))); // Remove config files $app->system->unlink("/etc/metronome/hosts/$domain.cfg.lua"); $app->system->unlink("/etc/metronome/status/$domain.cfg.lua"); $app->system->unlink("/etc/metronome/certs/$domain.cert"); $app->system->unlink("/etc/metronome/certs/$domain.key"); $app->system->unlink("/etc/metronome/certs/$domain.csr"); // Remove all stored data var_dump('rm -rf /var/lib/metronome/'.$folder); exec('rm -rf /var/lib/metronome/'.$folder); exec('rm -rf /var/lib/metronome/*%2e'.$folder); $app->services->restartServiceDelayed('metronome', 'reload'); } function userInsert($event_name, $data){ //$data['new']['auth_method'] // Check domain for auth settings // Don't allow manual user creation for mailaccount controlled domains // maybe metronomectl adduser for new local users } function userUpdate($event_name, $data){ // Check domain for auth settings // Don't allow manual user update for mailaccount controlled domains // maybe metronomectl passwd for existing local users } function userDelete($event_name, $data){ // Check domain for auth settings // Don't allow manual user deletion for mailaccount controlled domains // Remove account from metronome exec('metronomectl deluser '.$data['old']['jid']); } // Handle the creation of SSL certificates function ssl($event_name, $data) { global $app, $conf; $app->uses('system,tpl'); // load the server configuration options $app->uses('getconf'); $web_config = $app->getconf->get_server_config($conf['server_id'], 'web'); $ssl_dir = '/etc/metronome/certs'; $domain = $data['new']['domain']; $cnf_file = $ssl_dir.'/'.$domain.'.cnf'; $key_file = $ssl_dir.'/'.$domain.'.key'; $csr_file = $ssl_dir.'/'.$domain.'.csr'; $crt_file = $ssl_dir.'/'.$domain.'.cert'; //* Create a SSL Certificate, but only if this is not a mirror server. if($data['new']['ssl_action'] == 'create' && $conf['mirror_server_id'] == 0) { $this->ssl_certificate_changed = true; //* Rename files if they exist if(file_exists($cnf_file)) $app->system->rename($cnf_file, $cnf_file.'.bak'); if(file_exists($key_file)){ $app->system->rename($key_file, $key_file.'.bak'); $app->system->chmod($key_file.'.bak', 0400); $app->system->chown($key_file.'.bak', 'metronome'); } if(file_exists($csr_file)) $app->system->rename($csr_file, $csr_file.'.bak'); if(file_exists($crt_file)) $app->system->rename($crt_file, $crt_file.'.bak'); // Write new CNF file $tpl = new tpl(); $tpl->newTemplate('xmpp_metronome_conf_ssl.master'); $tpl->setVar('domain', $domain); $tpl->setVar('ssl_country', $data['new']['ssl_country']); $tpl->setVar('ssl_locality', $data['new']['ssl_locality']); $tpl->setVar('ssl_organisation', $data['new']['ssl_organisation']); $tpl->setVar('ssl_organisation_unit', $data['new']['ssl_organisation_unit']); $tpl->setVar('ssl_email', $data['new']['ssl_email']); $app->system->file_put_contents($cnf_file, $tpl->grab()); // Generate new key, csr and cert exec("(cd /etc/metronome/certs && make $domain.key)"); exec("(cd /etc/metronome/certs && make $domain.csr)"); exec("(cd /etc/metronome/certs && make $domain.cert)"); $ssl_key = $app->system->file_get_contents($key_file); $app->system->chmod($key_file, 0400); $app->system->chown($key_file, 'metronome'); $ssl_request = $app->system->file_get_contents($csr_file); $ssl_cert = $app->system->file_get_contents($crt_file); /* Update the DB of the (local) Server */ $app->db->query("UPDATE xmpp_domain SET ssl_request = ?, ssl_cert = ?, ssl_key = ? WHERE domain = ?", $ssl_request, $ssl_cert, $ssl_key, $data['new']['domain']); $app->db->query("UPDATE xmpp_domain SET ssl_action = '' WHERE domain = ?", $data['new']['domain']); /* Update also the master-DB of the Server-Farm */ $app->dbmaster->query("UPDATE xmpp_domain SET ssl_request = ?, ssl_cert = ?, ssl_key = ? WHERE domain = ?", $ssl_request, $ssl_cert, $ssl_key, $data['new']['domain']); $app->dbmaster->query("UPDATE xmpp_domain SET ssl_action = '' WHERE domain = ?", $data['new']['domain']); $app->log('Creating XMPP SSL Cert for: '.$domain, LOGLEVEL_DEBUG); } //* Save a SSL certificate to disk if($data["new"]["ssl_action"] == 'save') { $this->ssl_certificate_changed = true; //* Rename files if they exist if(file_exists($cnf_file)) $app->system->rename($cnf_file, $cnf_file.'.bak'); if(file_exists($key_file)){ $app->system->rename($key_file, $key_file.'.bak'); $app->system->chmod($key_file.'.bak', 0400); $app->system->chown($key_file.'.bak', 'metronome'); } if(file_exists($csr_file)) $app->system->rename($csr_file, $csr_file.'.bak'); if(file_exists($crt_file)) $app->system->rename($crt_file, $crt_file.'.bak'); //* Write new ssl files if(trim($data["new"]["ssl_request"]) != '') $app->system->file_put_contents($csr_file, $data["new"]["ssl_request"]); if(trim($data["new"]["ssl_cert"]) != '') $app->system->file_put_contents($crt_file, $data["new"]["ssl_cert"]); //* Write the key file, if field is empty then import the key into the db if(trim($data["new"]["ssl_key"]) != '') { $app->system->file_put_contents($key_file, $data["new"]["ssl_key"]); $app->system->chmod($key_file, 0400); $app->system->chown($key_file, 'metronome'); } else { $ssl_key = $app->system->file_get_contents($key_file); /* Update the DB of the (local) Server */ $app->db->query("UPDATE xmpp_domain SET ssl_key = ? WHERE domain = ?", $ssl_key, $data['new']['domain']); /* Update also the master-DB of the Server-Farm */ $app->dbmaster->query("UPDATE xmpp_domain SET ssl_key = '$ssl_key' WHERE domain = ?", $data['new']['domain']); } /* Update the DB of the (local) Server */ $app->db->query("UPDATE xmpp_domain SET ssl_action = '' WHERE domain = ?", $data['new']['domain']); /* Update also the master-DB of the Server-Farm */ $app->dbmaster->query("UPDATE xmpp_domain SET ssl_action = '' WHERE domain = ?", $data['new']['domain']); $app->log('Saving XMPP SSL Cert for: '.$domain, LOGLEVEL_DEBUG); } //* Delete a SSL certificate if($data['new']['ssl_action'] == 'del') { $this->ssl_certificate_deleted = true; $app->system->unlink($csr_file); $app->system->unlink($crt_file); $app->system->unlink($key_file); $app->system->unlink($cnf_file); $app->system->unlink($csr_file.'.bak'); $app->system->unlink($crt_file.'.bak'); $app->system->unlink($key_file.'.bak'); $app->system->unlink($cnf_file.'.bak'); /* Update the DB of the (local) Server */ $app->db->query("UPDATE xmpp_domain SET ssl_request = '', ssl_cert = '', ssl_key = '' WHERE domain = ?", $data['new']['domain']); $app->db->query("UPDATE xmpp_domain SET ssl_action = '' WHERE domain = ?", $data['new']['domain']); /* Update also the master-DB of the Server-Farm */ $app->dbmaster->query("UPDATE xmpp_domain SET ssl_request = '', ssl_cert = '', ssl_key = '' WHERE domain = ?", $data['new']['domain']); $app->dbmaster->query("UPDATE xmpp_domain SET ssl_action = '' WHERE domain = ?", $data['new']['domain']); $app->log('Deleting SSL Cert for: '.$domain, LOGLEVEL_DEBUG); } } } // end class ?>