mirror of
https://github.com/roundcube/roundcubemail.git
synced 2026-03-02 22:34:03 +01:00
361 lines
11 KiB
PHP
361 lines
11 KiB
PHP
<?php
|
|
|
|
/**
|
|
* Managesieve (Sieve Filters)
|
|
*
|
|
* Plugin that adds a possibility to manage Sieve filters in Thunderbird's style.
|
|
* It's clickable interface which operates on text scripts and communicates
|
|
* with server using managesieve protocol. Adds Filters tab in Settings.
|
|
*
|
|
* @author Aleksander Machniak <alec@alec.pl>
|
|
*
|
|
* Configuration (see config.inc.php.dist)
|
|
*
|
|
* Copyright (C) The Roundcube Dev Team
|
|
* Copyright (C) Kolab Systems AG
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see https://www.gnu.org/licenses/.
|
|
*/
|
|
|
|
class managesieve extends rcube_plugin
|
|
{
|
|
public $task = 'mail|settings';
|
|
|
|
private $rc;
|
|
private $engine;
|
|
private $ui_initialized = false;
|
|
private $mail_headers_done = false;
|
|
|
|
/**
|
|
* Plugin initialization
|
|
*/
|
|
#[Override]
|
|
public function init()
|
|
{
|
|
$this->rc = rcube::get_instance();
|
|
|
|
$this->load_config();
|
|
|
|
$allowed_hosts = $this->rc->config->get('managesieve_allowed_hosts');
|
|
if (!empty($allowed_hosts) && !in_array($_SESSION['storage_host'], (array) $allowed_hosts)) {
|
|
return;
|
|
}
|
|
|
|
// register actions
|
|
$this->register_action('plugin.managesieve', [$this, 'managesieve_actions']);
|
|
$this->register_action('plugin.managesieve-action', [$this, 'managesieve_actions']);
|
|
$this->register_action('plugin.managesieve-vacation', [$this, 'managesieve_actions']);
|
|
$this->register_action('plugin.managesieve-forward', [$this, 'managesieve_actions']);
|
|
$this->register_action('plugin.managesieve-save', [$this, 'managesieve_save']);
|
|
$this->register_action('plugin.managesieve-saveraw', [$this, 'managesieve_saveraw']);
|
|
|
|
$task = $this->rc->task ?? null; // @phpstan-ignore-line
|
|
$action = $this->rc->action ?? null; // @phpstan-ignore-line
|
|
|
|
if ($task == 'settings') {
|
|
$this->add_hook('settings_actions', [$this, 'settings_actions']);
|
|
$this->init_ui();
|
|
} elseif ($task == 'mail') {
|
|
$this->add_hook('storage_init', [$this, 'storage_init']);
|
|
|
|
if ($action == 'show') {
|
|
$this->add_hook('message_headers_output', [$this, 'mail_headers']);
|
|
}
|
|
|
|
// inject Create Filter popup stuff
|
|
if (empty($action) || $action == 'show' || strpos($action, 'plugin.managesieve') === 0) {
|
|
$this->mail_task_handler();
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Initializes plugin's UI (localization, js script)
|
|
*/
|
|
public function init_ui()
|
|
{
|
|
if (!empty($this->ui_initialized)) {
|
|
return;
|
|
}
|
|
|
|
// load localization
|
|
$this->add_texts('localization/');
|
|
|
|
$sieve_action = strpos($this->rc->action, 'plugin.managesieve') === 0;
|
|
|
|
if ($this->rc->task == 'mail' || $sieve_action) {
|
|
// Injection of Timezone information into the JS Frontend.
|
|
// All the specifiers may be included in $config['time_format']
|
|
// However not all are easily parseable in the JS world, especially
|
|
// when it comes to Timezone abbreviation
|
|
$tz = new DateTimeZone($this->rc->config->get('timezone'));
|
|
$dt = new DateTime('now', $tz);
|
|
|
|
$this->rc->output->set_env('server_timezone_info', [
|
|
'e' => $dt->format('e'),
|
|
'I' => $dt->format('I'),
|
|
'O' => $dt->format('O'),
|
|
'P' => $dt->format('P'),
|
|
'p' => $dt->format('p'),
|
|
'T' => $dt->format('T'),
|
|
'Z' => $dt->format('Z'),
|
|
]);
|
|
$this->include_script('managesieve.js');
|
|
}
|
|
|
|
// include styles
|
|
$skin_path = $this->local_skin_path();
|
|
if ($sieve_action || ($this->rc->task == 'settings' && empty($_REQUEST['_framed']))) {
|
|
$this->include_stylesheet("{$skin_path}/managesieve.css");
|
|
} elseif ($this->rc->task == 'mail') {
|
|
$this->include_stylesheet("{$skin_path}/managesieve_mail.css");
|
|
}
|
|
|
|
$this->ui_initialized = true;
|
|
}
|
|
|
|
/**
|
|
* Adds Filters section in Settings
|
|
*/
|
|
public function settings_actions($args)
|
|
{
|
|
$vacation_mode = (int) $this->rc->config->get('managesieve_vacation');
|
|
$forward_mode = (int) $this->rc->config->get('managesieve_forward');
|
|
|
|
// register Filters action
|
|
if ($vacation_mode != 2 && $forward_mode != 2) {
|
|
$args['actions'][] = [
|
|
'action' => 'plugin.managesieve',
|
|
'class' => 'filter',
|
|
'label' => 'filters',
|
|
'domain' => 'managesieve',
|
|
'title' => 'filterstitle',
|
|
];
|
|
}
|
|
|
|
// register Vacation action
|
|
if ($vacation_mode > 0) {
|
|
$args['actions'][] = [
|
|
'action' => 'plugin.managesieve-vacation',
|
|
'class' => 'vacation',
|
|
'label' => 'vacation',
|
|
'domain' => 'managesieve',
|
|
'title' => 'vacationtitle',
|
|
];
|
|
}
|
|
|
|
// register Forward action
|
|
if ($forward_mode > 0) {
|
|
$args['actions'][] = [
|
|
'action' => 'plugin.managesieve-forward',
|
|
'class' => 'forward',
|
|
'label' => 'forward',
|
|
'domain' => 'managesieve',
|
|
'title' => 'forwardtitle',
|
|
];
|
|
}
|
|
|
|
return $args;
|
|
}
|
|
|
|
/**
|
|
* Add UI elements to the 'mailbox view' and 'show message' UI.
|
|
*/
|
|
public function mail_task_handler()
|
|
{
|
|
// make sure we're not in ajax request
|
|
if ($this->rc->output->type != 'html') {
|
|
return;
|
|
}
|
|
|
|
$vacation_mode = (int) $this->rc->config->get('managesieve_vacation');
|
|
$forward_mode = (int) $this->rc->config->get('managesieve_forward');
|
|
|
|
if ($vacation_mode == 2 || $forward_mode == 2) {
|
|
return;
|
|
}
|
|
|
|
// include js script and localization
|
|
$this->init_ui();
|
|
|
|
// add 'Create filter' item to message menu
|
|
$this->add_button([
|
|
'command' => 'managesieve-create',
|
|
'label' => 'managesieve.filtercreate',
|
|
'type' => 'link-menuitem',
|
|
'classact' => 'icon filterlink active',
|
|
'class' => 'icon filterlink disabled',
|
|
'innerclass' => 'icon filterlink',
|
|
], 'messagemenu'
|
|
);
|
|
|
|
// register some labels/messages
|
|
$this->rc->output->add_label('managesieve.newfilter', 'managesieve.usedata',
|
|
'managesieve.nodata', 'managesieve.nextstep', 'save');
|
|
|
|
$this->rc->session->remove('managesieve_current');
|
|
}
|
|
|
|
/**
|
|
* Get message headers for popup window
|
|
*/
|
|
public function mail_headers($args)
|
|
{
|
|
// this hook can be executed many times
|
|
if (!empty($this->mail_headers_done)) {
|
|
return $args;
|
|
}
|
|
|
|
$this->mail_headers_done = true;
|
|
|
|
$headers = $this->parse_headers($args['headers']);
|
|
|
|
if ($this->rc->action == 'preview') {
|
|
$this->rc->output->command('parent.set_env', ['sieve_headers' => $headers]);
|
|
} else {
|
|
$this->rc->output->set_env('sieve_headers', $headers);
|
|
}
|
|
|
|
return $args;
|
|
}
|
|
|
|
/**
|
|
* Plugin action handler
|
|
*/
|
|
public function managesieve_actions()
|
|
{
|
|
$uids = rcmail_action::get_uids(null, null, $multifolder, rcube_utils::INPUT_POST);
|
|
|
|
// handle fetching email headers for the new filter form
|
|
if (!empty($uids)) {
|
|
$mailbox = key($uids);
|
|
$message = new rcube_message($uids[$mailbox][0], $mailbox);
|
|
$headers = $this->parse_headers($message->headers);
|
|
|
|
$this->rc->output->set_env('sieve_headers', $headers);
|
|
$this->rc->output->command('managesieve_create', true);
|
|
$this->rc->output->send();
|
|
}
|
|
|
|
// handle other actions
|
|
$engine_type = $this->rc->action == 'plugin.managesieve-vacation' ? 'vacation' : '';
|
|
$engine_type = $this->rc->action == 'plugin.managesieve-forward' ? 'forward' : $engine_type;
|
|
$engine = $this->get_engine($engine_type);
|
|
|
|
$this->init_ui();
|
|
|
|
$engine->actions();
|
|
}
|
|
|
|
/**
|
|
* Forms save action handler
|
|
*/
|
|
public function managesieve_save()
|
|
{
|
|
// load localization
|
|
$this->add_texts('localization/', ['filters', 'managefilters']);
|
|
|
|
// include main js script
|
|
if ($this->api->output->type == 'html') {
|
|
$this->include_script('managesieve.js');
|
|
}
|
|
|
|
$engine = $this->get_engine();
|
|
$engine->save();
|
|
}
|
|
|
|
/**
|
|
* Raw form save action handler
|
|
*/
|
|
public function managesieve_saveraw()
|
|
{
|
|
$engine = $this->get_engine();
|
|
|
|
if (!$this->rc->config->get('managesieve_raw_editor', true)) {
|
|
return;
|
|
}
|
|
|
|
// load localization
|
|
$this->add_texts('localization/', ['filters', 'managefilters']);
|
|
|
|
$engine->saveraw();
|
|
}
|
|
|
|
/**
|
|
* Initializes engine object
|
|
*/
|
|
public function get_engine($type = null)
|
|
{
|
|
if (!$this->engine) {
|
|
// Add include path for internal classes
|
|
$include_path = $this->home . '/lib' . \PATH_SEPARATOR;
|
|
$include_path .= ini_get('include_path');
|
|
set_include_path($include_path);
|
|
|
|
$class_name = 'rcube_sieve_' . ($type ?: 'engine');
|
|
$this->engine = new $class_name($this);
|
|
}
|
|
|
|
return $this->engine;
|
|
}
|
|
|
|
/**
|
|
* Extract mail headers for new filter form
|
|
*/
|
|
private function parse_headers($headers)
|
|
{
|
|
$result = [];
|
|
$got_list = false;
|
|
|
|
if ($list_id = ($headers->others['list-id'] ?? null)) {
|
|
if (preg_match('/<([^>]+)>/', $list_id, $m)) {
|
|
$result[] = ['List-Id', $m[1], true];
|
|
$got_list = true;
|
|
}
|
|
}
|
|
|
|
if ($headers->subject) {
|
|
$result[] = ['Subject', rcube_mime::decode_header($headers->subject), !$got_list];
|
|
}
|
|
|
|
foreach (['From', 'To'] as $h) {
|
|
$hl = strtolower($h);
|
|
if (!empty($headers->{$hl})) {
|
|
$list = rcube_mime::decode_address_list($headers->{$hl});
|
|
foreach ($list as $item) {
|
|
if (!empty($item['mailto'])) {
|
|
$result[] = [$h, $item['mailto'], !$got_list];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* Handler for 'storage_init' hook
|
|
*
|
|
* @params array $p Hook parameters
|
|
*
|
|
* @return array Modified hook parameters
|
|
*/
|
|
public function storage_init($p)
|
|
{
|
|
// Fetch extra mail headers used by the plugin
|
|
$p['fetch_headers'] = trim(($p['fetch_headers'] ?? '') . ' List-Id');
|
|
return $p;
|
|
}
|
|
}
|