E-Mails verzögert senden

Uhrzeit gerade unpassend? E-Mail zu früh abgeschickt? E-Mails zu verzögern kann in vielen Situationen sinnvoll sein: Sei es, um sich eine Überprüfungszeit einzuräumen oder seinen eigenen Workload effektiv zu steuern – das Timing kann entscheidend sein. Es folgen drei Möglichkeiten, wie man das Versenden von E-Mails verzögern und seine eigene Kommunikation optimieren kann.


Outlook

Microsoft Outlook bietet eine integrierte Funktion, mit der Sie E-Mails verzögern können. Dazu nutzt man die Funktion "Optionen > Übermittlung verzögern":

Der große Nachteil dieser Lösung ist es, dass (auch in Exchange-Umgebungen) Outlook hier clientseitig beim Versandzeitpunkt laufen muss.

Google

Gmail bietet hier eine etwas bessere Lösung an. Dazu geht man vor dem Versand einer E-Mail auf "Senden planen":

Bereits versandte E-Mails kann man auch innerhalb von 30 Sekunden "rückgängig" machen. Dazu legt setzt man die jeweilige Einstellung unter "Alle Einstellungen aufrufen" zunächst fest:

Wenn man nun eine E-Mail absendet, kann man sie links unten innerhalb dieser Zeitspanne rückgängig machen:

Boomerang

Es gibt kostenpflichtige Dienstleister wie Boomerang for Outlook, die neben Erinnerungen oder automatisierte Follow-Ups auch einen verzögerten E-Mail-Versand anbieten:

Nachteilig ist hier, dass die E-Mails auf fremden Servern landen und die Nutzung kostenpflichtig ist.

PHP

Grund genug, etwas selbst zu bauen: Mit Hilfe eines kleinen PHP-Scripts kann man ebenso das Gewünschte erreichen. Zunächst legen wir in unserem Postfach eine Ordnerstruktur an:

Diese gibt den gewünschten Versandzeitpunkt wieder. Der Workflow besteht nun darin, die verfasste E-Mail statt abzusenden als Entwurf zu speichern und in den jeweiligen Ordner zu verschieben. Wir installieren dann alle benötigten PHP-Packages:

composer require php-imap/php-imap phpmailer/phpmailer vlucas/phpdotenv

Schließlich erstellen wir eine .env-Datei mit Zugangsdaten und Einstellungen:

HOST_SMTP="xxx"
PORT_SMTP=465
HOST_IMAP="xxx"
PORT_IMAP=993
USERNAME="foo@bar.com"
PASSWORD="xxx"
ENCRYPTION="ssl"
FROM_ADDRESS="foo@bar.com"
FROM_NAME="Foo Bar"
FOLDER_INBOX="INBOX/DELAY"
FOLDER_OUTBOX="Gesendete Elemente"
PHP_EXECUTABLE="/usr/bin/php"

Das folgende PHP-Script erledigt dann den Rest:

