Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Microsoft EWS Integration with phpEWS library into [mail] app #15

Open
wants to merge 80 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
80 commits
Select commit Hold shift + click to select a range
469b184
EWS account creation in wizard, still errors left and SMTP
asig2016 Jul 26, 2017
0db6b91
Successfully create account with EWS as acc_imap_type
asig2016 Jul 31, 2017
99401ea
EWS account now displays folder tree
asig2016 Jul 31, 2017
fc2807d
Display Mail folders
asig2016 Aug 1, 2017
fbf4347
Get Message and attachments from EWS
asig2016 Aug 3, 2017
5721ec5
Fix git typo for zip download
asig2016 Aug 3, 2017
a3b19bf
Select inbox or public folders
asig2016 Aug 10, 2017
f6e7975
Added public folder settings
asig2016 Aug 28, 2017
7db160b
Switching between INBOX and public_folders type
asig2016 Aug 29, 2017
260b42b
Show only selected folders in tree
asig2016 Aug 29, 2017
6d291bf
Moving mails, deleting mails, setting read/unread
asig2016 Aug 29, 2017
85b54be
Check for permissions when deleting
asig2016 Aug 29, 2017
7ad0f10
Default Folder. Move Permissions
asig2016 Aug 30, 2017
74b044f
Embedded images displayed
asig2016 Aug 30, 2017
39d7319
Hide embedded from attachments, double check cids
asig2016 Aug 30, 2017
d53dd59
Fix bugs
asig2016 Aug 30, 2017
75872b4
Fix 'To' address
asig2016 Aug 30, 2017
0294816
Fetch special use folders. Sending, without storing to folder
asig2016 Sep 1, 2017
d2d91fe
Saving mail to drive/vfs. Saving copy of mail when sending
asig2016 Sep 7, 2017
e287770
Raw Header ok, js loop bug in EWS settings ok
asig2016 Sep 7, 2017
4bf33f1
Add mail_extra_actions hook
asig2016 Sep 7, 2017
2d6d819
Folder tree structure, ids in session, clean up
asig2016 Sep 7, 2017
b379187
Bug fixes, some documentation added
asig2016 Sep 7, 2017
bbe976f
Refactor getFolders. Add, move, rename, delete exchange folders
asig2016 Sep 8, 2017
eb82112
Unite permissions in ews_permissions. Apply permissions, order saved
asig2016 Sep 11, 2017
96e8c03
Global permissions flag
asig2016 Sep 11, 2017
5162e99
Delete hook, BINARY for folders
asig2016 Sep 11, 2017
30e54be
Filters functionality, build restrictions for phpEWS
asig2016 Sep 14, 2017
1f898c5
ews_folder is now case sensitive
asig2016 Sep 15, 2017
1096338
UI changes in EWS Folder settings
asig2016 Sep 15, 2017
8adb3fd
Add EWS Version in acc_imap_type
asig2016 Sep 18, 2017
59f7c13
Notifications for mails work
asig2016 Sep 18, 2017
e327504
Update DB table so that ews_folder field is case sensitive
asig2016 Sep 18, 2017
bdb8919
Add folder permissions, fix bugs
asig2016 Sep 18, 2017
2ebed9a
Custom folder permissions in popup
asig2016 Sep 18, 2017
e37b5c2
Add permissions file
asig2016 Sep 19, 2017
e9c5e05
Transition to popup, fixed bugs, fixed height
asig2016 Sep 19, 2017
3489852
UI improvements folder popup
asig2016 Sep 19, 2017
437e8c0
Save on apply, show all folders selected
asig2016 Sep 19, 2017
d15dc7b
Move to EWS functionality fixes
asig2016 Sep 19, 2017
7fbc866
Unknown error with some newsletters. Fall back to text
asig2016 Sep 19, 2017
0c9032f
Actually move to folder, refresh opener
asig2016 Sep 19, 2017
7974e68
Fix error when moving multiple items
asig2016 Sep 20, 2017
cbbc0d2
Get correct version, use appropriate port
asig2016 Sep 20, 2017
252a2d4
Add classes in rows in mail mobile
asig2016 Sep 20, 2017
b058685
Use folder names, not Ids
asig2016 Sep 20, 2017
50d661f
Include attachments when forwarding emails
asig2016 Sep 21, 2017
4640cd0
Checking for folder id ok
asig2016 Sep 21, 2017
c8060ef
Empty Trash
asig2016 Sep 21, 2017
12d57ec
Move to selected trash folder
asig2016 Sep 21, 2017
2f92362
Use unique identifiers for session
asig2016 Sep 22, 2017
a4b8064
Save mail to Sent in public folders
asig2016 Sep 22, 2017
79bee27
Do not create multiple drafts
asig2016 Sep 25, 2017
5419a27
Documentation, on copy to EWS, don't hide element
asig2016 Oct 2, 2017
76efdbe
Add hook after renaming folder
asig2016 Oct 2, 2017
3dfaf7b
Add hooks in mail_compose
asig2016 Oct 3, 2017
06986f3
Change hook placement in compose
asig2016 Oct 5, 2017
964b57d
Fix warning in compose
asig2016 Oct 5, 2017
d64c25c
Decode attachment id, fill in mimetype if missing
asig2016 Oct 10, 2017
92ab1ee
Displaying mails when attached to other mails
asig2016 Oct 11, 2017
fad324d
Get mail without body in case of corrupt XML
asig2016 Oct 24, 2017
39748bc
Disable not supported functionality
asig2016 Oct 30, 2017
0c19635
Identation fixes
asig2016 Nov 22, 2017
6169fea
Identation fixes
asig2016 Nov 22, 2017
16c502d
Disable flags for EWS
asig2016 Nov 22, 2017
7e4af2a
Custom Permissions Format like Acl
asig2016 Nov 22, 2017
4e21a7c
First merge with master
filkaris Nov 28, 2017
9f9cc26
New composer lockfile
asig2016 Nov 28, 2017
b081e96
Fix identation according to coding standards
filkaris Nov 28, 2017
73c8504
Merge branch 'ews_project' of https://github.com/filkaris/egroupware …
filkaris Nov 28, 2017
e7acdd6
More identation fixes
filkaris Nov 28, 2017
9ba678d
More Identation Fixes
filkaris Nov 28, 2017
4ca0231
More Identation Fixes
filkaris Nov 28, 2017
4da3a1a
More Identation Fixes
filkaris Nov 28, 2017
82fbe8e
More Identation Fixes
filkaris Nov 28, 2017
83337d0
More Identation Fixes
filkaris Nov 28, 2017
4de9e50
Last Identation Fixes
filkaris Nov 28, 2017
62ee140
Disable custom permissions for INBOX accounts
asig2016 Nov 29, 2017
fbafbb9
Disable custom permissions for INBOX accounts
asig2016 Nov 29, 2017
f4a82da
Added description for custom permissions
asig2016 Dec 19, 2017
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
152 changes: 130 additions & 22 deletions admin/inc/class.admin_mail.inc.php
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,8 @@ class admin_mail
public $public_functions = array(
'add' => true,
'edit' => true,
'ajax_activeAccounts' => true
'ajax_activeAccounts' => true,
'ews_custom_permissions' => true
);

