HEX
Server: LiteSpeed
System: Linux server902.web-hosting.com 4.18.0-553.54.1.lve.el8.x86_64 #1 SMP Wed Jun 4 13:01:13 UTC 2025 x86_64
User: deshuvsd (2181)
PHP: 8.1.33
Disabled: NONE
Upload Files
File: /home/deshuvsd/www/wp-content/plugins/suremails/inc/emails/providers/emailit/emailit-handler.php
<?php
/**
 * EmailitHandler.php
 *
 * Handles sending emails using Emailit service.
 *
 * @package SureMails\Inc\Emails\Providers\Emailit
 */

namespace SureMails\Inc\Emails\Providers\EMAILIT;

use SureMails\Inc\Emails\Handler\ConnectionHandler;
use SureMails\Inc\Emails\ProviderHelper;

if ( ! defined( 'ABSPATH' ) ) {
	exit; // Exit if accessed directly.
}

/**
 * Class EmailitHandler
 *
 * Implements the ConnectionHandler to handle Emailit email sending and authentication.
 */
class EmailitHandler implements ConnectionHandler {

	/**
	 * Emailit connection data.
	 *
	 * @var array
	 */
	protected $connection_data;

	/**
	 * Emailit API endpoint for sending emails.
	 *
	 * @var string
	 */
	private $api_url = 'https://api.emailit.com/v1/emails';

	/**
	 * Constructor.
	 *
	 * Initializes connection data.
	 *
	 * @param array $connection_data The connection details.
	 */
	public function __construct( array $connection_data ) {
		$this->connection_data = $connection_data;
	}

	/**
	 * Get headers for the Emailit connection.
	 *
	 * @param string $api_key The API key for the Emailit connection.
	 * @return array The headers for the Emailit connection.
	 */
	public function get_headers( $api_key ) {
		return [
			'Authorization' => 'Bearer ' . sanitize_text_field( $api_key ),
			'Content-Type'  => 'application/json',
		];
	}
	/**
	 * Authenticate the Emailit connection by verifying the API key.
	 *
	 * @return array The result of the authentication attempt.
	 */
	public function authenticate() {
		return [
			'success' => true,
		];
	}

