apiKey = $apiKey; // create the logger $this->logger = new PostmanLogger( get_class( $this ) ); } /** * (non-PHPdoc) * * @see PostmanSmtpEngine::send() */ public function send( PostmanMessage $message ) { $options = PostmanOptions::getInstance(); // add the From Header $sender = $message->getFromAddress(); $senderEmail = ! empty( $sender->getEmail() ) ? $sender->getEmail() : $options->getMessageSenderEmail(); $senderName = ! empty( $sender->getName() ) ? $sender->getName() : $options->getMessageSenderName(); $from = new SendGrid\Email( $senderName, $senderEmail ); // now log it $sender->log( $this->logger, 'From' ); // add the to recipients $counter = 0; $emails = array(); /** * @todo: Find a better approch. */ $duplicates = array(); foreach ( ( array ) $message->getToRecipients() as $recipient ) { $recipient->log( $this->logger, 'To' ); $email = $recipient->getEmail(); if ( $counter == 0 ) { $this->logger->debug( 'Adding to=' . $recipient->getEmail() ); $to = new SendGrid\Email($recipient->getName(), $recipient->getEmail() ); $duplicates[] = $email; } else { if ( ! in_array( $email, $duplicates ) ) { $duplicates[] = $email; $this->logger->debug( 'Adding personalization to=' . $recipient->getEmail() ); $emails[] = new SendGrid\Email($recipient->getName(), $recipient->getEmail() ); } } $counter++; } // add the subject if ( null !== $message->getSubject() ) { $subject = $message->getSubject(); } // add the message content $textPart = $message->getBodyTextPart(); if ( ! empty( $textPart ) ) { $this->logger->debug( 'Adding body as text' ); $content = new SendGrid\Content("text/plain", $textPart); } $htmlPart = $message->getBodyHtmlPart(); if ( ! empty( $htmlPart ) ) { $this->logger->debug( 'Adding body as html' ); $content = new SendGrid\Content("text/html", $htmlPart); } $mail = new SendGrid\Mail($from, $subject, $to, $content); foreach ( $emails as $email) { $mail->personalization[0]->addTo($email); } // add the reply-to $replyTo = $message->getReplyTo(); // $replyTo is null or a PostmanEmailAddress object if ( isset( $replyTo ) ) { $reply_to = new SendGrid\ReplyTo( $replyTo->getEmail(), $replyTo->getName() ); $mail->setReplyTo($reply_to); } // add the Postman signature - append it to whatever the user may have set if ( ! $options->isStealthModeEnabled() ) { $pluginData = apply_filters( 'postman_get_plugin_metadata', null ); $mail->personalization[0]->addHeader( 'X-Mailer', sprintf( 'Postman SMTP %s for WordPress (%s)', $pluginData ['version'], 'https://wordpress.org/plugins/post-smtp/' ) ); } // add the headers - see http://framework.zend.com/manual/1.12/en/zend.mail.additional-headers.html foreach ( ( array ) $message->getHeaders() as $header ) { $this->logger->debug( sprintf( 'Adding user header %s=%s', $header ['name'], $header ['content'] ) ); $mail->personalization[0]->addHeader( $header ['name'], $header ['content'] ); } // if the caller set a Content-Type header, use it $contentType = $message->getContentType(); if ( ! empty( $contentType ) ) { $this->logger->debug( 'Some header keys are reserved. You may not include any of the following reserved headers: x-sg-id, x-sg-eid, received, dkim-signature, Content-Type, Content-Transfer-Encoding, To, From, Subject, Reply-To, CC, BCC.' ); } // add the cc recipients foreach ( ( array ) $message->getCcRecipients() as $recipient ) { if ( ! in_array( $recipient->getEmail(), $duplicates ) ) { $recipient->log( $this->logger, 'Cc' ); $email = new SendGrid\Email( $recipient->getName(), $recipient->getEmail() ); $mail->personalization[0]->addCc( $email ); } } // add the bcc recipients foreach ( ( array ) $message->getBccRecipients() as $recipient ) { if ( ! in_array( $recipient->getEmail(), $duplicates ) ) { $recipient->log($this->logger, 'Bcc'); $email = new SendGrid\Email($recipient->getName(), $recipient->getEmail()); $mail->personalization[0]->addBcc($email); } } // add the messageId $messageId = $message->getMessageId(); if ( ! empty( $messageId ) ) { $mail->personalization[0]->addHeader( 'message-id', $messageId ); } // add attachments $this->logger->debug( 'Adding attachments' ); $attachments = $this->addAttachmentsToMail( $message ); foreach ( $attachments as $index => $attachment ) { $attach = new SendGrid\Attachment(); $attach->setContent($attachment['content']); $attach->setType($attachment['type']); $attach->setFilename($attachment['file_name']); $attach->setDisposition("attachment"); $attach->setContentId($attachment['id']); $mail->addAttachment($attach); } try { if ( $this->logger->isDebug() ) { $this->logger->debug( 'Creating SendGrid service with apiKey=' . $this->apiKey ); } $sendgrid = new SendGrid( $this->apiKey ); // send the message if ( $this->logger->isDebug() ) { $this->logger->debug( 'Sending mail' ); } $response = $sendgrid->client->mail()->send()->post($mail); if ( $this->logger->isInfo() ) { $this->logger->info( sprintf( 'Message %d accepted for delivery', PostmanState::getInstance()->getSuccessfulDeliveries() + 1 ) ); } $response_body = json_decode( $response->body() ); $response_code = $response->statusCode(); $email_sent = ( $response_code >= 200 and $response_code < 300 ); if ( isset( $response_body->errors[0]->message ) || ! $email_sent ) { $e = ! $email_sent ? $this->errorCodesMap($response_code) : $response_body->errors[0]->message; $this->transcript = $e; $this->transcript .= PostmanModuleTransport::RAW_MESSAGE_FOLLOWS; $this->transcript .= print_r( $mail, true ); $this->logger->debug( 'Transcript=' . $this->transcript ); throw new Exception( $e ); } $this->transcript = print_r( $response->body(), true ); $this->transcript .= PostmanModuleTransport::RAW_MESSAGE_FOLLOWS; $this->transcript .= print_r( $mail, true ); } catch ( SendGrid\Exception $e ) { $this->transcript = $e->getMessage(); $this->transcript .= PostmanModuleTransport::RAW_MESSAGE_FOLLOWS; $this->transcript .= print_r( $mail, true ); $this->logger->debug( 'Transcript=' . $this->transcript ); throw $e; } } private function errorCodesMap($error_code) { switch ($error_code) { case 413: $message = sprintf( __( 'ERROR: The JSON payload you have included in your request is too large. Status code is %1$s', Postman::TEXT_DOMAIN ), $error_code ); break; case 429: $message = sprintf( __( 'ERROR: The number of requests you have made exceeds SendGrid rate limitations. Status code is %1$s', Postman::TEXT_DOMAIN ), $error_code ); break; case 500: $message = sprintf( __( 'ERROR: An error occurred on a SendGrid server. Status code is %1$s', Postman::TEXT_DOMAIN ), $error_code ); break; case 513: $message = sprintf( __( 'ERROR: The SendGrid v3 Web API is not available. Status code is %1$s', Postman::TEXT_DOMAIN ), $error_code ); break; case 502: $message = sprintf( __( 'ERROR: No recipient supplied. Status code is %1$s', Postman::TEXT_DOMAIN ), $error_code ); break; default: $message = sprintf( __( 'ERROR: Status code is %1$s', Postman::TEXT_DOMAIN ), $error_code ); } return $message; } /** * Add attachments to the message * * @param Postman_Zend_Mail $mail */ private function addAttachmentsToMail( PostmanMessage $message ) { $attachments = $message->getAttachments(); if ( ! is_array( $attachments ) ) { // WordPress may a single filename or a newline-delimited string list of multiple filenames $attArray = explode( PHP_EOL, $attachments ); } else { $attArray = $attachments; } // otherwise WordPress sends an array $attachments = array(); foreach ( $attArray as $file ) { if ( ! empty( $file ) ) { $this->logger->debug( 'Adding attachment: ' . $file ); $file_name = basename( $file ); $file_parts = explode( '.', $file_name ); $file_type = wp_check_filetype( $file ); $attachments[] = array( 'content' => base64_encode( file_get_contents( $file ) ), 'type' => $file_type['type'], 'file_name' => $file_name, 'disposition' => 'attachment', 'id' => $file_parts[0], ); } } return $attachments; } // return the SMTP session transcript public function getTranscript() { return $this->transcript; } } }