256 lines
8.1 KiB
PHP
256 lines
8.1 KiB
PHP
<?php
|
|
/**
|
|
* Zend Framework
|
|
*
|
|
* LICENSE
|
|
*
|
|
* This source file is subject to the new BSD license that is bundled
|
|
* with this package in the file LICENSE.txt.
|
|
* It is also available through the world-wide-web at this URL:
|
|
* http://framework.zend.com/license/new-bsd
|
|
* If you did not receive a copy of the license and are unable to
|
|
* obtain it through the world-wide-web, please send an email
|
|
* to license@zend.com so we can send you a copy immediately.
|
|
*
|
|
* @category Zend
|
|
* @package Zend_Mail
|
|
* @subpackage Storage
|
|
* @copyright Copyright (c) 2005-2015 Zend Technologies USA Inc. (http://www.zend.com)
|
|
* @license http://framework.zend.com/license/new-bsd New BSD License
|
|
* @version $Id$
|
|
*/
|
|
|
|
|
|
/**
|
|
* @see Zend_Mail_Storage_Folder
|
|
*/
|
|
|
|
/**
|
|
* @see Zend_Mail_Storage_Folder_Interface
|
|
*/
|
|
|
|
/**
|
|
* @see Zend_Mail_Storage_Maildir
|
|
*/
|
|
|
|
|
|
/**
|
|
* @category Zend
|
|
* @package Zend_Mail
|
|
* @subpackage Storage
|
|
* @copyright Copyright (c) 2005-2015 Zend Technologies USA Inc. (http://www.zend.com)
|
|
* @license http://framework.zend.com/license/new-bsd New BSD License
|
|
*/
|
|
class Zend_Mail_Storage_Folder_Maildir extends Zend_Mail_Storage_Maildir implements Zend_Mail_Storage_Folder_Interface
|
|
{
|
|
/**
|
|
* Zend_Mail_Storage_Folder root folder for folder structure
|
|
* @var Zend_Mail_Storage_Folder
|
|
*/
|
|
protected $_rootFolder;
|
|
|
|
/**
|
|
* rootdir of folder structure
|
|
* @var string
|
|
*/
|
|
protected $_rootdir;
|
|
|
|
/**
|
|
* name of current folder
|
|
* @var string
|
|
*/
|
|
protected $_currentFolder;
|
|
|
|
/**
|
|
* delim char for subfolders
|
|
* @var string
|
|
*/
|
|
protected $_delim;
|
|
|
|
/**
|
|
* Create instance with parameters
|
|
* Supported parameters are:
|
|
* - dirname rootdir of maildir structure
|
|
* - delim delim char for folder structur, default is '.'
|
|
* - folder intial selected folder, default is 'INBOX'
|
|
*
|
|
* @param array $params mail reader specific parameters
|
|
* @throws Zend_Mail_Storage_Exception
|
|
*/
|
|
public function __construct($params)
|
|
{
|
|
if (is_array($params)) {
|
|
$params = (object)$params;
|
|
}
|
|
|
|
if (!isset($params->dirname) || !is_dir($params->dirname)) {
|
|
/**
|
|
* @see Zend_Mail_Storage_Exception
|
|
*/
|
|
throw new Zend_Mail_Storage_Exception('no valid dirname given in params');
|
|
}
|
|
|
|
$this->_rootdir = rtrim($params->dirname, DIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR;
|
|
|
|
$this->_delim = isset($params->delim) ? $params->delim : '.';
|
|
|
|
$this->_buildFolderTree();
|
|
$this->selectFolder(!empty($params->folder) ? $params->folder : 'INBOX');
|
|
$this->_has['top'] = true;
|
|
$this->_has['flags'] = true;
|
|
}
|
|
|
|
/**
|
|
* find all subfolders and mbox files for folder structure
|
|
*
|
|
* Result is save in Zend_Mail_Storage_Folder instances with the root in $this->_rootFolder.
|
|
* $parentFolder and $parentGlobalName are only used internally for recursion.
|
|
*
|
|
* @return null
|
|
* @throws Zend_Mail_Storage_Exception
|
|
*/
|
|
protected function _buildFolderTree()
|
|
{
|
|
$this->_rootFolder = new Zend_Mail_Storage_Folder('/', '/', false);
|
|
$this->_rootFolder->INBOX = new Zend_Mail_Storage_Folder('INBOX', 'INBOX', true);
|
|
|
|
$dh = @opendir($this->_rootdir);
|
|
if (!$dh) {
|
|
/**
|
|
* @see Zend_Mail_Storage_Exception
|
|
*/
|
|
throw new Zend_Mail_Storage_Exception("can't read folders in maildir");
|
|
}
|
|
$dirs = array();
|
|
while (($entry = readdir($dh)) !== false) {
|
|
// maildir++ defines folders must start with .
|
|
if ($entry[0] != '.' || $entry == '.' || $entry == '..') {
|
|
continue;
|
|
}
|
|
if ($this->_isMaildir($this->_rootdir . $entry)) {
|
|
$dirs[] = $entry;
|
|
}
|
|
}
|
|
closedir($dh);
|
|
|
|
sort($dirs);
|
|
$stack = array(null);
|
|
$folderStack = array(null);
|
|
$parentFolder = $this->_rootFolder;
|
|
$parent = '.';
|
|
|
|
foreach ($dirs as $dir) {
|
|
do {
|
|
if (strpos($dir, $parent) === 0) {
|
|
$local = substr($dir, strlen($parent));
|
|
if (strpos($local, $this->_delim) !== false) {
|
|
/**
|
|
* @see Zend_Mail_Storage_Exception
|
|
*/
|
|
throw new Zend_Mail_Storage_Exception('error while reading maildir');
|
|
}
|
|
array_push($stack, $parent);
|
|
$parent = $dir . $this->_delim;
|
|
$folder = new Zend_Mail_Storage_Folder($local, substr($dir, 1), true);
|
|
$parentFolder->$local = $folder;
|
|
array_push($folderStack, $parentFolder);
|
|
$parentFolder = $folder;
|
|
break;
|
|
} else if ($stack) {
|
|
$parent = array_pop($stack);
|
|
$parentFolder = array_pop($folderStack);
|
|
}
|
|
} while ($stack);
|
|
if (!$stack) {
|
|
/**
|
|
* @see Zend_Mail_Storage_Exception
|
|
*/
|
|
throw new Zend_Mail_Storage_Exception('error while reading maildir');
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* get root folder or given folder
|
|
*
|
|
* @param string $rootFolder get folder structure for given folder, else root
|
|
* @return Zend_Mail_Storage_Folder root or wanted folder
|
|
* @throws Zend_Mail_Storage_Exception
|
|
*/
|
|
public function getFolders($rootFolder = null)
|
|
{
|
|
if (!$rootFolder || $rootFolder == 'INBOX') {
|
|
return $this->_rootFolder;
|
|
}
|
|
|
|
// rootdir is same as INBOX in maildir
|
|
if (strpos($rootFolder, 'INBOX' . $this->_delim) === 0) {
|
|
$rootFolder = substr($rootFolder, 6);
|
|
}
|
|
$currentFolder = $this->_rootFolder;
|
|
$subname = trim($rootFolder, $this->_delim);
|
|
while ($currentFolder) {
|
|
@list($entry, $subname) = @explode($this->_delim, $subname, 2);
|
|
$currentFolder = $currentFolder->$entry;
|
|
if (!$subname) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ($currentFolder->getGlobalName() != rtrim($rootFolder, $this->_delim)) {
|
|
/**
|
|
* @see Zend_Mail_Storage_Exception
|
|
*/
|
|
throw new Zend_Mail_Storage_Exception("folder $rootFolder not found");
|
|
}
|
|
return $currentFolder;
|
|
}
|
|
|
|
/**
|
|
* select given folder
|
|
*
|
|
* folder must be selectable!
|
|
*
|
|
* @param Zend_Mail_Storage_Folder|string $globalName global name of folder or instance for subfolder
|
|
* @return null
|
|
* @throws Zend_Mail_Storage_Exception
|
|
*/
|
|
public function selectFolder($globalName)
|
|
{
|
|
$this->_currentFolder = (string)$globalName;
|
|
|
|
// getting folder from folder tree for validation
|
|
$folder = $this->getFolders($this->_currentFolder);
|
|
|
|
try {
|
|
$this->_openMaildir($this->_rootdir . '.' . $folder->getGlobalName());
|
|
} catch(Zend_Mail_Storage_Exception $e) {
|
|
// check what went wrong
|
|
if (!$folder->isSelectable()) {
|
|
/**
|
|
* @see Zend_Mail_Storage_Exception
|
|
*/
|
|
throw new Zend_Mail_Storage_Exception("{$this->_currentFolder} is not selectable", 0, $e);
|
|
}
|
|
// seems like file has vanished; rebuilding folder tree - but it's still an exception
|
|
$this->_buildFolderTree($this->_rootdir);
|
|
/**
|
|
* @see Zend_Mail_Storage_Exception
|
|
*/
|
|
throw new Zend_Mail_Storage_Exception('seems like the maildir has vanished, I\'ve rebuild the ' .
|
|
'folder tree, search for an other folder and try again', 0, $e);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* get Zend_Mail_Storage_Folder instance for current folder
|
|
*
|
|
* @return Zend_Mail_Storage_Folder instance of current folder
|
|
* @throws Zend_Mail_Storage_Exception
|
|
*/
|
|
public function getCurrentFolder()
|
|
{
|
|
return $this->_currentFolder;
|
|
}
|
|
}
|