<?php
/**
* Copyright 2008-2017 Horde LLC (http://www.horde.org/)
*
* See the enclosed file LICENSE for license information (LGPL). If you
* did not receive this file, see http://www.horde.org/licenses/lgpl21.
*
* @category Horde
* @copyright 2008-2017 Horde LLC
* @license http://www.horde.org/licenses/lgpl21 LGPL 2.1
* @package Imap_Client
*/
/**
* Abstraction of the IMAP4rev1 search criteria (see RFC 3501 [6.4.4]).
* Allows translation between abstracted search criteria and a generated IMAP
* search criteria string suitable for sending to a remote IMAP server.
*
* @author Michael Slusarz <slusarz@horde.org>
* @category Horde
* @copyright 2008-2017 Horde LLC
* @license http://www.horde.org/licenses/lgpl21 LGPL 2.1
* @package Imap_Client
*/
class Horde_Imap_Client_Search_Query implements Serializable
{
/**
* Serialized version.
*/
const VERSION = 3;
/**
* Constants for dateSearch()
*/
const DATE_BEFORE = 'BEFORE';
const DATE_ON = 'ON';
const DATE_SINCE = 'SINCE';
/**
* Constants for intervalSearch()
*/
const INTERVAL_OLDER = 'OLDER';
const INTERVAL_YOUNGER = 'YOUNGER';
/**
* The charset of the search strings. All text strings must be in
* this charset. By default, this is 'US-ASCII' (see RFC 3501 [6.4.4]).
*
* @var string
*/
protected $_charset = null;
/**
* The list of search params.
*
* @var array
*/
protected $_search = array();
/**
* String representation: The IMAP search string.
*/
public function __toString()
{
try {
$res = $this->build(null);
return $res['query']->escape();
} catch (Exception $e) {
return '';
}
}
/**
* Sets the charset of the search text.
*
* @param string $charset The charset to use for the search.
* @param boolean $convert Convert existing text values?
*
* @throws Horde_Imap_Client_Exception_SearchCharset
*/
public function charset($charset, $convert = true)
{
$oldcharset = $this->_charset;
$this->_charset = Horde_String::upper($charset);
if (!$convert || ($oldcharset == $this->_charset)) {
return;
}
foreach (array('and', 'or') as $item) {
if (isset($this->_search[$item])) {
foreach ($this->_search[$item] as &$val) {
$val->charset($charset, $convert);
}
}
}
// Unset the reference to avoid corrupting $this->_search below.
unset($val);
foreach (array('header', 'text') as $item) {
if (isset($this->_search[$item])) {
foreach ($this->_search[$item] as $key => $val) {
$new_val = Horde_String::convertCharset($val['text'], $oldcharset, $this->_charset);
if (Horde_String::convertCharset($new_val, $this->_charset, $oldcharset) != $val['text']) {
throw new Horde_Imap_Client_Exception_SearchCharset($this->_charset);
}
$this->_search[$item][$key]['text'] = $new_val;
}
}
}
}
/**
* Builds an IMAP4rev1 compliant search string.
*
* @todo Change default of $exts to null.
*
* @param Horde_Imap_Client_Base $exts The server object this query will
* be run on (@since 2.24.0), a
* Horde_Imap_Client_Data_Capability
* object (@since 2.24.0), or the
* list of extensions present
* on the server (@deprecated).
* If null, all extensions are
* assumed to be available.
*
* @return array An array with these elements:
* - charset: (string) The charset of the search string. If null, no
* text strings appear in query.
* - exts: (array) The list of IMAP extensions used to create the
* string.
* - query: (Horde_Imap_Client_Data_Format_List) The IMAP search
* command.
*
* @throws Horde_Imap_Client_Data_Format_Exception
* @throws Horde_Imap_Client_Exception_NoSupportExtension
*/
public function build($exts = array())
{
/* @todo: BC */
if (is_array($exts)) {
$tmp = new Horde_Imap_Client_Data_Capability_Imap();
foreach ($exts as $key => $val) {
$tmp->add($key, is_array($val) ? $val : null);
}
$exts = $tmp;
} elseif (!is_null($exts)) {
if ($exts instanceof Horde_Imap_Client_Base) {
$exts = $exts->capability;
} elseif (!($exts instanceof Horde_Imap_Client_Data_Capability)) {
throw new InvalidArgumentException('Incorrect $exts parameter');
}
}
$temp = array(
'cmds' => new Horde_Imap_Client_Data_Format_List(),
'exts' => $exts,
'exts_used' => array()
);
$cmds = &$temp['cmds'];
$charset = $charset_cname = null;
$default_search = true;
$exts_used = &$temp['exts_used'];
$ptr = &$this->_search;
$charset_get = function ($c) use (&$charset, &$charset_cname) {
$charset = is_null($c)
? 'US-ASCII'
: strval($c);
$charset_cname = ($charset === 'US-ASCII')
? 'Horde_Imap_Client_Data_Format_Astring'
: 'Horde_Imap_Client_Data_Format_Astring_Nonascii';
};
$create_return = function ($charset, $exts_used, $cmds) {
return array(
'charset' => $charset,
'exts' => array_keys(array_flip($exts_used)),
'query' => $cmds
);
};
/* Do IDs check first. If there is an empty ID query (without a NOT
* qualifier), the rest of this query is irrelevant since we already
* know the search will return no results. */
if (isset($ptr['ids'])) {
if (!count($ptr['ids']['ids']) && !$ptr['ids']['ids']->special) {
if (empty($ptr['ids']['not'])) {
/* This is a match on an empty list of IDs. We do need to
* process any OR queries that may exist, since they are
* independent of this result. */
if (isset($ptr['or'])) {
$this->_buildAndOr(
'OR', $ptr['or'], $charset, $exts_used, $cmds
);
}
return $create_return($charset, $exts_used, $cmds);
}
/* If reached here, this a NOT search of an empty list. We can
* safely discard this from the output. */
} else {
$this->_addFuzzy(!empty($ptr['ids']['fuzzy']), $temp);
if (!empty($ptr['ids']['not'])) {
$cmds->add('NOT');
}
if (!$ptr['ids']['ids']->sequence) {
$cmds->add('UID');
}
$cmds->add(strval($ptr['ids']['ids']));
}
}
if (isset($ptr['new'])) {
$this->_addFuzzy(!empty($ptr['newfuzzy']), $temp);
if ($ptr['new']) {
$cmds->add('NEW');
unset($ptr['flag']['UNSEEN']);
} else {
$cmds->add('OLD');
}
unset($ptr['flag']['RECENT']);
}
if (!empty($ptr['flag'])) {
foreach ($ptr['flag'] as $key => $val) {
$this->_addFuzzy(!empty($val['fuzzy']), $temp);
$tmp = '';
if (empty($val['set'])) {
// This is a 'NOT' search. All system flags but \Recent
// have 'UN' equivalents.
if ($key == 'RECENT') {
$cmds->add('NOT');
} else {
$tmp = 'UN';
}
}
if ($val['type'] == 'keyword') {
$cmds->add(array(
$tmp . 'KEYWORD',
$key
));
} else {
$cmds->add($tmp . $key);
}
}
}
if (!empty($ptr['header'])) {
/* The list of 'system' headers that have a specific search
* query. */
$systemheaders = array(
'BCC', 'CC', 'FROM', 'SUBJECT', 'TO'
);
foreach ($ptr['header'] as $val) {
$this->_addFuzzy(!empty($val['fuzzy']), $temp);
if (!empty($val['not'])) {
$cmds->add('NOT');
}
if (in_array($val['header'], $systemheaders)) {
$cmds->add($val['header']);
} else {
$cmds->add(array(
'HEADER',
new Horde_Imap_Client_Data_Format_Astring($val['header'])
));
}
$charset_get($this->_charset);
$cmds->add(
new $charset_cname(isset($val['text']) ? $val['text'] : '')
);
}
}
if (!empty($ptr['text'])) {
foreach ($ptr['text'] as $val) {
$this->_addFuzzy(!empty($val['fuzzy']), $temp);
if (!empty($val['not'])) {
$cmds->add('NOT');
}
$charset_get($this->_charset);
$cmds->add(array(
$val['type'],
new $charset_cname($val['text'])
));
}
}
if (!empty($ptr['size'])) {
foreach ($ptr['size'] as $key => $val) {
$this->_addFuzzy(!empty($val['fuzzy']), $temp);
if (!empty($val['not'])) {
$cmds->add('NOT');
}
$cmds->add(array(
$key,
new Horde_Imap_Client_Data_Format_Number(
empty($val['size']) ? 0 : $val['size']
)
));
}
}
if (!empty($ptr['date'])) {
foreach ($ptr['date'] as $val) {
$this->_addFuzzy(!empty($val['fuzzy']), $temp);
if (!empty($val['not'])) {
$cmds->add('NOT');
}
if (empty($val['header'])) {
$cmds->add($val['range']);
} else {
$cmds->add('SENT' . $val['range']);
}
$cmds->add($val['date']);
}
}
if (!empty($ptr['within'])) {
if (is_null($exts) || $exts->query('WITHIN')) {
$exts_used[] = 'WITHIN';
}
foreach ($ptr['within'] as $key => $val) {
$this->_addFuzzy(!empty($val['fuzzy']), $temp);
if (!empty($val['not'])) {
$cmds->add('NOT');
}
if (is_null($exts) || $exts->query('WITHIN')) {
$cmds->add(array(
$key,
new Horde_Imap_Client_Data_Format_Number($val['interval'])
));
} else {
// This workaround is only accurate to within 1 day, due
// to limitations with the IMAP4rev1 search commands.
$cmds->add(array(
($key == self::INTERVAL_OLDER) ? self::DATE_BEFORE : self::DATE_SINCE,
new Horde_Imap_Client_Data_Format_Date('now -' . $val['interval'] . ' seconds')
));
}
}
}
if (!empty($ptr['modseq'])) {
if (!is_null($exts) && !$exts->query('CONDSTORE')) {
throw new Horde_Imap_Client_Exception_NoSupportExtension('CONDSTORE');
}
$exts_used[] = 'CONDSTORE';
$this->_addFuzzy(!empty($ptr['modseq']['fuzzy']), $temp);
if (!empty($ptr['modseq']['not'])) {
$cmds->add('NOT');
}
$cmds->add('MODSEQ');
if (isset($ptr['modseq']['name'])) {
$cmds->add(array(
new Horde_Imap_Client_Data_Format_String($ptr['modseq']['name']),
$ptr['modseq']['type']
));
}
$cmds->add(new Horde_Imap_Client_Data_Format_Number($ptr['modseq']['value']));
}
if (isset($ptr['prevsearch'])) {
if (!is_null($exts) && !$exts->query('SEARCHRES')) {
throw new Horde_Imap_Client_Exception_NoSupportExtension('SEARCHRES');
}
$exts_used[] = 'SEARCHRES';
$this->_addFuzzy(!empty($ptr['prevsearchfuzzy']), $temp);
if (!$ptr['prevsearch']) {
$cmds->add('NOT');
}
$cmds->add('$');
}
// Add AND'ed queries
if (!empty($ptr['and'])) {
$default_search = $this->_buildAndOr(
'AND', $ptr['and'], $charset, $exts_used, $cmds
);
}
// Add OR'ed queries
if (!empty($ptr['or'])) {
$default_search = $this->_buildAndOr(
'OR', $ptr['or'], $charset, $exts_used, $cmds
);
}
// Default search is 'ALL'
if ($default_search && !count($cmds)) {
$cmds->add('ALL');
}
return $create_return($charset, $exts_used, $cmds);
}
/**
* Builds the AND/OR query.
*
* @param string $type 'AND' or 'OR'.
* @param array $data Query data.
* @param string &$charset Search charset.
* @param array &$exts_used IMAP extensions used.
* @param Horde_Imap_Client_Data_Format_List &$cmds Command list.
*
* @return boolean True if query might return results.
*/
protected function _buildAndOr($type, $data, &$charset, &$exts_used,
&$cmds)
{
$results = false;
foreach ($data as $val) {
$ret = $val->build();
/* Empty sub-query. */
if (!count($ret['query'])) {
switch ($type) {
case 'AND':
/* Any empty sub-query means that the query MUST return
* no results. */
$cmds = new Horde_Imap_Client_Data_Format_List();
$exts_used = array();
return false;
case 'OR':
/* Skip this query. */
continue 2;
}
}
$results = true;
if (!is_null($ret['charset']) && ($ret['charset'] != 'US-ASCII')) {
if (!is_null($charset) &&
($charset != 'US-ASCII') &&
($charset != $ret['charset'])) {
throw new InvalidArgumentException(
'AND/OR queries must all have the same charset.'
);
}
$charset = $ret['charset'];
}
$exts_used = array_merge($exts_used, $ret['exts']);
switch ($type) {
case 'AND':
$cmds->add($ret['query'], true);
break;
case 'OR':
// First OR'd query
if (count($cmds)) {
$new_cmds = new Horde_Imap_Client_Data_Format_List();
$new_cmds->add(array(
'OR',
$ret['query'],
$cmds
));
$cmds = $new_cmds;
} else {
$cmds = $ret['query'];
}
break;
}
}
return $results;
}
/**
* Adds fuzzy modifier to search keys.
*
* @param boolean $add Add the fuzzy modifier?
* @param array $temp Temporary build data.
*
* @throws Horde_Imap_Client_Exception_NoSupport_Extension
*/
protected function _addFuzzy($add, &$temp)
{
if ($add) {
if (!$temp['exts']->query('SEARCH', 'FUZZY')) {
throw new Horde_Imap_Client_Exception_NoSupportExtension('SEARCH=FUZZY');
}
$temp['cmds']->add('FUZZY');
$temp['exts_used'][] = 'SEARCH=FUZZY';
}
}
/**
* Search for a flag/keywords.
*
* @param string $name The flag or keyword name.
* @param boolean $set If true, search for messages that have the flag
* set. If false, search for messages that do not
* have the flag set.
* @param array $opts Additional options:
* - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server
* MUST support RFC 6203.
*/
public function flag($name, $set = true, array $opts = array())
{
$name = Horde_String::upper(ltrim($name, '\\'));
if (!isset($this->_search['flag'])) {
$this->_search['flag'] = array();
}
/* The list of defined system flags (see RFC 3501 [2.3.2]). */
$systemflags = array(
'ANSWERED', 'DELETED', 'DRAFT', 'FLAGGED', 'RECENT', 'SEEN'
);
$this->_search['flag'][$name] = array_filter(array(
'fuzzy' => !empty($opts['fuzzy']),
'set' => $set,
'type' => in_array($name, $systemflags) ? 'flag' : 'keyword'
));
}
/**
* Determines if flags are a part of the search.
*
* @return boolean True if search query involves flags.
*/
public function flagSearch()
{
return !empty($this->_search['flag']);
}
/**
* Search for either new messages (messages that have the '\Recent' flag
* but not the '\Seen' flag) or old messages (messages that do not have
* the '\Recent' flag). If new messages are searched, this will clear
* any '\Recent' or '\Unseen' flag searches. If old messages are searched,
* this will clear any '\Recent' flag search.
*
* @param boolean $newmsgs If true, searches for new messages. Else,
* search for old messages.
* @param array $opts Additional options:
* - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server
* MUST support RFC 6203.
*/
public function newMsgs($newmsgs = true, array $opts = array())
{
$this->_search['new'] = $newmsgs;
if (!empty($opts['fuzzy'])) {
$this->_search['newfuzzy'] = true;
}
}
/**
* Search for text in the header of a message.
*
* @param string $header The header field.
* @param string $text The search text.
* @param boolean $not If true, do a 'NOT' search of $text.
* @param array $opts Additional options:
* - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server
* MUST support RFC 6203.
*/
public function headerText($header, $text, $not = false,
array $opts = array())
{
if (!isset($this->_search['header'])) {
$this->_search['header'] = array();
}
$this->_search['header'][] = array_filter(array(
'fuzzy' => !empty($opts['fuzzy']),
'header' => Horde_String::upper($header),
'text' => $text,
'not' => $not
));
}
/**
* Search for text in either the entire message, or just the body.
*
* @param string $text The search text.
< * @param string $bodyonly If true, only search in the body of the
> * @param boolean $bodyonly If true, only search in the body of the
* message. If false, also search in the headers.
* @param boolean $not If true, do a 'NOT' search of $text.
* @param array $opts Additional options:
* - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server
* MUST support RFC 6203.
*/
public function text($text, $bodyonly = true, $not = false,
array $opts = array())
{
if (!isset($this->_search['text'])) {
$this->_search['text'] = array();
}
$this->_search['text'][] = array_filter(array(
'fuzzy' => !empty($opts['fuzzy']),
'not' => $not,
'text' => $text,
'type' => $bodyonly ? 'BODY' : 'TEXT'
));
}
/**
* Search for messages smaller/larger than a certain size.
*
* @todo: Remove $not for 3.0
*
* @param integer $size The size (in bytes).
* @param boolean $larger Search for messages larger than $size?
* @param boolean $not If true, do a 'NOT' search of $text.
* @param array $opts Additional options:
* - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server
* MUST support RFC 6203.
*/
public function size($size, $larger = false, $not = false,
array $opts = array())
{
if (!isset($this->_search['size'])) {
$this->_search['size'] = array();
}
$this->_search['size'][$larger ? 'LARGER' : 'SMALLER'] = array_filter(array(
'fuzzy' => !empty($opts['fuzzy']),
'not' => $not,
'size' => (float)$size
));
}
/**
* Search for messages within a given UID range. Only one message range
* can be specified per query.
*
* @param Horde_Imap_Client_Ids $ids The list of UIDs to search.
* @param boolean $not If true, do a 'NOT' search of the
* UIDs.
* @param array $opts Additional options:
* - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server
* MUST support RFC 6203.
*/
public function ids(Horde_Imap_Client_Ids $ids, $not = false,
array $opts = array())
{
$this->_search['ids'] = array_filter(array(
'fuzzy' => !empty($opts['fuzzy']),
'ids' => $ids,
'not' => $not
));
}
/**
* Search for messages within a date range.
*
* @param mixed $date DateTime or Horde_Date object.
* @param string $range Either:
* - Horde_Imap_Client_Search_Query::DATE_BEFORE
* - Horde_Imap_Client_Search_Query::DATE_ON
* - Horde_Imap_Client_Search_Query::DATE_SINCE
* @param boolean $header If true, search using the date in the message
* headers. If false, search using the internal
* IMAP date (usually arrival time).
* @param boolean $not If true, do a 'NOT' search of the range.
* @param array $opts Additional options:
* - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server
* MUST support RFC 6203.
*/
public function dateSearch($date, $range, $header = true, $not = false,
array $opts = array())
{
if (!isset($this->_search['date'])) {
$this->_search['date'] = array();
}
// We should really be storing the raw DateTime object as data,
// but all versions of the query object have converted at this stage.
$ob = new Horde_Imap_Client_Data_Format_Date($date);
$this->_search['date'][] = array_filter(array(
'date' => $ob->escape(),
'fuzzy' => !empty($opts['fuzzy']),
'header' => $header,
'range' => $range,
'not' => $not
));
}
/**
* Search for messages within a date and time range.
*
* @param mixed $date DateTime or Horde_Date object.
* @param string $range Either:
* - Horde_Imap_Client_Search_Query::DATE_BEFORE
* - Horde_Imap_Client_Search_Query::DATE_ON
* - Horde_Imap_Client_Search_Query::DATE_SINCE
* @param boolean $header If true, search using the date in the message
* headers. If false, search using the internal
* IMAP date (usually arrival time).
* @param boolean $not If true, do a 'NOT' search of the range.
* @param array $opts Additional options:
* - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server
* MUST support RFC 6203.
*/
public function dateTimeSearch($date, $range, $header = true, $not = false,
array $opts = array())
{
if (!isset($this->_search['date'])) {
$this->_search['date'] = array();
}
// We should really be storing the raw DateTime object as data,
// but all versions of the query object have converted at this stage.
$ob = new Horde_Imap_Client_Data_Format_DateTime($date);
$this->_search['date'][] = array_filter(array(
'date' => $ob->escape(),
'fuzzy' => !empty($opts['fuzzy']),
'header' => $header,
'range' => $range,
'not' => $not
));
}
/**
* Search for messages within a given interval. Only one interval of each
* type can be specified per search query. If the IMAP server supports
* the WITHIN extension (RFC 5032), it will be used. Otherwise, the
* search query will be dynamically created using IMAP4rev1 search
* terms.
*
* @param integer $interval Seconds from the present.
* @param string $range Either:
* - Horde_Imap_Client_Search_Query::INTERVAL_OLDER
* - Horde_Imap_Client_Search_Query::INTERVAL_YOUNGER
* @param boolean $not If true, do a 'NOT' search.
* @param array $opts Additional options:
* - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server
* MUST support RFC 6203.
*/
public function intervalSearch($interval, $range, $not = false,
array $opts = array())
{
if (!isset($this->_search['within'])) {
$this->_search['within'] = array();
}
$this->_search['within'][$range] = array(
'fuzzy' => !empty($opts['fuzzy']),
'interval' => $interval,
'not' => $not
);
}
/**
* AND queries - the contents of this query will be AND'ed (in its
* entirety) with the contents of EACH of the queries passed in. All
* AND'd queries must share the same charset as this query.
*
* @param mixed $queries A query, or an array of queries, to AND with the
* current query.
*/
public function andSearch($queries)
{
if (!isset($this->_search['and'])) {
$this->_search['and'] = array();
}
if ($queries instanceof Horde_Imap_Client_Search_Query) {
$queries = array($queries);
}
$this->_search['and'] = array_merge($this->_search['and'], $queries);
}
/**
* OR a query - the contents of this query will be OR'ed (in its entirety)
* with the contents of EACH of the queries passed in. All OR'd queries
* must share the same charset as this query. All contents of any single
* query will be AND'ed together.
*
* @param mixed $queries A query, or an array of queries, to OR with the
* current query.
*/
public function orSearch($queries)
{
if (!isset($this->_search['or'])) {
$this->_search['or'] = array();
}
if ($queries instanceof Horde_Imap_Client_Search_Query) {
$queries = array($queries);
}
$this->_search['or'] = array_merge($this->_search['or'], $queries);
}
/**
* Search for messages modified since a specific moment. The IMAP server
* must support the CONDSTORE extension (RFC 7162) for this query to be
* used.
*
* @param integer $value The mod-sequence value.
* @param string $name The entry-name string.
* @param string $type Either 'shared', 'priv', or 'all'. Defaults to
* 'all'
* @param boolean $not If true, do a 'NOT' search.
* @param array $opts Additional options:
* - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server
* MUST support RFC 6203.
*/
public function modseq($value, $name = null, $type = null, $not = false,
array $opts = array())
{
if (!is_null($type)) {
$type = Horde_String::lower($type);
if (!in_array($type, array('shared', 'priv', 'all'))) {
$type = 'all';
}
}
$this->_search['modseq'] = array_filter(array(
'fuzzy' => !empty($opts['fuzzy']),
'name' => $name,
'not' => $not,
'type' => (!is_null($name) && is_null($type)) ? 'all' : $type,
'value' => $value
));
}
/**
* Use the results from the previous SEARCH command. The IMAP server must
* support the SEARCHRES extension (RFC 5182) for this query to be used.
*
* @param boolean $not If true, don't match the previous query.
* @param array $opts Additional options:
* - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server
* MUST support RFC 6203.
*/
public function previousSearch($not = false, array $opts = array())
{
$this->_search['prevsearch'] = $not;
if (!empty($opts['fuzzy'])) {
$this->_search['prevsearchfuzzy'] = true;
}
}
/* Serializable methods. */
/**
* Serialization.
*
* @return string Serialized data.
*/
public function serialize()
{
$data = array(
// Serialized data ID.
self::VERSION,
$this->_search
);
if (!is_null($this->_charset)) {
$data[] = $this->_charset;
}
return serialize($data);
}
/**
* Unserialization.
*
* @param string $data Serialized data.
*
* @throws Exception
*/
public function unserialize($data)
{
$data = @unserialize($data);
if (!is_array($data) ||
!isset($data[0]) ||
($data[0] != self::VERSION)) {
throw new Exception('Cache version change');
}
$this->_search = $data[1];
if (isset($data[2])) {
$this->_charset = $data[2];
}
}
}