| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240 |
- <?php // phpcs:disable WebimpressCodingStandard.NamingConventions.ValidVariableName.NotCamelCaps
- namespace Laminas\Mime;
- use Laminas\Mail\Headers;
- use Laminas\Stdlib\ErrorHandler;
- use function count;
- use function explode;
- use function iconv_mime_decode;
- use function preg_match;
- use function preg_match_all;
- use function preg_split;
- use function str_replace;
- use function strcasecmp;
- use function strlen;
- use function strpos;
- use function strtok;
- use function strtolower;
- use function substr;
- use const E_NOTICE;
- use const E_WARNING;
- use const ICONV_MIME_DECODE_CONTINUE_ON_ERROR;
- class Decode
- {
- /**
- * Explode MIME multipart string into separate parts
- *
- * Parts consist of the header and the body of each MIME part.
- *
- * @param string $body raw body of message
- * @param string $boundary boundary as found in content-type
- * @return array parts with content of each part, empty if no parts found
- * @throws Exception\RuntimeException
- */
- public static function splitMime($body, $boundary)
- {
- // TODO: we're ignoring \r for now - is this function fast enough and is it safe to assume noone needs \r?
- $body = str_replace("\r", '', $body);
- $start = 0;
- $res = [];
- // find every mime part limiter and cut out the
- // string before it.
- // the part before the first boundary string is discarded:
- $p = strpos($body, '--' . $boundary . "\n", $start);
- if ($p === false) {
- // no parts found!
- return [];
- }
- // position after first boundary line
- $start = $p + 3 + strlen($boundary);
- while (($p = strpos($body, '--' . $boundary . "\n", $start)) !== false) {
- $res[] = substr($body, $start, $p - $start);
- $start = $p + 3 + strlen($boundary);
- }
- // no more parts, find end boundary
- $p = strpos($body, '--' . $boundary . '--', $start);
- if ($p === false) {
- throw new Exception\RuntimeException('Not a valid Mime Message: End Missing');
- }
- // the remaining part also needs to be parsed:
- $res[] = substr($body, $start, $p - $start);
- return $res;
- }
- /**
- * decodes a mime encoded String and returns a
- * struct of parts with header and body
- *
- * @param string $message raw message content
- * @param string $boundary boundary as found in content-type
- * @param string $EOL EOL string; defaults to {@link Laminas\Mime\Mime::LINEEND}
- * @return array|null parts as array('header' => array(name => value), 'body' => content), null if no parts found
- * @throws Exception\RuntimeException
- */
- public static function splitMessageStruct($message, $boundary, $EOL = Mime::LINEEND)
- {
- $parts = static::splitMime($message, $boundary);
- if (! $parts) {
- return;
- }
- $result = [];
- $headers = null; // "Declare" variable before the first usage "for reading"
- $body = null; // "Declare" variable before the first usage "for reading"
- foreach ($parts as $part) {
- static::splitMessage($part, $headers, $body, $EOL);
- $result[] = [
- 'header' => $headers,
- 'body' => $body,
- ];
- }
- return $result;
- }
- /**
- * split a message in header and body part, if no header or an
- * invalid header is found $headers is empty
- *
- * The charset of the returned headers depend on your iconv settings.
- *
- * @param string|Headers $message raw message with header and optional content
- * @param Headers $headers output param, headers container
- * @param string $body output param, content of message
- * @param string $EOL EOL string; defaults to {@link Laminas\Mime\Mime::LINEEND}
- * @param bool $strict enable strict mode for parsing message
- * @return null
- */
- public static function splitMessage($message, &$headers, &$body, $EOL = Mime::LINEEND, $strict = false)
- {
- if ($message instanceof Headers) {
- $message = $message->toString();
- }
- // check for valid header at first line
- $firstlinePos = strpos($message, "\n");
- $firstline = $firstlinePos === false ? $message : substr($message, 0, $firstlinePos);
- if (! preg_match('%^[^\s]+[^:]*:%', $firstline)) {
- $headers = new Headers();
- // TODO: we're ignoring \r for now - is this function fast enough and is it safe to assume noone needs \r?
- $body = str_replace(["\r", "\n"], ['', $EOL], $message);
- return;
- }
- // see @Laminas-372, pops the first line off a message if it doesn't contain a header
- if (! $strict) {
- $parts = explode(':', $firstline, 2);
- if (count($parts) !== 2) {
- $message = substr($message, strpos($message, $EOL) + 1);
- }
- }
- // @todo splitMime removes "\r" sequences, which breaks valid mime
- // messages as returned by many mail servers
- $headersEOL = $EOL;
- // find an empty line between headers and body
- // default is set new line
- // @todo Maybe this is too much "magic"; we should be more strict here
- if (strpos($message, $EOL . $EOL)) {
- [$headers, $body] = explode($EOL . $EOL, $message, 2);
- // next is the standard new line
- } elseif ($EOL !== "\r\n" && strpos($message, "\r\n\r\n")) {
- [$headers, $body] = explode("\r\n\r\n", $message, 2);
- $headersEOL = "\r\n"; // Headers::fromString will fail with incorrect EOL
- // next is the other "standard" new line
- } elseif ($EOL !== "\n" && strpos($message, "\n\n")) {
- [$headers, $body] = explode("\n\n", $message, 2);
- $headersEOL = "\n";
- // at last resort find anything that looks like a new line
- } else {
- ErrorHandler::start(E_NOTICE | E_WARNING);
- [$headers, $body] = preg_split("%([\r\n]+)\\1%U", $message, 2);
- ErrorHandler::stop();
- }
- $headers = Headers::fromString($headers, $headersEOL);
- }
- /**
- * split a content type in its different parts
- *
- * @param string $type content-type
- * @param string $wantedPart the wanted part, else an array with all parts is returned
- * @return string|array wanted part or all parts as array('type' => content-type, partname => value)
- */
- public static function splitContentType($type, $wantedPart = null)
- {
- return static::splitHeaderField($type, $wantedPart, 'type');
- }
- /**
- * split a header field like content type in its different parts
- *
- * @param string $field header field
- * @param string $wantedPart the wanted part, else an array with all parts is returned
- * @param string $firstName key name for the first part
- * @return string|array wanted part or all parts as array($firstName => firstPart, partname => value)
- * @throws Exception\RuntimeException
- */
- public static function splitHeaderField($field, $wantedPart = null, $firstName = '0')
- {
- $wantedPart = strtolower($wantedPart ?? '');
- $firstName = strtolower($firstName);
- // special case - a bit optimized
- if ($firstName === $wantedPart) {
- $field = strtok($field, ';');
- return $field[0] === '"' ? substr($field, 1, -1) : $field;
- }
- $field = $firstName . '=' . $field;
- if (! preg_match_all('%([^=\s]+)\s*=\s*("[^"]+"|[^;]+)(;\s*|$)%', $field, $matches)) {
- throw new Exception\RuntimeException('not a valid header field');
- }
- if ($wantedPart) {
- foreach ($matches[1] as $key => $name) {
- if (strcasecmp($name, $wantedPart)) {
- continue;
- }
- if ($matches[2][$key][0] !== '"') {
- return $matches[2][$key];
- }
- return substr($matches[2][$key], 1, -1);
- }
- return;
- }
- $split = [];
- foreach ($matches[1] as $key => $name) {
- $name = strtolower($name);
- if ($matches[2][$key][0] === '"') {
- $split[$name] = substr($matches[2][$key], 1, -1);
- } else {
- $split[$name] = $matches[2][$key];
- }
- }
- return $split;
- }
- /**
- * decode a quoted printable encoded string
- *
- * The charset of the returned string depends on your iconv settings.
- *
- * @param string $string encoded string
- * @return string decoded string
- */
- public static function decodeQuotedPrintable($string)
- {
- return iconv_mime_decode($string, ICONV_MIME_DECODE_CONTINUE_ON_ERROR, 'UTF-8');
- }
- }
|