<?php
require_once __DIR__ . '/vendor/autoload.php';
 class MailDelay
{
    private \PhpImap\Mailbox $mailbox;
    private array $folders;
     public function init(): void
    {
        $this->loadEnvironmentVariables();
        $this->initMailbox();
        $this->initFolders();
        $this->processFolders();
    }
     private function loadEnvironmentVariables(): void
    {
        $dotenv = \Dotenv\Dotenv::createImmutable(__DIR__);
        $dotenv->load();
    }
     private function initMailbox(): void
    {
        $this->mailbox = new \PhpImap\Mailbox(
            '{' . $_SERVER['HOST_IMAP'] . ':' . $_SERVER['PORT_IMAP'] . '/imap/ssl}' . $_SERVER['FOLDER_INBOX'],
            $_SERVER['USERNAME'],
            $_SERVER['PASSWORD'],
            sys_get_temp_dir(),
            'UTF-8'
        );
    }
     private function initFolders(): void
    {
        $this->folders = [];
        $folders = $this->mailbox->getMailboxes('*');
        foreach ($folders as $folder) {
            if ($folder['shortpath'] === $_SERVER['FOLDER_INBOX']) {
                continue;
            }
            $this->folders[] = (object) $folder;
        }
    }
     private function processFolders(): void
    {
        foreach ($this->folders as $folder) {
            $this->mailbox->switchMailbox($folder->fullpath);
            $mailIds = $this->mailbox->searchMailbox('ALL');
            foreach ($mailIds as $mailId) {
                $preparedMail = $this->prepareMailData($mailId, $folder->shortpath);
                if (
                    $preparedMail->subject !== 'Dies ist Plain Text' &&
                    strtotime($preparedMail->time_to_send) > strtotime('now')
                ) {
                    continue;
                }
                try {
                    $this->sendMail($preparedMail);
                    echo 'Successfully sent mail #' . $preparedMail->id . '.' . PHP_EOL;
                } catch (\Exception $e) {
                    echo 'Error in sending mail #' . $preparedMail->id . ': ' . $e->getMessage() . PHP_EOL;
                }
            }
        }
        echo 'All mails have been processed.' . PHP_EOL;
    }
     private function prepareMailData(int $id, string $folder): object
    {
        $mail = $this->mailbox->getMail($id, false); // don't mark as unread
         return (object) [
            'id' => (string) $mail->id,
            'to' => $this->formatEmailAddresses($mail->to),
            'cc' => $this->formatEmailAddresses($mail->cc),
            'bcc' => $this->formatEmailAddresses($mail->bcc),
            'subject' => (string) $mail->subject,
            'content_html' => $this->convertEncoding($mail->textHtml),
            'content_plain' => $this->convertEncoding($mail->textPlain),
            'attachments' => $this->determineAttachments($mail->getAttachments()),
            'time_to_send' => $this->determineTimeToSend(explode('/', $folder)[2], $mail->date)
        ];
    }
     private function formatEmailAddresses(?array $addresses): ?array
    {
        if (empty($addresses)) {
            return null;
        }
        return array_map(
            function ($key, $value) {
                return [
                    'email' => $key,
                    'name' => $key === $value ? null : str_replace(' (' . $key . ')', '', $value)
                ];
            },
            array_keys($addresses),
            $addresses
        );
    }
     private function convertEncoding(string $text): string
    {
        return mb_detect_encoding($text, 'UTF-8, ISO-8859-1') !== 'UTF-8'
            ? \UConverter::transcode($text, 'UTF8', 'ISO-8859-1')
            : $text;
    }
     private function determineAttachments(array $attachmentsImap): array
    {
        $attachments = [];
        if (!empty($attachmentsImap)) {
            foreach ($attachmentsImap as $attachment) {
                $attachments[] = [
                    'name' => $attachment->name,
                    'file' => $attachment->filePath,
                    'disposition' => $attachment->disposition,
                    'inline_id' => $attachment->contentId
                ];
            }
        }
        return $attachments;
    }
     private function determineTimeToSend(string $delayTime, string $date): ?string
    {
        $timeToSend = null;
        if ($delayTime === 'THIS EVENING') {
            $timeToSend = date('Y-m-d', strtotime($date)) . ' 18:00:00';
        } elseif ($delayTime === 'THIS NIGHT') {
            $timeToSend =
                date('Y-m-d', strtotime($date . (date('H', strtotime($date)) >= 4 ? ' + 1 day' : ''))) . ' 03:42:00';
        } elseif ($delayTime === 'NEXT MORNING') {
            $timeToSend =
                date('Y-m-d', strtotime($date . (date('H', strtotime($date)) >= 9 ? ' + 1 day' : ''))) . ' 09:00:00';
        } elseif ($delayTime === 'NEXT WEEK') {
            $date = new \DateTime(date('Y-m-d', strtotime($date)));
            $date->modify('next monday');
            $timeToSend = $date->format('Y-m-d') . ' 09:00:00';
        }
        return $timeToSend;
    }
     private function sendMail(object $preparedMail): void
    {
        $mail = new \PHPMailer\PHPMailer\PHPMailer(true);
         $mail->isSMTP();
        $mail->Host = $_SERVER['HOST_SMTP'];
        $mail->Port = $_SERVER['PORT_SMTP'];
        $mail->Username = $_SERVER['USERNAME'];
        $mail->Password = $_SERVER['PASSWORD'];
        $mail->SMTPSecure = $_SERVER['ENCRYPTION'];
        $mail->setFrom($_SERVER['FROM_ADDRESS'], $_SERVER['FROM_NAME']);
        $mail->SMTPAuth = true;
        $mail->SMTPOptions = [
            'tls' => ['verify_peer' => false, 'verify_peer_name' => false, 'allow_self_signed' => true],
            'ssl' => ['verify_peer' => false, 'verify_peer_name' => false, 'allow_self_signed' => true]
        ];
        $mail->CharSet = 'utf-8';
         $this->addRecipients($mail, $preparedMail->to, 'addAddress');
        $this->addRecipients($mail, $preparedMail->cc, 'addCC');
        $this->addRecipients($mail, $preparedMail->bcc, 'addBCC');
         $mail->isHTML(!empty($preparedMail->content_html));
        $mail->Subject = $preparedMail->subject;
         if (!empty($preparedMail->content_html)) {
            $mail->Body = $preparedMail->content_html;
            $mail->AltBody = !empty($preparedMail->content_plain)
                ? $preparedMail->content_plain
                : strip_tags(str_replace(['<br>', '<br/>', '<br />'], "\r\n", $preparedMail->content_html));
        } else {
            $mail->Body = $preparedMail->content_plain;
        }
         $this->addAttachments($mail, $preparedMail->attachments);
         $mail->send();
         $this->mailbox->moveMail($preparedMail->id, $_SERVER['FOLDER_OUTBOX']);
    }
     private function addRecipients(\PHPMailer\PHPMailer\PHPMailer $mail, ?array $recipients, string $method): void
    {
        if (!empty($recipients)) {
            foreach ($recipients as $recipient) {
                $mail->$method($recipient['email'], $recipient['name']);
            }
        }
    }
     private function addAttachments(\PHPMailer\PHPMailer\PHPMailer $mail, array $attachments): void
    {
        if (!empty($attachments)) {
            foreach ($attachments as $attachment) {
                if (!empty($attachment['file']) && !empty($attachment['name']) && file_exists($attachment['file'])) {
                    if ($attachment['disposition'] === 'attachment') {
                        $mail->addAttachment($attachment['file'], $attachment['name']);
                    } elseif ($attachment['disposition'] === 'inline') {
                        $mail->AddEmbeddedImage(
                            $attachment['file'],
                            $attachment['inline_id'],
                            $attachment['name'],
                            'base64',
                            'image/png'
                        );
                    }
                }
            }
        }
    }
}
 $md = new MailDelay();
$md->init();

Damit dieses Script wiederkehrend läuft, legen wir uns eine Bash-Datei an:

#!/usr/bin/env bash
source $(dirname "$0")/.env
"$PHP_EXECUTABLE" $(dirname "$0")/maildelay.php

Wir führen dieses Script dann alle 10 Minuten per Cronjob aus:

*/10 * * * * /path/to/maildelay/maildelay.sh 2>&1

Diese Lösung ist flexibel anpassbar, datenschutzfreundlich und kostenlos. Sie funktioniert mit jedem (IMAP-kompatiblen) Konto. Doch unabhängig davon, ob man die integrierte Funktion in Outlook oder Gmail nutzt, auf erweiterte Tools wie Boomerang zurückgreift oder eine individuelle Lösung implementiert – das zeitversetzte Senden von E-Mails ist ein wertvolles Instrument, um die eigene Produktivität zu steigern.

Zurück