setFormatter($consoleFormatter); $log->pushHandler($consoleHandler); // File handler - Rotates daily, keeps logs for 14 days $fileHandler = new RotatingFileHandler($logFilePath, 14, Logger::DEBUG); $fileFormatter = new LineFormatter( "[%datetime%] %channel%.%level_name%: %message% %context% %extra%\n", "Y-m-d H:i:s.u" ); $fileHandler->setFormatter($fileFormatter); $log->pushHandler($fileHandler); // Archive logs older than 14 days archiveOldLogs($logFilePath); // Pushover Handler (For CRITICAL, ALERT, EMERGENCY) if (!empty($config['pushover_key'])) { $pushoverHandler = new PushoverHandler($config['pushover_key'], Logger::ALERT); $log->pushHandler($pushoverHandler); } // Email Handler (For CRITICAL, ALERT, EMERGENCY) if (!empty($config['mailer_smtp_host'])) { // Create a PHPMailer instance $mail = new PHPMailer(true); try { $mail->isSMTP(); $mail->Host = $config['mailer_smtp_host']; $mail->SMTPAuth = true; $mail->Username = $config['mailer_smtp_username']; $mail->Password = $config['mailer_smtp_password']; $mail->SMTPSecure = PHPMailer::ENCRYPTION_STARTTLS; $mail->Port = $config['mailer_smtp_port']; $mail->setFrom($config['mailer_from'], 'Registry System'); $mail->addAddress($config['iana_email']); // Attach PHPMailer to Monolog $mailerHandler = new PHPMailerHandler($mail); $mailerHandler->setFormatter(new HtmlFormatter); $filteredMailHandler = new FilterHandler($mailerHandler, Logger::ALERT, Logger::EMERGENCY); $safeMailHandler = new WhatFailureGroupHandler([$filteredMailHandler]); $log->pushHandler($safeMailHandler); } catch (Exception $e) { error_log("Failed to initialize PHPMailer: " . $e->getMessage()); } } return $log; } function archiveOldLogs($logFilePath) { $logDir = dirname($logFilePath); $backupDir = '/opt/backup'; $lockFile = $backupDir . '/log_archive.lock'; // Prevent multiple processes from running archive at the same time if (file_exists($lockFile)) { return; // Another process is already archiving } touch($lockFile); // Create lock file if (!is_dir($backupDir)) { mkdir($backupDir, 0755, true); } $logFiles = glob($logDir . '/*.log'); // Get all log files $thresholdDate = strtotime('-14 days'); // Logs older than 14 days foreach ($logFiles as $file) { if (filemtime($file) < $thresholdDate) { $filename = basename($file); $monthYear = date('F-Y', filemtime($file)); $zipPath = $backupDir . "/logs-{$monthYear}.zip"; // Open or create ZIP archive $zip = new ZipArchive(); if ($zip->open($zipPath, ZipArchive::CREATE) === true) { if (!$zip->locateName($filename)) { // Prevent duplicate addition $zip->addFile($file, $filename); unlink($file); // Delete original log after archiving } $zip->close(); } } } unlink($lockFile); // Remove lock when done } function mapContactToVCard($contactDetails, $role, $c) { // Determine which type of disclosure to use $disclose_name = ($contactDetails['type'] == 'loc') ? $contactDetails['disclose_name_loc'] : $contactDetails['disclose_name_int']; $disclose_org = ($contactDetails['type'] == 'loc') ? $contactDetails['disclose_org_loc'] : $contactDetails['disclose_org_int']; $disclose_addr = ($contactDetails['type'] == 'loc') ? $contactDetails['disclose_addr_loc'] : $contactDetails['disclose_addr_int']; $disclose_voice = $contactDetails['disclose_voice']; $disclose_fax = $contactDetails['disclose_fax']; $disclose_email = $contactDetails['disclose_email']; return [ 'objectClassName' => 'entity', 'handle' => 'C' . $contactDetails['id'] . '-' . $c['roid'], 'roles' => [$role], 'remarks' => [ [ "description" => [ "This object's data has been partially omitted for privacy.", "Only the registrar managing the record can view personal contact data." ], "links" => [ [ "href" => "https://namingo.org", "rel" => "alternate", "type" => "text/html" ] ], "title" => "REDACTED FOR PRIVACY", "type" => "Details are withheld due to privacy restrictions." ], [ "description" => [ "To obtain contact information for the domain registrant, please refer to the Registrar of Record's RDDS service as indicated in this report." ], "title" => "EMAIL REDACTED FOR PRIVACY", "type" => "Details are withheld due to privacy restrictions." ], ], 'vcardArray' => [ "vcard", [ ['version', new stdClass(), 'text', '4.0'], ["fn", new stdClass(), 'text', $disclose_name ? $contactDetails['name'] : "REDACTED FOR PRIVACY"], ["org", new stdClass(), 'text', $disclose_org ? $contactDetails['org'] : "REDACTED FOR PRIVACY"], ["adr", ["CC" => strtoupper($contactDetails['cc'])], 'text', [ // specify "CC" parameter for country code $disclose_addr ? $contactDetails['street1'] : "REDACTED FOR PRIVACY", // Extended address $disclose_addr ? $contactDetails['street2'] : "REDACTED FOR PRIVACY", // Street address $disclose_addr ? $contactDetails['street3'] : "REDACTED FOR PRIVACY", // Additional street address $disclose_addr ? $contactDetails['city'] : "REDACTED FOR PRIVACY", // Locality $disclose_addr ? $contactDetails['sp'] : "REDACTED FOR PRIVACY", // Region $disclose_addr ? $contactDetails['pc'] : "REDACTED FOR PRIVACY", // Postal code "" // Add empty last element as required for ADR structure ]], ["tel", ["type" => "voice"], 'uri', $disclose_voice ? "tel:" . $contactDetails['voice'] : "REDACTED FOR PRIVACY"], ["tel", ["type" => "fax"], 'uri', $disclose_fax ? "tel:" . $contactDetails['fax'] : "REDACTED FOR PRIVACY"], ["email", new stdClass(), 'text', $disclose_email ? $contactDetails['email'] : "REDACTED FOR PRIVACY"], ] ], ]; } function isIpWhitelisted($ip, $pdo) { $stmt = $pdo->prepare("SELECT COUNT(*) FROM registrar_whitelist WHERE addr = ?"); $stmt->execute([$ip]); $count = $stmt->fetchColumn(); return $count > 0; } // Function to update the permitted IPs from the database function updatePermittedIPs($pool, $permittedIPsTable) { $pdo = $pool->get(); $query = "SELECT addr FROM registrar_whitelist"; $stmt = $pdo->query($query); $permittedIPs = $stmt->fetchAll(PDO::FETCH_COLUMN, 0); $pool->put($pdo); // Manually clear the table by removing each entry foreach ($permittedIPsTable as $key => $value) { $permittedIPsTable->del($key); } // Insert new values foreach ($permittedIPs as $ip) { $permittedIPsTable->set($ip, ['addr' => $ip]); } }