/**
Expand Down Expand Up @@ -196,16 +197,19 @@ public function add(array $content=array(), $msg='', $msg_type='success')
'acc_imap_port' => 993,
'manual_class' => 'emailadmin_manual',
);
// Select Options
$sel_options['acc_imap_ssl'] = self::$ssl_types;
// Not listing other server types, since this could be single account
// EGroupware only allows different types for multiple accounts
$sel_options['acc_imap_type'] = Mail\Types::getIMAPServerTypes(false);
Framework::message($msg ? $msg : (string)$_GET['msg'], $msg_type);

if (!empty($content['acc_imap_host']) || !empty($content['acc_imap_username']))
{
$readonlys['button[manual]'] = true;
unset($content['manual_class']);
}
$tpl->exec(static::APP_CLASS.'autoconfig', $content, array(
'acc_imap_ssl' => self::$ssl_types,
), $readonlys, $content, 2);
$tpl->exec(static::APP_CLASS.'autoconfig', $content, $sel_options, $readonlys, $content, 2);
}

/**
Expand Down Expand Up @@ -285,17 +289,17 @@ public function autoconfig(array $content)

$e = null;
try {
$content['output'] .= "\n".Api\DateTime::to('now', 'H:i:s').": Trying $ssl connection to $host:$port ...\n";
$content['output'] .= "\n" . Api\DateTime::to('now', 'H:i:s') . ": Trying $ssl connection to $host:$port ...\n";
$content['acc_imap_port'] = $port;

$imap = self::imap_client($content, self::TIMEOUT);

//$content['output'] .= array2string($imap->capability());
$imap->login();
$content['output'] .= "\n".lang('Successful connected to %1 server%2.', 'IMAP', ' '.lang('and logged in'))."\n";
$content['output'] .= "\n" . lang('Successful connected to %1 server%2.', 'IMAP', ' ' . lang('and logged in')) . "\n";
if (!$imap->isSecureConnection())
{
$content['output'] .= lang('Connection is NOT secure! Everyone can read eg. your credentials.')."\n";
$content['output'] .= lang('Connection is NOT secure! Everyone can read eg. your credentials.') . "\n";
$content['acc_imap_ssl'] = 'no';
}
//$content['output'] .= "\n\n".array2string($imap->capability());
Expand Down Expand Up @@ -331,8 +335,13 @@ public function autoconfig(array $content)
if ($connected) // continue with next wizard step: define folders
{
unset($content['button']);
return $this->folder($content, lang('Successful connected to %1 server%2.', 'IMAP', ' '.lang('and logged in')).
($imap->isSecureConnection() ? '' : "\n".lang('Connection is NOT secure! Everyone can read eg. your credentials.')));
//EWS: skip steps
if (Mail\Account::is_ews_type($content['acc_imap_type']))
return $this->smtp($content, lang('Successful connected to %1 server%2.', 'EWS', ' ' . lang('and logged in')) .
($imap->isSecureConnection() ? '' : "\n" . lang('Connection is NOT secure! Everyone can read eg. your credentials.')));
else
return $this->folder($content, lang('Successful connected to %1 server%2.', 'IMAP', ' ' . lang('and logged in')) .
($imap->isSecureConnection() ? '' : "\n".lang('Connection is NOT secure! Everyone can read eg. your credentials.')));
}
// add validation error, if we can identify a field
if (!$connected && $e instanceof Horde_Imap_Client_Exception)
Expand All @@ -352,6 +361,9 @@ public function autoconfig(array $content)
$readonlys['button[manual]'] = true;
unset($content['manual_class']);
$sel_options['acc_imap_ssl'] = self::$ssl_types;
// Not listing other server types, since this could be single account
// EGroupware only allows different types for multiple accounts
$sel_options['acc_imap_type'] = Mail\Types::getIMAPServerTypes(false);
$tpl = new Etemplate('admin.mailwizard');
$tpl->exec(static::APP_CLASS.'autoconfig', $content, $sel_options, $readonlys, $content, 2);
}
Expand Down Expand Up @@ -630,7 +642,10 @@ public function smtp(array $content, $msg='')
switch($button)
{
case 'back':
return $this->sieve($content);
if (Mail\Account::is_ews_type($content['acc_imap_type']))
return $this->add($content);
else
return $this->sieve($content);
}
}
// first try: hide manual config
Expand Down Expand Up @@ -824,6 +839,7 @@ public function smtp(array $content, $msg='')
*/
public function edit(array $content=null, $msg='', $msg_type='success')
{
unset($content['manual_class']);
// app is trying to tell something, while redirecting to wizard
if (empty($content) && $_GET['acc_id'] && empty($msg) && !empty( $_GET['msg']))
{
Expand Down Expand Up @@ -893,6 +909,7 @@ public function edit(array $content=null, $msg='', $msg_type='success')
{
if (!$content['acc_'.$type.'_ssl']) $content['acc_'.$type.'_ssl'] = 'no';
}

}
catch(Api\Exception\NotFound $e) {
if (self::$debug) _egw_log_exception($e);
Expand Down Expand Up @@ -933,8 +950,9 @@ public function edit(array $content=null, $msg='', $msg_type='success')
'acc_smtp_type' => true, 'acc_smtp_auth_session' => true,
);
}