	/**
	 * Send an email via Emailit, including attachments if provided.
	 *
	 * @param array $atts        The email attributes, such as 'to', 'from', 'subject', 'message', 'headers', 'attachments', etc.
	 * @param int   $log_id      The log ID for the email.
	 * @param array $connection  The connection details.
	 * @param array $processed_data The processed email data.
	 * @return array             The result of the email send operation.
	 * @throws \Exception If the email payload cannot be encoded to JSON.
	 */
	public function send( array $atts, $log_id, array $connection, $processed_data ) {
		$result = [
			'success' => false,
			'message' => '',
			'send'    => false,
		];

		// Prepare basic email payload.
		$email_payload = [
			'from'    => $this->format_email_address(
				sanitize_email( $connection['from_email'] ),
				! empty( $connection['from_name'] ) ? sanitize_text_field( $connection['from_name'] ) : __( 'WordPress', 'suremails' )
			),
			'subject' => sanitize_text_field( $atts['subject'] ?? '' ),
		];

		// Prepare recipients.
		$to_recipients = $processed_data['to'] ?? [];
		if ( ! empty( $to_recipients ) ) {
			$primary_recipient   = reset( $to_recipients );
			$email_payload['to'] = sanitize_email( $primary_recipient['email'] );
		}

		// Handle reply-to.
		$reply_to = $processed_data['headers']['reply_to'] ?? [];
		if ( ! empty( $reply_to ) ) {
			$reply_to_email            = reset( $reply_to );
			$email_payload['reply_to'] = sanitize_email( $reply_to_email['email'] );
		}

		// Add content based on content type.
		$content_type = $processed_data['headers']['content_type'] ?? '';
		$is_html      = ProviderHelper::is_html( $content_type );

		if ( $is_html ) {
			$email_payload['html'] = $atts['message'];
		}

		// Always include text version.
		$email_payload['text'] = $is_html ? wp_strip_all_tags( $atts['message'] ) : $atts['message'];

		// Handle headers for CC and BCC.
		$headers = [];

		// Handle CC.
		if ( ! empty( $processed_data['headers']['cc'] ) ) {
			$cc_addresses = [];
			foreach ( $processed_data['headers']['cc'] as $cc ) {
				$cc_addresses[] = $this->format_email_address(
					sanitize_email( $cc['email'] ),
					isset( $cc['name'] ) ? sanitize_text_field( $cc['name'] ) : ''
				);
			}
			if ( ! empty( $cc_addresses ) ) {
				$headers['cc'] = implode( ', ', $cc_addresses );
			}
		}

		// Handle BCC.
		if ( ! empty( $processed_data['headers']['bcc'] ) ) {
			$bcc_addresses = [];
			foreach ( $processed_data['headers']['bcc'] as $bcc ) {
				$bcc_addresses[] = $this->format_email_address(
					sanitize_email( $bcc['email'] ),
					isset( $bcc['name'] ) ? sanitize_text_field( $bcc['name'] ) : ''
				);
			}
			if ( ! empty( $bcc_addresses ) ) {
				$headers['bcc'] = implode( ', ', $bcc_addresses );
			}
		}

		// Add headers to payload if not empty.
		if ( ! empty( $headers ) ) {
			$email_payload['headers'] = $headers;
		}

		// Handle attachments.
		if ( ! empty( $processed_data['attachments'] ) ) {
			$attachments = [];
			foreach ( $processed_data['attachments'] as $attachment ) {
				$attachment_values = ProviderHelper::get_attachment( $attachment );

				if ( ! $attachment_values ) {
					continue;
				}

				$attachments[] = [
					'filename'     => $attachment_values['name'],
					'content'      => $attachment_values['blob'],
					'content_type' => $attachment_values['type'],
				];
			}

			if ( ! empty( $attachments ) ) {
				$email_payload['attachments'] = $attachments;
			}
		}

		// Send email via Emailit API.
		try {
			$json_payload = wp_json_encode( $email_payload );
			if ( $json_payload === false ) {
				throw new \Exception( __( 'Failed to encode email payload to JSON.', 'suremails' ) );
			}

			$response = wp_safe_remote_post(
				$this->api_url,
				[
					'headers' => $this->get_headers( $connection['api_key'] ),
					'body'    => $json_payload,
					'timeout' => 30,
				]
			);

			if ( is_wp_error( $response ) ) {
				$result['message']    = __( 'Emailit send failed: ', 'suremails' ) . $response->get_error_message();
				$result['error_code'] = $response->get_error_code();
				return $result;
			}

			$response_code = wp_remote_retrieve_response_code( $response );
			$response_body = wp_remote_retrieve_body( $response );

			if ( $response_code === 200 || $response_code === 202 ) {
				$result['success'] = true;
				$result['message'] = __( 'Email sent successfully via Emailit.', 'suremails' );
				$result['send']    = true;

				// Try to get message ID from response.
				$decoded_response = json_decode( $response_body, true );
				if ( is_array( $decoded_response ) && isset( $decoded_response['id'] ) ) {
					$result['email_id'] = $decoded_response['id'];
				}
			} else {
				$decoded_body  = json_decode( $response_body, true );
				$error_message = $this->extract_error_message( $decoded_body, (int) $response_code );

				// translators: %s is the error message from Emailit API.
				$result['message']    = sprintf( __( 'Email sending failed via Emailit: %s', 'suremails' ), $error_message );
				$result['error_code'] = $response_code;
				$result['retries']    = 1;
			}
		} catch ( \Exception $e ) {
			$result['message']    = __( 'Emailit send failed: ', 'suremails' ) . $e->getMessage();
			$result['error_code'] = 500;
			$result['retries']    = 1;
		}

		return $result;
	}

