Server IP : 184.154.167.98 / Your IP : 3.133.122.95 Web Server : Apache System : Linux pink.dnsnetservice.com 4.18.0-553.22.1.lve.1.el8.x86_64 #1 SMP Tue Oct 8 15:52:54 UTC 2024 x86_64 User : puertode ( 1767) PHP Version : 8.2.26 Disable Function : NONE MySQL : OFF | cURL : ON | WGET : ON | Perl : ON | Python : ON | Sudo : ON | Pkexec : ON Directory : /home/puertode/public_html/mesa/include/ |
Upload File : |
<?php include_once INCLUDE_DIR.'class.api.php'; include_once INCLUDE_DIR.'class.ticket.php'; class TicketApiController extends ApiController { # Supported arguments -- anything else is an error. These items will be # inspected _after_ the fixup() method of the ApiXxxDataParser classes # so that all supported input formats should be supported function getRequestStructure($format, $data=null) { $supported = array( "alert", "autorespond", "source", "topicId", "attachments" => array("*" => array("name", "type", "data", "encoding", "size") ), "message", "ip", "priorityId", "system_emails" => array( "*" => "*" ), "thread_entry_recipients" => array ( "*" => array("to", "cc") ) ); # Fetch dynamic form field names for the given help topic and add # the names to the supported request structure if (isset($data['topicId']) && ($topic = Topic::lookup($data['topicId'])) && ($forms = $topic->getForms())) { foreach ($forms as $form) foreach ($form->getDynamicFields() as $field) $supported[] = $field->get('name'); } # Ticket form fields # TODO: Support userId for existing user if(($form = TicketForm::getInstance())) foreach ($form->getFields() as $field) $supported[] = $field->get('name'); # User form fields if(($form = UserForm::getInstance())) foreach ($form->getFields() as $field) $supported[] = $field->get('name'); if(!strcasecmp($format, 'email')) { $supported = array_merge($supported, array('header', 'mid', 'emailId', 'to-email-id', 'ticketId', 'reply-to', 'reply-to-name', 'in-reply-to', 'references', 'thread-type', 'system_emails', 'mailflags' => array('bounce', 'auto-reply', 'spam', 'viral'), 'recipients' => array('*' => array('name', 'email', 'source')) )); $supported['attachments']['*'][] = 'cid'; } return $supported; } /* Validate data - overwrites parent's validator for additional validations. */ function validate(&$data, $format, $strict=true) { global $ost; //Call parent to Validate the structure if(!parent::validate($data, $format, $strict) && $strict) $this->exerr(400, __('Unexpected or invalid data received')); // Use the settings on the thread entry on the ticket details // form to validate the attachments in the email $tform = TicketForm::objects()->one()->getForm(); $messageField = $tform->getField('message'); $fileField = $messageField->getWidget()->getAttachments(); // Nuke attachments IF API files are not allowed. if (!$messageField->isAttachmentsEnabled()) $data['attachments'] = array(); //Validate attachments: Do error checking... soft fail - set the error and pass on the request. if (isset($data['attachments']) && is_array($data['attachments'])) { foreach($data['attachments'] as &$file) { if ($file['encoding'] && !strcasecmp($file['encoding'], 'base64')) { if(!($file['data'] = base64_decode($file['data'], true))) $file['error'] = sprintf(__('%s: Poorly encoded base64 data'), Format::htmlchars($file['name'])); } // Validate and save immediately try { $F = $fileField->uploadAttachment($file); $file['id'] = $F->getId(); } catch (FileUploadError $ex) { $name = $file['name']; $file = array(); $file['error'] = Format::htmlchars($name) . ': ' . $ex->getMessage(); } } unset($file); } return true; } function create($format) { if (!($key=$this->requireApiKey()) || !$key->canCreateTickets()) return $this->exerr(401, __('API key not authorized')); $ticket = null; if (!strcasecmp($format, 'email')) { // Process remotely piped emails - could be a reply...etc. $ticket = $this->processEmailRequest(); } else { // Get and Parse request body data for the format $ticket = $this->createTicket($this->getRequest($format)); } if ($ticket) $this->response(201, $ticket->getNumber()); else $this->exerr(500, _S("unknown error")); } /* private helper functions */ function createTicket($data, $source = 'API') { # Pull off some meta-data $alert = (bool) (isset($data['alert']) ? $data['alert'] : true); $autorespond = (bool) (isset($data['autorespond']) ? $data['autorespond'] : true); // Assign default value to source if not defined, or defined as NULL $data['source'] ??= $source; // Create the ticket with the data (attempt to anyway) $errors = array(); if (($ticket = Ticket::create($data, $errors, $data['source'], $autorespond, $alert)) && !$errors) return $ticket; // Ticket create failed Bigly - got errors? $title = null; // Got errors? if (count($errors)) { // Ticket denied? Say so loudly so it can standout from generic // validation errors if (isset($errors['errno']) && $errors['errno'] == 403) { $title = _S('Ticket denied'); $error = sprintf("%s: %s\n\n%s", $title, $data['email'], $errors['err']); } else { // unpack the errors $error = Format::array_implode("\n", "\n", $errors); } } else { // unknown reason - default $error = _S('unknown error'); } $error = sprintf('%s :%s', _S('Unable to create new ticket'), $error); return $this->exerr($errors['errno'] ?: 500, $error, $title); } function processEmailRequest() { return $this->processEmail(); } function processEmail($data=false, array $defaults = []) { try { if (!$data) $data = $this->getEmailRequest(); elseif (!is_array($data)) $data = $this->parseEmail($data); } catch (Exception $ex) { throw new EmailParseError($ex->getMessage()); } $data = array_merge($defaults, $data); $seen = false; if (($entry = ThreadEntry::lookupByEmailHeaders($data, $seen)) && ($message = $entry->postEmail($data)) ) { if ($message instanceof ThreadEntry) { return $message->getThread()->getObject(); } else if ($seen) { // Email has been processed previously return $entry->getThread()->getObject(); } } // Allow continuation of thread without initial message or note elseif (($thread = Thread::lookupByEmailHeaders($data)) && ($message = $thread->postEmail($data)) ) { return $thread->getObject(); } // All emails which do not appear to be part of an existing thread // will always create new "Tickets". All other objects will need to // be created via the web interface or the API try { return $this->createTicket($data, 'Email'); } catch (TicketApiError $err) { // Check if the ticket was denied by a filter or banlist if ($err->isDenied() && $data['mid']) { // We need to log the Message-Id (mid) so we don't // process the same email again in subsequent fetches $entry = new ThreadEntry(); $entry->logEmailHeaders(0, $data['mid']); // throw TicketDenied exception so the caller can handle it // accordingly throw new TicketDenied($err->getMessage()); } else { // otherwise rethrow this bad baby as it is! throw $err; } } } } //Local email piping controller - no API key required! class PipeApiController extends TicketApiController { // Overwrite grandparent's (ApiController) response method. function response($code, $resp) { // It's important to use postfix exit codes for local piping instead // of HTTP's so the piping script can process them accordingly switch($code) { case 201: //Success $exitcode = 0; break; case 400: $exitcode = 66; break; case 401: /* permission denied */ case 403: $exitcode = 77; break; case 415: case 416: case 417: case 501: $exitcode = 65; break; case 503: $exitcode = 69; break; case 500: //Server error. default: //Temp (unknown) failure - retry $exitcode = 75; } //We're simply exiting - MTA will take care of the rest based on exit code! exit($exitcode); } static function process($sapi=null) { $pipe = new PipeApiController($sapi); if (($ticket=$pipe->processEmail())) return $pipe->response(201, is_object($ticket) ? $ticket->getNumber() : $ticket); return $pipe->exerr(416, __('Request failed - retry again!')); } static function local() { return self::process('cli'); } } class TicketApiError extends Exception { // Check if exception is because of denial public function isDenied() { return ($this->getCode() === 403); } } class TicketDenied extends Exception {} class EmailParseError extends Exception {} ?>