// ensure correct values for single user mail accounts (we only hide them client-side)
if (!($is_multiple = Mail\Account::is_multiple($content)))
if (!($is_multiple = Mail\Account::is_multiple($content)) && $content['acc_imap_type'] != 'EGroupware\Api\Mail\Imap' && !Mail\Account::is_ews_type( $content['acc_imap_type'] ) )
{
$content['acc_imap_type'] = 'EGroupware\\Api\\Mail\\Imap';
unset($content['acc_imap_login_type']);
Expand Down Expand Up @@ -1086,6 +1104,15 @@ public function edit(array $content=null, $msg='', $msg_type='success')
}
$content['accounts'][$content['acc_id']] = Mail\Account::identity_name($content, false);
}
if ($content['acc_imap_type'] && Mail\Account::is_ews_type($content['acc_imap_type']))
{
if ($content['clear_grid'])
{
$content['ews_permissions'] = array();
$content['clear_grid'] = false;
Api\Mail_EWS::storeFolderPermissions($content['ews_permissions'], $content['acc_id']);
}
}
}
else
{
Expand Down Expand Up @@ -1220,6 +1247,24 @@ public function edit(array $content=null, $msg='', $msg_type='success')
$tpl->setElementAttribute($folder, 'allowFreeEntries', true);
}
}
elseif (Mail\Account::is_ews_type($content['acc_imap_type']))
{
try {

$sel_options['acc_folder_sent'] = $sel_options['acc_folder_trash'] =
$sel_options['acc_folder_draft'] = $sel_options['acc_folder_template'] =
$sel_options['acc_folder_junk'] = $sel_options['acc_folder_archive'] =
$sel_options['notify_folders'] = $sel_options['acc_folder_ham'] =
Api\Mail\EWS\Lib::getFoldersSelOptions($content['acc_id'], true);
}
catch (Exception $e)
{
if (self::$debug) _egw_log_exception($e);
// let user know what the problem is and that he can fix it using wizard or deleting
$msg = lang($e->getMessage()) . "\n\n" . lang('You can use wizard to fix account settings or delete account.');
$msg_type = 'error';
}
}
else
{
try {
Expand Down Expand Up @@ -1253,6 +1298,13 @@ public function edit(array $content=null, $msg='', $msg_type='success')
$sel_options['ident_id'] = $content['identities'];
$sel_options['acc_id'] = $content['accounts'];
$sel_options['acc_further_identities'] = self::$further_identities;
$sel_options['acc_ews_type'] = array(
'inbox' => 'Inbox',
'public_folders' => 'Public Folders'
);

// Disable permissions if inbox
$content['isInbox'] = ( $content['acc_ews_type'] == 'inbox' );

// user is allowed to create or edit further identities
if ($edit_access || $content['acc_further_identities'])
Expand Down Expand Up @@ -1334,6 +1386,12 @@ public function edit(array $content=null, $msg='', $msg_type='success')
}
}