	/**
	 * Return the option configuration for Emailit.
	 *
	 * @return array
	 */
	public static function get_options() {
		return [
			'title'             => __( 'Emailit Connection', 'suremails' ),
			'description'       => __( 'Enter the details below to connect with your Emailit account. Important: Your sending domain must be verified in Emailit before you can send emails.', 'suremails' ),
			'fields'            => self::get_specific_fields(),
			'display_name'      => __( 'Emailit', 'suremails' ),
			'icon'              => 'EmailitIcon',
			'provider_type'     => 'free',
			'field_sequence'    => [ 'connection_title', 'api_key', 'from_email', 'force_from_email', 'from_name', 'force_from_name', 'priority' ],
			'provider_sequence' => 45,
		];
	}

	/**
	 * Get the specific schema fields for Emailit.
	 *
	 * @return array
	 */
	public static function get_specific_fields() {
		return [
			'api_key' => [
				'required'    => true,
				'datatype'    => 'string',
				'help_text'   => sprintf(
					// translators: %1$s: API key link, %2$s: domain verification link.
					__( 'Get your API key from your Emailit dashboard. %1$s. Important: Before sending emails, you must %2$s in your Emailit account.', 'suremails' ),
					'<a href="https://app.emailit.com/settings/api" target="_blank">' . __( 'Get API Key', 'suremails' ) . '</a>',
					'<a href="https://app.emailit.com/domains" target="_blank">' . __( 'verify your sending domain', 'suremails' ) . '</a>'
				),
				'label'       => __( 'API Key', 'suremails' ),
				'input_type'  => 'password',
				'placeholder' => __( 'Enter your Emailit API Key', 'suremails' ),
				'encrypt'     => true,
			],
		];
	}

	/**
	 * Format email address with name.
	 *
	 * @param string $email The email address.
	 * @param string $name  The name (optional).
	 * @return string Formatted email address.
	 */
	private function format_email_address( $email, $name = '' ) {
		if ( ! empty( $name ) && $name !== $email ) {
			return sprintf( '%s <%s>', $name, $email );
		}
		return $email;
	}

	/**
	 * Extract error message from API response.
	 *
	 * @param array|null $decoded_body The decoded response body.
	 * @param int        $response_code The HTTP response code.
	 * @return string The error message.
	 */
	private function extract_error_message( $decoded_body, $response_code ) {
		if ( is_array( $decoded_body ) ) {
			if ( isset( $decoded_body['message'] ) ) {
				return $decoded_body['message'];
			}
			if ( isset( $decoded_body['error'] ) ) {
				return is_string( $decoded_body['error'] ) ? $decoded_body['error'] :
					( $decoded_body['error']['message'] ?? __( 'Unknown error', 'suremails' ) );
			}
			if ( isset( $decoded_body['errors'] ) && is_array( $decoded_body['errors'] ) && ! empty( $decoded_body['errors'] ) ) {
				$first_error = reset( $decoded_body['errors'] );
				return is_string( $first_error ) ? $first_error :
					( $first_error['message'] ?? __( 'Unknown error', 'suremails' ) );
			}
		}

		// Default error messages based on HTTP status codes.
		switch ( $response_code ) {
			case 400:
				return __( 'Bad request. Please check your email data.', 'suremails' );
			case 401:
				return __( 'Unauthorized. Please check your API key.', 'suremails' );
			case 403:
				return __( 'Forbidden. Access denied.', 'suremails' );
			case 404:
				return __( 'Not found. Please check the API endpoint.', 'suremails' );
			case 422:
				return __( 'Domain verification required. Your sending domain must be verified in Emailit before you can send emails. Please verify your domain in your Emailit dashboard at https://app.emailit.com/domains', 'suremails' );
			case 429:
				return __( 'Rate limit exceeded. Please try again later.', 'suremails' );
			case 500:
				return __( 'Internal server error. Please try again later.', 'suremails' );
			default:
				// translators: %d is the HTTP error code.
				return sprintf( __( 'HTTP error %d occurred.', 'suremails' ), $response_code );
		}
	}
}