["type"=>"INT","nullable"=>false], "messageId"=>["type"=>"VARCHAR(255)", "nullable"=>true], "direction"=>["type"=>"CHAR(2)", "nullable"=>false], "date"=>["type"=>"DATETIME", "nullable"=>true], "createDate"=>["type"=>"DATETIME", "nullable"=>true], "inReptyTo"=>["type"=>"VARCHAR(255)", "nullable"=>true], "subject"=>["type"=>"VARCHAR(255)", "nullable"=>true], "bodyHTML"=>["type"=>"TEXT", "nullable"=>true], "bodyPlain"=>["type"=>"TEXT", "nullable"=>true], "txProto"=>["type"=>"VARCHAR(255)", "nullable"=>true], "conn"=>["type"=>"SKIP"], "refNum"=>["type"=>"SKIP"], ]; protected function addAddressToIdx(array &$idx, $val) { $addr=null; if ($val instanceof mailAddress) { $addr = $val; } elseif ($val instanceof user) { if (!common::validateEMail($val->eMail)) { throw new \Exception("Invalid user eMail. Failed.", 1509);}; if ($val->eMailConfirmed) { $addr = new mailAddress($val->fullName, $val->eMail); } else { trigger_error("Address for user ".$val->id."not added - not eMailConfirmed"); } } else { $addr = new mailAddress($val); } if (!$addr) { return false; } if (mailBlocklist::getByAddress($addr->address)) { trigger_error("Address ".$addr->address." blockListed."); return false; } foreach ($idx as $rcpt) { if ($rcpt->address == $addr->address) { return; } } array_push($idx,$addr); } protected function getAttachments() { if (empty($this->__attachments) && !empty($this->__attachmentsID)) { foreach ($this->__attachmentsID as $attId) { $att = new mailAttachment($attId); $att->message=$this; array_push($this->__attachments,$att); } } return $this->__attachments; } public function addRecipient($rcptTo) { $this->addAddressToIdx($this->rcptTo, $rcptTo); } public function addSender($mailFrom) { $this->addAddressToIdx($this->mailFrom, $mailFrom); } public function addCC($mailFrom) { $this->addAddressToIdx($this->cc, $mailFrom); } public function addBCC($mailFrom) { $this->addAddressToIdx($this->bcc, $mailFrom); } public function addAttachment(mailAttachment $att) { array_push($this->attachments, $att); } public function __get($key) { switch ($key) { case "bodyPlain": if (empty($this->__bodyPlain)) { $this->__bodyPlain = (new Html2Text($this->bodyHTML))->getText(); }; return $this->__bodyPlain; case "bodyHTML": if (empty($this->__bodyHTML)) { return base64_decode($this->__bodyPlain); } else { return base64_decode($this->__bodyHTML); } case "attachments": return $this->getAttachments(); break; case "isHTML": return (!empty($this->bodyHTML)); break; case "account": if (empty($this->__account) and !empty($this->__accountId)) { $this->__account = new mailAccount($this->__accountId); } return $this->__account; case "subject": return base64_decode($this->__subject); default: return parent::__get($key); } } public function __set($key, $val) { switch ($key) { case "account": if (empty($val)) { $this->__account=null; $this->__accountId=null; } elseif ($val instanceof mailAccount) { $this->__account = $val; $this->__accountId=$val->id; } elseif (gettype($val) == 'string' || gettype($val) == 'integer') { $this->__account = new mailAccount($val); $this->__accountId = $this->__account->id; } break; case "references": $ref = explode(" ", $val); $this->refIds=[]; foreach ($ref as $ref_item) { $ref_item=preg_replace("!^\<|\>$!", '', $ref_item); array_push($this->refIds, $ref_item); } break; case "inReplyTo": $this->inReptyTo=preg_replace("!^\<|\>$!", '', $val); break; case "messageId": $this->messageId=preg_replace("!^\<|\>$!", '', $val); break; case "subject": $this->__subject=base64_encode($val); break; case "direction": $dir = strtoupper($val); if ($dir=='RX' || $dir=='TX') { $this->direction=$dir; } else { throw new \Exception("Invalid direction"); } break; case "mailFrom": $this->mailFrom=[]; if (!empty($val)) { $this->addSender($val); } break; case "bodyPlain": $this->__bodyPlain=base64_encode($val); break; case "bodyHTML": $this->__bodyHTML=base64_encode($val); break; default: parent::__set($key, $val); } } public function createMessageID($uid=null) { if (empty($uid)) ($uid="XXXX-0000-00"); $this->messageId=common::getGUIDc()."-CHIMERA-".$uid."-FOX-".time(); } protected function serializeAddresses(&$arr) { $rv=[]; foreach ($arr as $addr) { array_push($rv, $addr->full); } return $rv; } protected function deSerealizeAddresses($arr) { $rv=[]; foreach ($arr as $addr) { array_push($rv, new mailAddress($addr)); } return $rv; } protected function validateSave() { if (empty($this->accountId)) { throw new \Exception("AccountID can't be empty"); } if (empty($this->direction) || ($this->direction !== 'RX' && $this->direction !== 'TX')) { throw new \Exception("direction must be in [RX:TX]");} return true; } protected function create() { $this->attachmentsID=[]; foreach ($this->attachments as $att) { $att->writeAttachment(); array_push($this->attachmentsID, $att->id); } parent::create(); if (isset($this->conn)) { imap_setflag_full($this->conn, $this->refNum, "\Seen");} } public function preSave() { if ($this->direction=='TX') { if (empty($this->__accountId)) { $this->__account=mailAccount::getDefaultAccount($this->sql); if (empty($this->__account)) { throw new \Exception("Default mail account absent!"); } $this->__accountId=$this->__account->id; } if (empty($this->mailFrom)) { $this->addSender($this->__account->address);} if (empty($this->messageId)) { $this->createMessageID();} if (empty($this->date)) { $this->date=time::current();}; } } public function save() { $this->preSave(); parent::save(); } public function send($save=false) { if (empty($this->direction)) { $this->direction="TX";} if ($this->direction != 'TX') { throw new \Exception("Unable to send message with ".$this->direction." type!"); } $this->preSave(); $this->__get("account"); if ($this->account->txProto!='smtp' || $this->account->txServer == null) { throw new \Exception("This account can't send messages"); return false; } $mail = new PHPMailer(true); $mail->CharSet = PHPMailer::CHARSET_UTF8; $mail->ContentType=PHPMailer::CHARSET_UTF8; try { $mail->isSMTP(); // Send using SMTP $mail->Host = $this->account->txServer; // Set the SMTP server to send through $mail->SMTPAuth = !empty($this->account->login); // Enable SMTP authentication $mail->Username = $this->account->login; // SMTP username $mail->Password = $this->account->password; // SMTP password $mail->SMTPOptions = ['ssl'=> ['allow_self_signed' => true]]; if ($this->account->txSSL) { $mail->SMTPSecure = PHPMailer::ENCRYPTION_SMTPS; // Enable TLS encryption; `PHPMailer::ENCRYPTION_SMTPS` encouraged } else { $mail->SMTPSecure = false; if ($this->account->txPort==587) { $mail->SMTPAutoTLS=true; } else { $mail->SMTPAutoTLS=false; } } $mail->Port = $this->account->txPort; // TCP port to connect to, use 465 for `PHPMailer::ENCRYPTION_SMTPS` above if (empty($this->rcptTo)) { trigger_error("Empty recipients list, send aborted"); return false; } //Recipients $mail->setFrom($this->mailFrom[0]->address, $this->mailFrom[0]->name); foreach ($this->rcptTo as $addr) { $mail->addAddress($addr->address, '=?UTF-8?B?'.base64_encode($addr->name).'?='); } foreach ($this->cc as $addr) { $mail->addCC($addr->address, $addr->name); } foreach ($this->bcc as $addr) { $mail->addBCC($addr->address, $addr->name); } $this->getAttachments(); foreach ($this->__get("attachments") as $att) { $mail->addAttachment($att->getPath(), $att->filename); // Add attachments } // Content $mail->isHTML($this->isHTML); // Set email format to HTML $mail->Subject = '=?UTF-8?B?'.base64_encode($this->subject).'?='; $mail->Subject = $this->subject; $mail->Body = $this->bodyHTML; $mail->AltBody = $this->bodyPlain; $mail->addCustomHeader("Auto-Submitted", "auto-replied"); $mail->send(); } catch (mException $e) { throw new \Exception("Message could not be sent. Mailer Error: {$mail->ErrorInfo}"); } } } ?>