2010-06-09 20:36:13 +02:00
< ? php
2010-08-19 20:13:53 +02:00
/**
* EGroupware EMailAdmin : Support for Cyrus IMAP ( or other IMAP Server supporting Sieve )
*
* @ link http :// www . stylite . de
* @ package emailadmin
* @ author Ralf Becker < rb @ stylite . de >
* @ author Klaus Leithoff < kl @ stylite . de >
* @ author Lars Kneschke
* @ license http :// opensource . org / licenses / gpl - license . php GPL - GNU General Public License
* @ version $Id $
*/
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
require_once 'Net/IMAP.php' ;
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
define ( 'IMAP_NAMESPACE_PERSONAL' , 'personal' );
define ( 'IMAP_NAMESPACE_OTHERS' , 'others' );
define ( 'IMAP_NAMESPACE_SHARED' , 'shared' );
define ( 'IMAP_NAMESPACE_ALL' , 'all' );
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* This class holds all information about the imap connection .
* This is the base class for all other imap classes .
*
* Also proxies Sieve calls to emailadmin_sieve ( eg . it behaves like the former felamimail bosieve ),
* to allow IMAP plugins to also manage Sieve connection .
*/
class defaultimap extends Net_IMAP
{
2010-06-09 20:36:13 +02:00
/**
2010-08-19 20:13:53 +02:00
* Capabilities of this class ( pipe - separated ) : default , sieve , admin , logintypeemail
*/
const CAPABILITIES = 'default|sieve' ;
/**
* the password to be used for admin connections
2010-06-09 20:36:13 +02:00
*
2010-08-19 20:13:53 +02:00
* @ var string
2010-06-09 20:36:13 +02:00
*/
2010-08-19 20:13:53 +02:00
var $adminPassword ;
/**
* the username to be used for admin connections
*
* @ var string
*/
var $adminUsername ;
/**
* enable encryption
*
* @ var bool
*/
var $encryption ;
/**
* the hostname / ip address of the imap server
*
* @ var string
*/
var $host ;
/**
* the password for the user
*
* @ var string
*/
var $password ;
/**
* the port of the imap server
*
* @ var integer
*/
var $port = 143 ;
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* the username
*
* @ var string
*/
var $username ;
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* the domainname to be used for vmailmgr logins
*
* @ var string
*/
var $domainName = false ;
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* validate ssl certificate
*
* @ var bool
*/
var $validatecert ;
/**
* the mailbox delimiter
*
* @ var string
*/
var $mailboxDelimiter = '/' ;
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* the mailbox prefix . maybe used by uw - imap only ?
*
* @ var string
*/
var $mailboxPrefix = '~/mail' ;
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* is the mbstring extension available
*
* @ var unknown_type
*/
var $mbAvailable ;
/**
* Mailboxes which get automatic created for new accounts ( INBOX == '' )
*
* @ var array
*/
var $imapLoginType ;
var $defaultDomain ;
/**
* disable internal conversion from / to ut7
* get ' s used by Net_IMAP
*
* @ var array
*/
var $_useUTF_7 = false ;
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* a debug switch
*/
var $debug = false ;
/**
* Sieve available
*
* @ var boolean
*/
var $enableSieve = false ;
/**
* Hostname / IP of sieve host
*
* @ var string
*/
var $sieveHost ;
/**
* Port of Sieve service
*
* @ var int
*/
var $sievePort = 2000 ;
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* the construtor
*
* @ return void
*/
function __construct ()
{
if ( function_exists ( 'mb_convert_encoding' )) {
$this -> mbAvailable = TRUE ;
2010-06-09 20:36:13 +02:00
}
2010-08-19 20:13:53 +02:00
$this -> restoreSessionData ();
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
// construtor for Net_IMAP stuff
$this -> Net_IMAPProtocol ();
}
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* Magic method to re - connect with the imapserver , if the object get ' s restored from the session
*/
function __wakeup ()
{
#$this->openConnection($this->isAdminConnection); // we need to re-connect
}
/**
* adds a account on the imap server
*
* @ param array $_hookValues
* @ return bool true on success , false on failure
*/
function addAccount ( $_hookValues )
{
return true ;
}
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* updates a account on the imap server
*
* @ param array $_hookValues
* @ return bool true on success , false on failure
*/
function updateAccount ( $_hookValues )
{
return true ;
}
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* deletes a account on the imap server
*
* @ param array $_hookValues
* @ return bool true on success , false on failure
*/
function deleteAccount ( $_hookValues )
{
return true ;
}
function disconnect ()
{
//error_log(__METHOD__.function_backtrace());
$retval = parent :: disconnect ();
if ( PEAR :: isError ( $retval )) error_log ( __METHOD__ . $retval -> message );
$this -> _connected = false ;
}
/**
* converts a foldername from current system charset to UTF7
*
* @ param string $_folderName
* @ return string the encoded foldername
*/
function encodeFolderName ( $_folderName )
{
if ( $this -> mbAvailable ) {
return mb_convert_encoding ( $_folderName , " UTF7-IMAP " , $GLOBALS [ 'egw' ] -> translation -> charset ());
2010-06-09 20:36:13 +02:00
}
2010-08-19 20:13:53 +02:00
// if not
// we can encode only from ISO 8859-1
return imap_utf7_encode ( $_folderName );
}
/**
* returns the supported capabilities of the imap server
* return false if the imap server does not support capabilities
*
* @ return array the supported capabilites
*/
function getCapabilities ()
{
if ( ! is_array ( $this -> sessionData [ 'capabilities' ][ $this -> host ])) {
return false ;
2010-06-09 20:36:13 +02:00
}
2010-08-19 20:13:53 +02:00
return $this -> sessionData [ 'capabilities' ][ $this -> host ];
}
/**
* return the delimiter used by the current imap server
*
* @ return string the delimimiter
*/
function getDelimiter ()
{
return isset ( $this -> sessionData [ 'delimiter' ][ $this -> host ]) ? $this -> sessionData [ 'delimiter' ][ $this -> host ] : $this -> mailboxDelimiter ;
}
/**
* Create transport string
*
* @ return string the transportstring
*/
function _getTransportString ()
{
if ( $this -> encryption == 2 ) {
$connectionString = " tls:// " . $this -> host ;
} elseif ( $this -> encryption == 3 ) {
$connectionString = " ssl:// " . $this -> host ;
} else {
// no tls
$connectionString = $this -> host ;
2010-06-09 20:36:13 +02:00
}
2010-08-19 20:13:53 +02:00
return $connectionString ;
}
/**
* Create the options array for SSL / TLS connections
*
* @ return string the transportstring
*/
function _getTransportOptions ()
{
if ( $this -> validatecert === false ) {
2010-06-09 20:36:13 +02:00
if ( $this -> encryption == 2 ) {
2010-08-19 20:13:53 +02:00
return array (
'tls' => array (
'verify_peer' => false ,
'allow_self_signed' => true ,
)
);
2010-06-09 20:36:13 +02:00
} elseif ( $this -> encryption == 3 ) {
2010-08-19 20:13:53 +02:00
return array (
'ssl' => array (
'verify_peer' => false ,
'allow_self_signed' => true ,
)
);
2010-06-09 20:36:13 +02:00
}
2010-08-19 20:13:53 +02:00
} else {
if ( $this -> encryption == 2 ) {
return array (
'tls' => array (
'verify_peer' => true ,
'allow_self_signed' => false ,
)
);
} elseif ( $this -> encryption == 3 ) {
return array (
'ssl' => array (
'verify_peer' => true ,
'allow_self_signed' => false ,
)
);
2010-06-09 20:36:13 +02:00
}
}
2010-08-19 20:13:53 +02:00
return null ;
}
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* get the effective Username for the Mailbox , as it is depending on the loginType
* @ param string $_username
* @ return string the effective username to be used to access the Mailbox
*/
function getMailBoxUserName ( $_username )
{
2010-08-31 12:15:24 +02:00
switch ( $this -> loginType )
2010-06-09 20:36:13 +02:00
{
2010-08-31 12:15:24 +02:00
case 'email' :
$_username = $_username ;
$accountID = $GLOBALS [ 'egw' ] -> accounts -> name2id ( $_username );
$accountemail = $GLOBALS [ 'egw' ] -> accounts -> id2name ( $accountID , 'account_email' );
//$accountemail = $GLOBALS['egw']->accounts->read($GLOBALS['egw']->accounts->name2id($_username,'account_email'));
if ( ! empty ( $accountemail ))
2010-06-09 20:36:13 +02:00
{
2010-08-31 12:15:24 +02:00
list ( $lusername , $domain ) = explode ( '@' , $accountemail , 2 );
if ( strtolower ( $domain ) == strtolower ( $this -> domainName ) && ! empty ( $lusername ))
{
$_username = $lusername ;
}
2010-06-09 20:36:13 +02:00
}
2010-08-31 12:15:24 +02:00
break ;
case 'uidNumber' :
$_username = 'u' . $GLOBALS [ 'egw' ] -> accounts -> name2id ( $_username );
break ;
2010-06-09 20:36:13 +02:00
}
2011-09-20 12:20:56 +02:00
return strtolower ( $_username );
2010-08-19 20:13:53 +02:00
}
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* Create mailbox string from given mailbox - name and user - name
*
* @ param string $_folderName = ''
* @ return string utf - 7 encoded ( done in getMailboxName )
*/
function getUserMailboxString ( $_username , $_folderName = '' )
{
$nameSpaces = $this -> getNameSpaces ();
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
if ( ! isset ( $nameSpaces [ 'others' ])) {
return false ;
}
$_username = $this -> getMailBoxUserName ( $_username );
2010-08-31 12:15:24 +02:00
if ( $this -> loginType == 'vmailmgr' || $this -> loginType == 'email' || $this -> loginType == 'uidNumber' ) {
2010-08-19 20:13:53 +02:00
$_username .= '@' . $this -> domainName ;
}
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
$mailboxString = $nameSpaces [ 'others' ][ 0 ][ 'name' ] . $_username . ( ! empty ( $_folderName ) ? $nameSpaces [ 'others' ][ 0 ][ 'delimiter' ] . $_folderName : '' );
return $mailboxString ;
}
/**
* get list of namespaces
*
* @ return array array containing information about namespace
*/
function getNameSpaces ()
{
if ( ! $this -> _connected ) {
return false ;
2010-06-09 20:36:13 +02:00
}
2011-09-21 11:06:24 +02:00
static $nameSpace ;
if ( isset ( $nameSpace )) return $nameSpace ;
2010-08-19 20:13:53 +02:00
$retrieveDefault = false ;
if ( $this -> hasCapability ( 'NAMESPACE' )) {
2011-09-23 12:16:03 +02:00
$lNameSpace = $this -> getNamespace ();
if ( PEAR :: isError ( $lNameSpace )) {
if ( $this -> debug ) error_log ( " emailadmin::defaultimap->getNameSpaces: " . print_r ( $lNameSpace , true ));
2010-08-19 20:13:53 +02:00
$retrieveDefault = true ;
} else {
$result = array ();
2010-06-09 20:36:13 +02:00
2011-09-23 12:16:03 +02:00
$result [ 'personal' ] = $lNameSpace [ 'personal' ];
2010-06-09 20:36:13 +02:00
2011-09-23 12:16:03 +02:00
if ( is_array ( $lNameSpace [ 'others' ])) {
$result [ 'others' ] = $lNameSpace [ 'others' ];
2010-06-09 20:36:13 +02:00
}
2011-09-23 12:16:03 +02:00
if ( is_array ( $lNameSpace [ 'shared' ])) {
$result [ 'shared' ] = $lNameSpace [ 'shared' ];
2010-08-19 20:13:53 +02:00
}
2010-06-09 20:36:13 +02:00
}
2010-08-19 20:13:53 +02:00
}
if ( ! $this -> hasCapability ( 'NAMESPACE' ) || $retrieveDefault ) {
$delimiter = $this -> getHierarchyDelimiter ();
if ( PEAR :: isError ( $delimiter )) $delimiter = '/' ;
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
$result [ 'personal' ] = array (
0 => array (
'name' => '' ,
'delimiter' => $delimiter
)
);
}
2011-09-21 11:06:24 +02:00
$nameSpace = $result ;
2010-08-19 20:13:53 +02:00
return $result ;
}
/**
* returns the quota for given foldername
* gets quota for the current user only
*
* @ param string $_folderName
* @ return string the current quota for this folder
*/
# function getQuota($_folderName)
# {
# if(!is_resource($this->mbox)) {
# $this->openConnection();
# }
#
# if(function_exists('imap_get_quotaroot') && $this->supportsCapability('QUOTA')) {
# $quota = @imap_get_quotaroot($this->mbox, $this->encodeFolderName($_folderName));
# if(is_array($quota) && isset($quota['STORAGE'])) {
# return $quota['STORAGE'];
# }
# }
#
# return false;
# }
/**
* return the quota for another user
* used by admin connections only
*
* @ param string $_username
* @ return string the quota for specified user
*/
function getQuotaByUser ( $_username )
{
$mailboxName = $this -> getUserMailboxString ( $_username );
//error_log(__METHOD__.$mailboxName);
$storageQuota = $this -> getStorageQuota ( $mailboxName );
//error_log(__METHOD__.$_username);
//error_log(__METHOD__.$mailboxName);
if ( PEAR :: isError ( $storageQuota )) error_log ( __METHOD__ . $storageQuota -> message );
if ( is_array ( $storageQuota ) && isset ( $storageQuota [ 'QMAX' ])) {
return ( int ) $storageQuota [ 'QMAX' ];
2010-06-09 20:36:13 +02:00
}
2010-08-19 20:13:53 +02:00
return false ;
}
/**
* returns information about a user
2010-08-20 11:11:16 +02:00
*
* Only a stub , as admin connection requires , which is only supported for Cyrus
2010-08-19 20:13:53 +02:00
*
* @ param string $_username
* @ return array userdata
*/
function getUserData ( $_username )
{
2010-08-20 11:11:16 +02:00
return array ();
2010-08-19 20:13:53 +02:00
}
/**
* opens a connection to a imap server
*
* @ param bool $_adminConnection create admin connection if true
*
* @ return resource the imap connection
*/
2011-09-29 12:22:30 +02:00
function openConnection ( $_adminConnection = false , $_timeout = 20 )
2010-08-19 20:13:53 +02:00
{
//error_log(__METHOD__.function_backtrace());
unset ( $this -> _connectionErrorObject );
if ( $_adminConnection ) {
$username = $this -> adminUsername ;
$password = $this -> adminPassword ;
$options = '' ;
$this -> isAdminConnection = true ;
} else {
$username = $this -> loginName ;
$password = $this -> password ;
$options = $_options ;
$this -> isAdminConnection = false ;
2010-06-09 20:36:13 +02:00
}
2010-08-19 20:13:53 +02:00
$this -> setStreamContextOptions ( $this -> _getTransportOptions ());
2011-09-29 12:22:30 +02:00
$this -> setTimeout ( $_timeout );
2010-08-19 20:13:53 +02:00
if ( PEAR :: isError ( $status = parent :: connect ( $this -> _getTransportString (), $this -> port , $this -> encryption == 1 )) ) {
if ( $this -> debug ) error_log ( __METHOD__ . " Could not connect with " . $this -> _getTransportString () . " on Port " . $this -> port . " Encryption==1? " . $this -> encryption );
if ( $this -> debug ) error_log ( __METHOD__ . " Status connect: " . $status -> message );
$this -> _connectionErrorObject = $status ;
return false ;
}
if ( empty ( $username ))
2010-06-09 20:36:13 +02:00
{
2010-08-19 20:13:53 +02:00
if ( $this -> debug ) error_log ( __METHOD__ . " No username supplied. " . function_backtrace ());
2011-10-10 12:20:33 +02:00
if ( $this -> _connected ) $this -> disconnect (); // disconnect (if connected)
2010-08-19 20:13:53 +02:00
return false ;
2010-06-09 20:36:13 +02:00
}
2011-06-15 15:04:36 +02:00
if ( PEAR :: isError ( $status = parent :: login ( $username , $password , 'LOGIN' , ! $this -> isAdminConnection )) ) {
2010-08-19 20:13:53 +02:00
if ( $this -> debug ) error_log ( __METHOD__ . " Could not log in with -> " . $username . " : " . $password . " <- " );
if ( $this -> debug ) error_log ( __METHOD__ . " Status login: " . array2string ( $status -> message ));
//error_log(__METHOD__.'Called from:'.function_backtrace());
$this -> disconnect ();
$this -> _connectionErrorObject = $status ;
return false ;
}
return true ;
}
/**
* restore session variable
*
*/
function restoreSessionData ()
{
$this -> sessionData = $GLOBALS [ 'egw' ] -> session -> appsession ( 'imap_session_data' );
}
/**
* save session variable
*
*/
function saveSessionData ()
{
$GLOBALS [ 'egw' ] -> session -> appsession ( 'imap_session_data' , '' , $this -> sessionData );
}
/**
* set userdata
*
* @ param string $_username username of the user
* @ param int $_quota quota in bytes
* @ return bool true on success , false on failure
*/
function setUserData ( $_username , $_quota )
{
return true ;
}
/**
* check if imap server supports given capability
*
* @ param string $_capability the capability to check for
* @ return bool true if capability is supported , false if not
*/
function supportsCapability ( $_capability )
{
return $this -> hasCapability ( $_capability );
}
/**
* Instance of emailadmin_sieve
*
* @ var emailadmin_sieve
*/
private $sieve ;
public $scriptName ;
public $error ;
//public $error;
2010-06-09 20:36:13 +02:00
2010-08-19 20:13:53 +02:00
/**
* Proxy former felamimail bosieve methods to internal emailadmin_sieve instance
*
* @ param string $name
* @ param array $params
*/
public function __call ( $name , array $params = null )
{
2010-08-26 10:10:22 +02:00
if ( $this -> debug ) error_log ( __METHOD__ . '->' . $name . ' with params:' . array2string ( $params ));
2010-08-19 20:13:53 +02:00
switch ( $name )
2010-06-09 20:36:13 +02:00
{
2010-08-19 20:13:53 +02:00
case 'installScript' :
case 'getScript' :
case 'setActive' :
case 'setEmailNotification' :
case 'getEmailNotification' :
case 'setRules' :
case 'getRules' :
case 'retrieveRules' :
case 'getVacation' :
case 'setVacation' :
if ( is_null ( $this -> sieve ))
{
$this -> sieve = new emailadmin_sieve ( $this );
$this -> scriptName =& $this -> sieve -> scriptName ;
$this -> error =& $this -> sieve -> error ;
}
$ret = call_user_func_array ( array ( $this -> sieve , $name ), $params );
2010-08-20 09:57:15 +02:00
//error_log(__CLASS__.'->'.$name.'('.array2string($params).') returns '.array2string($ret));
2010-08-19 20:13:53 +02:00
return $ret ;
2010-06-09 20:36:13 +02:00
}
2010-08-19 20:13:53 +02:00
throw new egw_exception_wrong_parameter ( " No method ' $name ' implemented! " );
2010-06-09 20:36:13 +02:00
}
2010-08-26 09:45:39 +02:00
public function setVacationUser ( $_euser , $_scriptName , $_vacation )
{
2010-08-26 10:10:22 +02:00
if ( $this -> debug ) error_log ( __CLASS__ . '::' . __METHOD__ . ' User:' . array2string ( $_euser ) . ' Scriptname:' . array2string ( $_scriptName ) . ' VacationMessage:' . array2string ( $_vacation ));
2010-08-26 09:45:39 +02:00
if ( is_null ( $this -> sieve ))
{
$this -> sieve = new emailadmin_sieve ();
$this -> scriptName =& $this -> sieve -> scriptName ;
$this -> error =& $this -> sieve -> error ;
2010-08-26 10:22:34 +02:00
$this -> sieve -> icServer = $this ;
2010-08-26 09:45:39 +02:00
}
return $this -> sieve -> setVacationUser ( $_euser , $_scriptName , $_vacation );
}
2011-04-26 15:47:15 +02:00
/**
* set the asyncjob for a timed vacation
*
* @ param array $_vacation the vacation to set / unset
* @ return void
*/
function setAsyncJob ( $_vacation , $_scriptName = null )
{
// setting up an async job to enable/disable the vacation message
$async = new asyncservice ();
$user = ( isset ( $_vacation [ 'account_id' ]) &&! empty ( $_vacation [ 'account_id' ]) ? $_vacation [ 'account_id' ] : $GLOBALS [ 'egw_info' ][ 'user' ][ 'account_id' ]);
$async_id = ( isset ( $_vacation [ 'id' ]) &&! empty ( $_vacation [ 'id' ]) ? $_vacation [ 'id' ] : " felamimail-vacation- $user " );
$async -> delete ( $async_id ); // ="felamimail-vacation-$user");
$_scriptName = ( ! empty ( $_scriptName ) ? $_scriptName : ( isset ( $_vacation [ 'scriptName' ]) &&! empty ( $_vacation [ 'scriptName' ]) ? $_vacation [ 'scriptName' ] : 'felamimail' ));
$end_date = $_vacation [ 'end_date' ] + 24 * 3600 ; // end-date is inclusive, so we have to add 24h
if ( $_vacation [ 'status' ] == 'by_date' && time () < $end_date )
{
$time = time () < $_vacation [ 'start_date' ] ? $_vacation [ 'start_date' ] : $end_date ;
$async -> set_timer ( $time , $async_id , 'felamimail.bosieve.async_vacation' , $_vacation + array ( 'scriptName' => $_scriptName ), $user );
}
}
2010-08-19 20:13:53 +02:00
}