protonlabs / libsieve-php
libsieve-php 是一个用于管理和修改 sieve (RFC5228) 脚本的库。
v3.0.0
2022-05-11 07:51 UTC
Requires
- php: >=8.0
- ext-mbstring: *
- ext-simplexml: *
Requires (Dev)
- phpunit/phpunit: ^9
- protonlabs/php-coding-standard: ^4.0
- squizlabs/php_codesniffer: ^3.5
- vimeo/psalm: ^4.22
This package is auto-updated.
Last update: 2024-08-26 17:54:06 UTC
README
libsieve-php 是一个用于管理和修改 sieve (RFC5228) 脚本的库。它包含 sieve 语言(包括扩展)的解析器和 managesieve 协议的客户端。
本项目是从已停止维护的 PHP sieve 库https://sourceforge.net/projects/libsieve-php 衍生而来。
与 RFC 的不同之处
date
和currentdate
都允许传递任何字符串作为zone
参数。这使用户可以输入类似Europe/Zurich
的时区名称,而不是+0100
。我们允许这样做的原因是,像+0100
这样的偏移量不包含关于夏令时的信息,而这通常很需要。
使用示例
libsieve 将 sieve 脚本解析成树形结构。然后可以使用这个树来解释其含义。
将提供两个示例:一个基本示例和一个更复杂的示例。
基本示例:检查是否加载了扩展
在这个第一个例子中,我们将检查是否通过 require 节点加载了特定的扩展
<?php use Sieve\SieveParser; class ExtensionCheckExample { /** @var \Sieve\SieveTree the tree, obtained from the SieveParser */ protected $tree; /** * Parser constructor. * * @param string $sieve the sieve script to read. */ public function __construct(string $sieve) { $parser = new SieveParser(); try { $parser->parse($sieve); } catch (\Sieve\SieveException $se) { throw new \Exception("The provided sieve script is invalid!"); } // we store the tree, because it contains all the information. $this->tree = $parser->GetParseTree(); } /** * Checks if an extension is loaded. * * @param string $extension * @return bool */ public function isLoaded(string $extension) { /** @var int $root_node_id */ $root_node_id = $this->tree->getRoot(); // The root is not a node, we can only access its children $children = $this->tree->getChildren($root_node_id); foreach ($children as $child) { // The child is an id to a node, which can be access using the following: $node = $this->tree->getNode($child); // is can be used to check the type of node. if ($node->is(\Sieve\SieveToken::IDENTIFIER) && $node->text === "require") { if ($this->checkChildren($this->tree->getChildren($child), $extension)) { return true; } } } return false; } /** * Checks the arguments given to a require node, to know if it includes * * @param $children * @param string $extension * @return bool */ private function checkChildren($children, string $extension): bool { if (is_array($children)) { // it's a string list, let's loop over them. foreach ($children as $child) { if ($this->checkChildren($child, $extension)) { return true; } } return false; } $node = $this->tree->getNode($children); return $node->is(\Sieve\SieveToken::QUOTED_STRING) && $extension === trim($node->text, '"'); } } // load a script, from the tests folder. $sieve = file_get_contents(__DIR__ . './tests/good/currentdate.siv'); $runner = new ExtensionCheckExample($sieve); var_dump($runner->isLoaded("variables"));
复杂示例:输出树
这个第二个例子将从解析后的树中打印出 sieve 脚本(注意这也可以通过方法 $tree->dump()
直接完成)。
<?php use Sieve\SieveParser; use Sieve; class UsageExample { /** @var \Sieve\SieveTree the tree, obtained from the SieveParser */ protected $tree; /** * Parser constructor. * * @param string $sieve the sieve script to read. */ public function __construct(string $sieve) { $parser = new SieveParser(); try { $parser->parse($sieve); } catch (\Sieve\SieveException $se) { throw new \Exception("The provided sieve script is invalid!"); } // we store the tree, because it contains all the information. $this->tree = $parser->GetParseTree(); } /** * Displays the tree */ public function display() { /** @var int $root_node_id */ $root_node_id = $this->tree->getRoot(); // The root is not a node, we can only access its children $children = $this->tree->getChildren($root_node_id); $this->displayNodeList($children); } /** * Loop over a list of nodes, and display them. * * @param int[] $nodes a list of node ids. * @param string $indent */ private function displayNodeList(array $nodes, string $indent = '') { foreach ($nodes as $node) { $this->displayNode($node, $indent); } } /** * Display a node and its children. * * @param int $node_id the current node id. * @param string $indent */ private function displayNode(int $node_id, string $indent) { /** * @var \Sieve\SieveToken $node can be used to get info about a specific node. */ $node = $this->tree->getNode($node_id); // All the possible node types are listed as constants in the class SieveToken... switch ($node->type) { case \Sieve\SieveToken::SCRIPT_END: printf($indent . "EOS"); break; case Sieve\SieveToken::WHITESPACE: case Sieve\SieveToken::COMMENT: break; default: // the $node->type is a integer. It can be turned into an explicit string this way... $type = \Sieve\SieveToken::typeString($node->type); $open_color = ''; $end_color = ''; // The type of a node can be checked with the is method. Mask can be used to match several types. if ($node->is(\Sieve\SieveToken::QUOTED_STRING | Sieve\SieveToken::MULTILINE_STRING)) { // we want to put a specific color arround strings... $open_color = "\e[1;32;47m"; $end_color = "\e[0m"; } // The children of a node can be obtain through this method: $children = $this->tree->getChildren($node_id); // do whatever you want with a node and its children :) Here we are going to display them. printf("[%4d, %-10.10s (%5d) ]%s ${open_color}%s$end_color" . PHP_EOL, $node->line, $type, $node->type, $indent, $node->text); $this->displayNodeList($children, $indent . "\t"); } } } $sieve = file_get_contents(__DIR__ . '/tests/good/currentdate.siv'); $parser = new UsageExample($sieve); $parser->display();