// Disable EWS tab for other types
if ($content['acc_imap_type'] && !Mail\Account::is_ews_type($content['acc_imap_type']))
{
$readonlys['tabs']['admin.mailaccount.ews'] = true;
}

// account allows users to change forwards
if (!$edit_access && !$readonlys['tabs']['admin.mailaccount.aliases'] && $content['acc_user_forward'])
{
Expand Down Expand Up @@ -1371,6 +1429,23 @@ public function edit(array $content=null, $msg='', $msg_type='success')
}
$content['admin_actions'] = (bool)$admin_actions;

if ($content['acc_imap_type'] && Mail\Account::is_ews_type($content['acc_imap_type']))
{
try
{
$content['acc_ews_apply_permissions'] = (int) $content['acc_ews_apply_permissions'];
} catch (Exception $e)
{
if (self::$debug)
{
_egw_log_exception($e);
}
// let user know what the problem is and that he can fix it using wizard or deleting
$msg = lang($e->getMessage()) . "\n\n" . lang('You can use wizard to fix account settings or delete account.');
$msg_type = 'error';
}
}

//try to fix identities with no domain part set e.g. alias as identity
if (!strpos($content['ident_email'], '@'))
{
Expand All @@ -1380,6 +1455,31 @@ public function edit(array $content=null, $msg='', $msg_type='success')
$tpl->exec(static::APP_CLASS.'edit', $content, $sel_options, $readonlys, $content, 2);
}

public function ews_custom_permissions( $content = array() )
{
$dtmpl = new Etemplate('admin.mailaccount.permissions');
$acc_id = $_GET['acc_id']? $_GET['acc_id']: $content['acc_id'];
$content['acc_id'] = $acc_id;

$sel_options['ews_permissions'] = Api\Mail_EWS::getFolderPermissionsSelOptions( $content['acc_id'] );
$names = array_values( $sel_options['ews_permissions'][1]['ews_move_to'] );
$sel_options['mailbox'] = array_combine( $names, $names );

if ( $content['save'] || $content['apply'] ) {
$res = Api\Mail_EWS::storeFolderPermissions( $content['ews_permissions'], $content['acc_id'] );
$msg = lang('Operation Successful');
if ( $res && $content['save'] ) {
Framework::message( $msg );
Framework::window_close();
}
$content['msg'] = $msg;
}

$content['ews_permissions'] = Api\Mail_EWS::getFolderPermissions( $content['acc_id'] );
$readonlys = array();
$dtmpl->exec('admin.admin_mail.ews_custom_permissions', $content,$sel_options,$readonlys,$content,2);
}

