/library/Zend/Mail/Storage/Folder/Maildir.php
https://github.com/zucchi/zf2 · PHP · 217 lines · 151 code · 14 blank · 52 comment · 6 complexity · 1425ed15938f402f263d6997baee7f91 MD5 · raw file
- <?php
- /**
- * Zend Framework (http://framework.zend.com/)
- *
- * @link http://github.com/zendframework/zf2 for the canonical source repository
- * @copyright Copyright (c) 2005-2012 Zend Technologies USA Inc. (http://www.zend.com)
- * @license http://framework.zend.com/license/new-bsd New BSD License
- * @package Zend_Mail
- */
- namespace Zend\Mail\Storage\Folder;
- use Zend\Mail\Storage;
- use Zend\Mail\Storage\Exception;
- use Zend\Stdlib\ErrorHandler;
- /**
- * @category Zend
- * @package Zend_Mail
- * @subpackage Storage
- */
- class Maildir extends Storage\Maildir implements FolderInterface
- {
- /**
- * 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 structure, default is '.'
- * - folder initial selected folder, default is 'INBOX'
- *
- * @param $params array mail reader specific parameters
- * @throws \Zend\Mail\Storage\Exception\InvalidArgumentException
- */
- public function __construct($params)
- {
- if (is_array($params)) {
- $params = (object)$params;
- }
- if (!isset($params->dirname) || !is_dir($params->dirname)) {
- throw new Exception\InvalidArgumentException('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.
- *
- * @throws \Zend\Mail\Storage\Exception\RuntimeException
- */
- protected function _buildFolderTree()
- {
- $this->rootFolder = new Storage\Folder('/', '/', false);
- $this->rootFolder->INBOX = new Storage\Folder('INBOX', 'INBOX', true);
- ErrorHandler::start(E_WARNING);
- $dh = opendir($this->rootdir);
- $error = ErrorHandler::stop();
- if (!$dh) {
- throw new Exception\RuntimeException("can't read folders in maildir", 0, $error);
- }
- $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) {
- throw new Exception\RuntimeException('error while reading maildir');
- }
- array_push($stack, $parent);
- $parent = $dir . $this->delim;
- $folder = new Storage\Folder($local, substr($dir, 1), true);
- $parentFolder->$local = $folder;
- array_push($folderStack, $parentFolder);
- $parentFolder = $folder;
- break;
- } elseif ($stack) {
- $parent = array_pop($stack);
- $parentFolder = array_pop($folderStack);
- }
- } while ($stack);
- if (!$stack) {
- throw new Exception\RuntimeException('error while reading maildir');
- }
- }
- }
- /**
- * get root folder or given folder
- *
- * @param string $rootFolder get folder structure for given folder, else root
- * @throws \Zend\Mail\Storage\Exception\InvalidArgumentException
- * @return \Zend\Mail\Storage\Folder root or wanted folder
- */
- 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) {
- ErrorHandler::start(E_NOTICE);
- list($entry, $subname) = explode($this->delim, $subname, 2);
- ErrorHandler::stop();
- $currentFolder = $currentFolder->$entry;
- if (!$subname) {
- break;
- }
- }
- if ($currentFolder->getGlobalName() != rtrim($rootFolder, $this->delim)) {
- throw new Exception\InvalidArgumentException("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
- * @throws \Zend\Mail\Storage\Exception\RuntimeException
- */
- 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 (Exception\ExceptionInterface $e) {
- // check what went wrong
- if (!$folder->isSelectable()) {
- throw new Exception\RuntimeException("{$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);
- throw new Exception\RuntimeException('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
- */
- public function getCurrentFolder()
- {
- return $this->currentFolder;
- }
- }