/**
* Replace 0 with '' or back
*
Expand Down Expand Up @@ -1407,17 +1507,25 @@ private static function fix_account_id_0(&$account_id=null, $back=false)
* @param int $timeout =null default use value returned by Mail\Imap::getTimeOut()
* @return Horde_Imap_Client_Socket
*/
protected static function imap_client(array $content, $timeout=null)
{
return new Horde_Imap_Client_Socket(array(
'username' => $content['acc_imap_username'],
'password' => $content['acc_imap_password'],
'hostspec' => $content['acc_imap_host'],
'port' => $content['acc_imap_port'],
'secure' => self::$ssl2secure[(string)array_search($content['acc_imap_ssl'], self::$ssl2type)],
'timeout' => $timeout > 0 ? $timeout : Mail\Imap::getTimeOut(),
'debug' => self::DEBUG_LOG,
));
protected static function imap_client(array $content, $timeout = null) {
//EWS: Instantiate different object
if (Mail\Account::is_ews_type($content['acc_imap_type']))
{
$class = $content['acc_imap_type'];
return new $class($content);
}
else
{
return new Horde_Imap_Client_Socket(array(
'username' => $content['acc_imap_username'],
'password' => $content['acc_imap_password'],
'hostspec' => $content['acc_imap_host'],
'port' => $content['acc_imap_port'],
'secure' => self::$ssl2secure[(string) array_search($content['acc_imap_ssl'], self::$ssl2type)],
'timeout' => $timeout > 0 ? $timeout : Mail\Imap::getTimeOut(),
'debug' => self::DEBUG_LOG,
));
}
}

/**
Expand Down
54 changes: 54 additions & 0 deletions admin/js/app.js
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,11 @@ app.classes.admin = AppJS.extend(
*/
acl_dialog: null,

/**
* Keep widgets from triggering again
*/
widget_active: false,

/**
* Constructor
*
Expand Down Expand Up @@ -117,6 +122,9 @@ app.classes.admin = AppJS.extend(
case 'admin.mailaccount':
this.account_hide_not_applying();
break;
case 'admin.mailaccount.permissions':
this.ews_switch_folder();
break;
}
},

Expand Down Expand Up @@ -1273,5 +1281,51 @@ app.classes.admin = AppJS.extend(
resizable: false,
position: 'left top'
}, et2_dialog._create_parent('mail'));
},
/**
* Switch EWS Type
*
* When switching, we want to immediately save and change the array that's being loaded
*
*/
ews_switch_type: function (_element, _widget)
{
var option1 = 'inbox';
var option2 = 'public_folders';
var clear = this.et2.getWidgetById('clear_grid');
var apply = this.et2.getWidgetById('button[apply]');
var that = this;
if (!that.widget_active) {
that.widget_active = true;
et2_dialog.show_dialog(function (_button) {
if (_button == 2) {
clear.set_value(true);
apply.getInstanceManager().submit(apply, false, apply.options.novalidate);
} else {
if (_widget.getValue() == option1)
_widget.set_value(option2);
else
_widget.set_value(option1);
that.widget_active = false;
}
}, egw.lang('Changing Account Type will delete all your current settings. Are you sure you want to continue?'),
egw.lang('Change Type'), null, et2_dialog.BUTTON_YES_NO, et2_dialog.WARNING_MESSAGE, undefined, egw);
}
},

/**
* Switch Folder in ews permissions
*
*/
ews_switch_folder: function()
{
var folder = this.et2.getWidgetById('mailbox');
jQuery('#admin-mailaccount-permissions_ews_permissions tr').each( function() {
if ( jQuery( this ).hasClass('th') ) return;
if ( jQuery( this ).find('input[name*=ews_name]').val() == folder.getValue() )
jQuery( this ).show();
else
jQuery( this ).hide();
});
}
});
11 changes: 11 additions & 0 deletions admin/templates/default/app.css
Original file line number Diff line number Diff line change
Expand Up @@ -177,3 +177,14 @@ select#admin-mailaccount_ident_id {
font-size: 110%;
}
#admin-acl {width:99%;}

#admin-mailaccount-permissions_ews_tabs .th {
border-bottom: none;
}
#admin-mailaccount-permissions_ews_tabs .et2_tabheader {
display:none;
}
#admin-mailaccount-permissions_ews_tabs #admin-mailaccount-permissions_ews_permissions .th {
visibility: collapse;
}

Loading