#!/usr/bin/php * @author Greg Beaver * @link http://www.synapticmedia.net Synaptic Media * @version $Id: Archive.php,v 1.38 2007/02/06 04:31:45 cellog Exp $ * @package PHP_Archive * @category PHP */ class PHP_Archive { const GZ = 0x00001000; const BZ2 = 0x00002000; const SIG = 0x00010000; const SHA1 = 0x0002; const MD5 = 0x0001; /** * Whether this archive is compressed with zlib * * @var bool */ private $_compressed; /** * @var string Real path to the .phar archive */ private $_archiveName = null; /** * Current file name in the phar * @var string */ protected $currentFilename = null; /** * Length of current file in the phar * @var string */ protected $internalFileLength = null; /** * Current file statistics (size, creation date, etc.) * @var string */ protected $currentStat = null; /** * @var resource|null Pointer to open .phar */ protected $fp = null; /** * @var int Current Position of the pointer */ protected $position = 0; /** * Map actual realpath of phars to meta-data about the phar * * Data is indexed by the alias that is used by internal files. In other * words, if a file is included via: * * require_once 'phar://PEAR.phar/PEAR/Installer.php'; * * then the alias is "PEAR.phar" * * Information stored is a boolean indicating whether this .phar is compressed * with zlib, another for bzip2, phar-specific meta-data, and * the precise offset of internal files * within the .phar, used with the {@link $_manifest} to load actual file contents * @var array */ private static $_pharMapping = array(); /** * Map real file paths to alias used * * @var array */ private static $_pharFiles = array(); /** * File listing for the .phar * * The manifest is indexed per phar. * * Files within the .phar are indexed by their relative path within the * .phar. Each file has this information in its internal array * * - 0 = uncompressed file size * - 1 = timestamp of when file was added to phar * - 2 = offset of file within phar relative to internal file's start * - 3 = compressed file size (actual size in the phar) * @var array */ private static $_manifest = array(); /** * Absolute offset of internal files within the .phar, indexed by absolute * path to the .phar * * @var array */ private static $_fileStart = array(); /** * file name of the phar * * @var string */ private $_basename; /** * Allows loading an external Phar archive without include()ing it * * @param string $file * @throws Exception */ public static final function loadPhar($file) { $file = realpath($file); if ($file) { $fp = fopen($file, 'rb'); $buffer = ''; while (!$found && !feof($fp)) { $buffer .= fread($fp, 8192); // don't break phars if ($pos = strpos($buffer, '__HALT_COMPI' . 'LER();')) { fclose($fp); return self::_mapPhar($file, $pos); } } fclose($fp); } } /** * Map a full real file path to an alias used to refer to the .phar * * This function can only be called from the initialization of the .phar itself. * Any attempt to call from outside the .phar or to re-alias the .phar will fail * as a security measure. * @param string $file full realpath() filepath, like /path/to/go-pear.phar * @param string $alias alias used in opening a file within the phar * like phar://go-pear.phar/file * @param bool $compressed determines whether to attempt zlib uncompression * on accessing internal files * @param int $dataoffset the value of __COMPILER_HALT_OFFSET__ */ public static final function mapPhar($file, $dataoffset) { try { // this ensures that this is safe if (!in_array($file, get_included_files())) { die('SECURITY ERROR: PHP_Archive::mapPhar can only be called from within ' . 'the phar that initiates it'); } self::_mapPhar($file, $dataoffset); } catch (Exception $e) { die($e->getMessage()); } } /** * Sub-function, allows recovery from errors * * @param unknown_type $file * @param unknown_type $dataoffset */ private static function _mapPhar($file, $dataoffset) { $file = realpath($file); if (isset(self::$_manifest[$file])) { return; } if (!is_array(self::$_pharMapping)) { self::$_pharMapping = array(); } $fp = fopen($file, 'rb'); // seek to __HALT_COMPILER_OFFSET__ fseek($fp, $dataoffset); $manifest_length = unpack('Vlen', fread($fp, 4)); $manifest = ''; $last = '1'; while (strlen($last) && strlen($manifest) < $manifest_length['len']) { $read = 8192; if ($manifest_length['len'] - strlen($manifest) < 8192) { $read = $manifest_length['len'] - strlen($manifest); } $last = fread($fp, $read); $manifest .= $last; } if (strlen($manifest) < $manifest_length['len']) { throw new Exception('ERROR: manifest length read was "' . strlen($manifest) .'" should be "' . $manifest_length['len'] . '"'); } $info = self::_unserializeManifest($manifest); if ($info['alias']) { $alias = $info['alias']; $explicit = true; } else { $alias = $file; $explicit = false; } self::$_manifest[$file] = $info['manifest']; $compressed = $info['compressed']; self::$_fileStart[$file] = ftell($fp); fclose($fp); if ($compressed & 0x00001000) { if (!function_exists('gzinflate')) { throw new Exception('Error: zlib extension is not enabled - gzinflate() function needed' . ' for compressed .phars'); } } if ($compressed & 0x00002000) { if (!function_exists('bzdecompress')) { throw new Exception('Error: bzip2 extension is not enabled - bzdecompress() function needed' . ' for compressed .phars'); } } if (isset(self::$_pharMapping[$alias])) { throw new Exception('ERROR: PHP_Archive::mapPhar has already been called for alias "' . $alias . '" cannot re-alias to "' . $file . '"'); } self::$_pharMapping[$alias] = array($file, $compressed, $dataoffset, $explicit, $info['metadata']); self::$_pharFiles[$file] = $alias; } /** * extract the manifest into an internal array * * @param string $manifest * @return false|array */ private static function _unserializeManifest($manifest) { // retrieve the number of files in the manifest $info = unpack('V', substr($manifest, 0, 4)); $apiver = substr($manifest, 4, 2); $apiver = bin2hex($apiver); $apiver_dots = hexdec($apiver[0]) . '.' . hexdec($apiver[1]) . '.' . hexdec($apiver[2]); $majorcompat = hexdec($apiver[0]); $calcapi = explode('.', '1.0.0'); if ($calcapi[0] != $majorcompat) { throw new Exception('Phar is incompatible API version ' . $apiver_dots . ', but ' . 'PHP_Archive is API version 1.0.0'); } if ($calcapi[0] === '0') { if ('1.0.0' != $apiver_dots) { throw new Exception('Phar is API version ' . $apiver_dots . ', but PHP_Archive is API version 1.0.0', E_USER_ERROR); } } $flags = unpack('V', substr($manifest, 6, 4)); $ret = array('compressed' => $flags & 0x00003000); // signature is not verified by default in PHP_Archive, phar is better $ret['hassignature'] = $flags & 0x00010000; $aliaslen = unpack('V', substr($manifest, 10, 4)); if ($aliaslen) { $ret['alias'] = substr($manifest, 14, $aliaslen[1]); } else { $ret['alias'] = false; } $manifest = substr($manifest, 14 + $aliaslen[1]); $metadatalen = unpack('V', substr($manifest, 0, 4)); if ($metadatalen[1]) { $ret['metadata'] = unserialize(substr($manifest, 4, $metadatalen[1])); $manifest = substr($manifest, 4 + $metadatalen[1]); } else { $ret['metadata'] = null; $manifest = substr($manifest, 4); } $offset = 0; $start = 0; for ($i = 0; $i < $info[1]; $i++) { // length of the file name $len = unpack('V', substr($manifest, $start, 4)); $start += 4; // file name $savepath = substr($manifest, $start, $len[1]); $start += $len[1]; // retrieve manifest data: // 0 = uncompressed file size // 1 = timestamp of when file was added to phar // 2 = compressed filesize // 3 = crc32 // 4 = flags // 5 = metadata length $ret['manifest'][$savepath] = array_values(unpack('Va/Vb/Vc/Vd/Ve/Vf', substr($manifest, $start, 24))); $ret['manifest'][$savepath][3] = sprintf('%u', $ret['manifest'][$savepath][3]); if ($ret['manifest'][$savepath][5]) { $ret['manifest'][$savepath][6] = unserialize(substr($manifest, $start + 24, $ret['manifest'][$savepath][5])); } else { $ret['manifest'][$savepath][6] = null; } $ret['manifest'][$savepath][7] = $offset; $offset += $ret['manifest'][$savepath][2]; $start += 24 + $ret['manifest'][$savepath][5]; } return $ret; } /** * @param string */ private static function processFile($path) { if ($path == '.') { return ''; } $std = str_replace("\\", "/", $path); while ($std != ($std = ereg_replace("[^\/:?]+/\.\./", "", $std))) ; $std = str_replace("/./", "", $std); if (strlen($std) > 1 && $std[0] == '/') { $std = substr($std, 1); } if (strncmp($std, "./", 2) == 0) { return substr($std, 2); } else { return $std; } } /** * Seek in the master archive to a matching file or directory * @param string */ protected function selectFile($path, $allowdirs = true) { $std = self::processFile($path); if (isset(self::$_manifest[$this->_archiveName][$path])) { $this->_setCurrentFile($path); return true; } if (!$allowdirs) { return 'Error: "' . $path . '" is not a file in phar "' . $this->_basename . '"'; } foreach (self::$_manifest[$this->_archiveName] as $file => $info) { if (empty($std) || //$std is a directory strncmp($std.'/', $path, strlen($std)+1) == 0) { $this->currentFilename = $this->internalFileLength = $this->currentStat = null; return true; } } return 'Error: "' . $path . '" not found in phar "' . $this->_basename . '"'; } private function _setCurrentFile($path) { $this->currentStat = array( 2 => 0100444, // file mode, readable by all, writeable by none 4 => 0, // uid 5 => 0, // gid 7 => self::$_manifest[$this->_archiveName][$path][0], // size 9 => self::$_manifest[$this->_archiveName][$path][1], // creation time ); $this->currentFilename = $path; $this->internalFileLength = self::$_manifest[$this->_archiveName][$path][2]; // seek to offset of file header within the .phar if (is_resource(@$this->fp)) { fseek($this->fp, self::$_fileStart[$this->_archiveName] + self::$_manifest[$this->_archiveName][$path][7]); } } /** * Seek to a file within the master archive, and extract its contents * @param string * @return array|string an array containing an error message string is returned * upon error, otherwise the file contents are returned */ public function extractFile($path) { $this->fp = @fopen($this->_archiveName, "rb"); if (!$this->fp) { return array('Error: cannot open phar "' . $this->_archiveName . '"'); } if (($e = $this->selectFile($path, false)) === true) { $data = ''; $count = $this->internalFileLength; while ($count) { if ($count < 8192) { $data .= @fread($this->fp, $count); $count = 0; } else { $count -= 8192; $data .= @fread($this->fp, 8192); } } @fclose($this->fp); if (self::$_manifest[$this->_archiveName][$path][4] & self::GZ) { $data = gzinflate($data); } elseif (self::$_manifest[$this->_archiveName][$path][4] & self::BZ2) { $data = bzdecompress($data); } if (!isset(self::$_manifest[$this->_archiveName][$path]['ok'])) { if (strlen($data) != $this->currentStat[7]) { return array("Not valid internal .phar file (size error {$size} != " . $this->currentStat[7] . ")"); } if (self::$_manifest[$this->_archiveName][$path][3] != sprintf("%u", crc32($data))) { return array("Not valid internal .phar file (checksum error)"); } self::$_manifest[$this->_archiveName][$path]['ok'] = true; } return $data; } else { @fclose($this->fp); return array($e); } } /** * Parse urls like phar:///fullpath/to/my.phar/file.txt * * @param string $file * @return false|array */ static protected function parseUrl($file) { if (substr($file, 0, 7) != 'phar://') { return false; } $file = substr($file, 7); $ret = array('scheme' => 'phar'); $pos_p = strpos($file, '.phar.php'); $pos_z = strpos($file, '.phar.gz'); $pos_b = strpos($file, '.phar.bz2'); if ($pos_p) { if ($pos_z) { return false; } $ret['host'] = substr($file, 0, $pos_p + strlen('.phar.php')); $ret['path'] = substr($file, strlen($ret['host'])); } elseif ($pos_z) { $ret['host'] = substr($file, 0, $pos_z + strlen('.phar.gz')); $ret['path'] = substr($file, strlen($ret['host'])); } elseif ($pos_b) { $ret['host'] = substr($file, 0, $pos_z + strlen('.phar.bz2')); $ret['path'] = substr($file, strlen($ret['host'])); } elseif (($pos_p = strpos($file, ".phar")) !== false) { $ret['host'] = substr($file, 0, $pos_p + strlen('.phar')); $ret['path'] = substr($file, strlen($ret['host'])); } else { return false; } if (!$ret['path']) { $ret['path'] = '/'; } return $ret; } /** * Locate the .phar archive in the include_path and detect the file to open within * the archive. * * Possible parameters are phar://pharname.phar/filename_within_phar.ext * @param string a file within the archive * @return string the filename within the .phar to retrieve */ public function initializeStream($file) { $info = @parse_url($file); if (!$info) { $info = self::parseUrl($file); } if (!$info) { return false; } if (!isset($info['host'])) { // malformed internal file return false; } if (!isset(self::$_pharFiles[$info['host']]) && !isset(self::$_pharMapping[$info['host']])) { try { self::loadPhar($info['host']); // use alias from here out $info['host'] = self::$_pharFiles[$info['host']]; } catch (Exception $e) { return false; } } if (!isset($info['path'])) { return false; } elseif (strlen($info['path']) > 1) { $info['path'] = substr($info['path'], 1); } if (isset(self::$_pharMapping[$info['host']])) { $this->_basename = $info['host']; $this->_archiveName = self::$_pharMapping[$info['host']][0]; $this->_compressed = self::$_pharMapping[$info['host']][1]; } elseif (isset(self::$_pharFiles[$info['host']])) { $this->_archiveName = $info['host']; $this->_basename = self::$_pharFiles[$info['host']]; $this->_compressed = self::$_pharMapping[$this->_basename][1]; } $file = $info['path']; return $file; } /** * Open the requested file - PHP streams API * * @param string $file String provided by the Stream wrapper * @access private */ public function stream_open($file) { return $this->_streamOpen($file); } /** * @param string filename to opne, or directory name * @param bool if true, a directory will be matched, otherwise only files * will be matched * @uses trigger_error() * @return bool success of opening * @access private */ private function _streamOpen($file, $searchForDir = false) { $path = $this->initializeStream($file); if (!$path) { trigger_error('Error: Unknown phar in "' . $file . '"', E_USER_ERROR); } if (is_array($this->file = $this->extractFile($path))) { trigger_error($this->file[0], E_USER_ERROR); return false; } if ($path != $this->currentFilename) { if (!$searchForDir) { trigger_error("Cannot open '$file', is a directory", E_USER_ERROR); return false; } else { $this->file = ''; return true; } } if (!is_null($this->file) && $this->file !== false) { return true; } else { return false; } } /** * Read the data - PHP streams API * * @param int * @access private */ public function stream_read($count) { $ret = substr($this->file, $this->position, $count); $this->position += strlen($ret); return $ret; } /** * Whether we've hit the end of the file - PHP streams API * @access private */ function stream_eof() { return $this->position >= $this->currentStat[7]; } /** * For seeking the stream - PHP streams API * @param int * @param SEEK_SET|SEEK_CUR|SEEK_END * @access private */ public function stream_seek($pos, $whence) { switch ($whence) { case SEEK_SET: if ($pos < 0) { return false; } $this->position = $pos; break; case SEEK_CUR: if ($pos + $this->currentStat[7] < 0) { return false; } $this->position += $pos; break; case SEEK_END: if ($pos + $this->currentStat[7] < 0) { return false; } $this->position = $pos + $this->currentStat[7]; default: return false; } return true; } /** * The current position in the stream - PHP streams API * @access private */ public function stream_tell() { return $this->position; } /** * The result of an fstat call, returns mod time from creation, and file size - * PHP streams API * @uses _stream_stat() * @access private */ public function stream_stat() { return $this->_stream_stat(); } /** * Retrieve statistics on a file or directory within the .phar * @param string file/directory to stat * @access private */ public function _stream_stat($file = null) { $std = $file ? self::processFile($file) : $this->currentFilename; if ($file) { if (isset(self::$_manifest[$this->_archiveName][$file])) { $this->_setCurrentFile($file); $isdir = false; } else { $isdir = true; } } else { $isdir = false; // open streams must be files } $mode = $isdir ? 0040444 : 0100444; // 040000 = dir, 010000 = file // everything is readable, nothing is writeable return array( 0, 0, $mode, 0, 0, 0, 0, 0, 0, 0, 0, 0, // non-associative indices 'dev' => 0, 'ino' => 0, 'mode' => $mode, 'nlink' => 0, 'uid' => 0, 'gid' => 0, 'rdev' => 0, 'blksize' => 0, 'blocks' => 0, 'size' => $this->currentStat[7], 'atime' => $this->currentStat[9], 'mtime' => $this->currentStat[9], 'ctime' => $this->currentStat[9], ); } /** * Stat a closed file or directory - PHP streams API * @param string * @param int * @access private */ public function url_stat($url, $flags) { $path = $this->initializeStream($url); return $this->_stream_stat($path); } /** * Open a directory in the .phar for reading - PHP streams API * @param string directory name * @access private */ public function dir_opendir($path) { $info = @parse_url($path); if (!$info) { $info = self::parseUrl($path); if (!$info) { trigger_error('Error: "' . $path . '" is a file, and cannot be opened with opendir', E_USER_ERROR); return false; } } $path = !empty($info['path']) ? $info['host'] . $info['path'] : $info['host'] . '/'; $path = $this->initializeStream('phar://' . $path); if (isset(self::$_manifest[$this->_archiveName][$path])) { trigger_error('Error: "' . $path . '" is a file, and cannot be opened with opendir', E_USER_ERROR); return false; } if ($path == false) { trigger_error('Error: Unknown phar in "' . $file . '"', E_USER_ERROR); return false; } $this->fp = @fopen($this->_archiveName, "rb"); if (!$this->fp) { trigger_error('Error: cannot open phar "' . $this->_archiveName . '"'); return false; } $this->_dirFiles = array(); foreach (self::$_manifest[$this->_archiveName] as $file => $info) { if ($path == '/') { if (strpos($file, '/')) { $a = explode('/', $file); $this->_dirFiles[array_shift($a)] = true; } else { $this->_dirFiles[$file] = true; } } elseif (strpos($file, $path) === 0) { $fname = substr($file, strlen($path) + 1); if (strpos($fname, '/')) { $a = explode('/', $fname); $this->_dirFiles[array_shift($a)] = true; } elseif (strlen($file) != strlen($path)) { // if the two match exactly, the path searched for was // not a directory, but was a file. $this->_dirFiles[$fname] = true; } } } @fclose($this->fp); @uksort($this->_dirFiles, 'strnatcmp'); return true; } /** * Read the next directory entry - PHP streams API * @access private */ public function dir_readdir() { $ret = key($this->_dirFiles); @next($this->_dirFiles); if (!$ret) { return false; } return $ret; } /** * Close a directory handle opened with opendir() - PHP streams API * @access private */ public function dir_closedir() { $this->_dirFiles = array(); reset($this->_dirFiles); return true; } /** * Rewind to the first directory entry - PHP streams API * @access private */ public function dir_rewinddir() { reset($this->_dirFiles); return true; } /** * API version of this class * @return string */ public final function APIVersion() { return '1.0.0'; } /** * Retrieve Phar-specific metadata for a Phar archive * * @param string $phar full path to Phar archive, or alias * @return null|mixed The value that was serialized for the Phar * archive's metadata * @throws Exception */ public static function getPharMetadata($phar) { if (isset(self::$_pharFiles[$phar])) { $phar = self::$_pharFiles[$phar]; } if (!isset(self::$_pharMapping[$phar])) { throw new Exception('Unknown Phar archive: "' . $phar . '"'); } return self::$_pharMapping[$phar][4]; } /** * Retrieve File-specific metadata for a Phar archive file * * @param string $phar full path to Phar archive, or alias * @param string $file relative path to file within Phar archive * @return null|mixed The value that was serialized for the Phar * archive's metadata * @throws Exception */ public static function getFileMetadata($phar, $file) { if (!isset(self::$_pharFiles[$phar])) { if (!isset(self::$_pharMapping[$phar])) { throw new Exception('Unknown Phar archive: "' . $phar . '"'); } $phar = self::$_pharMapping[$phar][0]; } if (!isset(self::$_manifest[$phar])) { throw new Exception('Unknown Phar: "' . $phar . '"'); } $file = self::processFile($file); if (!isset(self::$_manifest[$phar][$file])) { throw new Exception('Unknown file "' . $file . '" within Phar "'. $phar . '"'); } return self::$_manifest[$phar][$file][6]; } } ?> 6 pharcommandclicommand.inc?,AH -zFdirectorygraphiterator.incAHurdirectorytreeiterator.inc%AH]pinvertedregexiterator.incAHICCpharcommand.incAH"f phar.incAH)k<koFZЕT,ɖ]wv^.w $mA Eꖤ>]{Ɲ[wv;WQP&qAixBynYVȇ][ <'oV,v#}2'(Rk2o1K~Z6/X6ȁ,3\ =M;r#'(ܖx'e !Àm;adNƂoȵkv0zT!GkT@ʰ9Kc)Xq`:.' \(9M--vr,x|m?ã #"'Hc_b;e,c6阄;~%Gxyz";'B"8ik,jXzQ'үivVLxl=B'5 lvHPI.sGp-2&hoL1 5=8 z(㟝p+P#ω7\ @Bz\#P5ҕ! @V&nZ .b.2l,1~"#{ZP^ Yt5_N/o( .HD R(D DXsG]W^OQLi T-Wfksq~֊O3 t-ie˂89XLm/MaҢYKpcMZƁQ<(c L/Q.t?#A.bB/ [2GZOwnZ[A霐a1AY:z1WKnF#۸զ6hŸN-)I9a]~|>fX?# ENl\F C\@uXdHЊk;zIvDo:YMG+`+3Z,=j6q]mj_*WfP*o_K=?0Uc |QB9ӯ 6l zQE(6Y)зxg>%6cE bL.@חN h#*N>,u 09S"YK6H,r=[v*[3zH8jiX) 4 ris1n\Q !Xr$݈8 C_Ǘ65,QZE @נWB2؟Cu'Ob:0k' x _Bq{x?fQwdZe И`È,KT#~kaG{i@I4sz?9c$N'ٗyAn) R薲icpbɤH*^5+W 25|?Lf+G,'3NN"j\Lay,EMK{}LſݑYX78 ^ie5IKOj?T54%U HRZx˿8?tYj#tYBrH!(v, u |;9ʼn̼gq.hE  ͱECbJ_\iiŏx?d ! ~NoTf#G[L.#.D.#k "8bTS>m*j*Lu* ({ҭ?^4>R3aJb<-N|! |m]lD|>\d&" -5UiPĐﯽT$%%?&ԯr, ~ g聀cw gH] zЊA>c{@,rzyȄ}1Kd8@[8Q]Vk`1D x+գm i+Ri+Ћh`pqx,u1ɟrx 5L#☰8š`BDc=瞞zmVnl0Teb$OBA_qj[IDVmKFYu%5F(;G f`J-$e(V:/ԔK`8t+T'D$0!`ς{MC"ԡ@DA[i& e($8uv&(i/߷n[e2qL3.(:>ʉJ5i}m&8C %4շS+FI\~7Vs8S< {?fémW݁j˗ 0ZpNFH-I'{}"L}i^ Sn@}$i'!@T'b^*U3+ۚ]'&Y\r%x.g_fQ֨7'GGp1\:&D?te452 )32QI\@ `2(D)RUn.4__z1o&Q$C\n36؆BLN&"T#7'$%Nj4&sӨR4Q@Â%Ȅ4S3Ab MڧB(c.E`h}6gͬcagk >Eeے:*?;܍KX wSg8p]ӛ>No^>IOxo'ـm5{&ljT~EsJ@GLtƸ4p!W8gɲ*+^[[RpY[f|;f ԑT~z,ďm(^vRv ul ^QmWaX;-ʫ{z' C*t  _%ٙ" j+k)s2_Ud_Y0Pj1>o 0ުPzVқ bvv7 2A|wu[Zu.&M۴RH1L2cK;1&Zl]/{2X>(aۋxۋ4qSTc|J@XR0nU8 oӼx і6$E#@FPZc/HY\US(dOs~d=kw۸_R{ON$gqR 5E$Gwf $e9i{9D`0ݝo˾e?Le8s勧== W<{`V9{lEIXpͶ!`VBٹ~79>*x2*7[Q?{7r icbgK})'''ً]FqFr>a,] qi̱[4)xL`ǫBgU2."5TQzh'E`(~.=$W`X tf8;yO YL\?*KX^dg_,{??J?D3d =#L=b0  r"@1`k/43gq9g|ə;0q(пyIV~ƩBo`&31-zqpd.pdt!!Q(yA7h*7VŸuπ*%y H 2 KQH& |yfCvMH_AXC`6A< 2.5;OUƍ+gaYnHrpȠ W C!&m"Ǫ ӣ ;#=2hF<eLw nS"٧#69`MQw 4܆Zc'0͐zagE %CP '2@D$؜ʜLdym_ )5N~^E8Ox0%uuafA*Ll <{FTV|ί)Og5|VPp."]Q*B=MMA(}vM^pcc',AЄ9%HDFMT#U{}[$sFl,8係geND2 3ȴ T֋"-Op@Akl*U69 X@G(bיEtt\?\1I0ѻҺh3<\45[S>5<ܩ:==灚/JÛ{̓gz ^L]:H;)i%#Z}E@஢elan?9k2L8%hʄ 6xsg?d.`sVe('HԼuRĀИ6FBeBl~{vqƯ]ͩm]O/> EnΗqTHBIw\‹Ρ|ΨaӿoqC̡n IDaM6utc-!>܋=1EA*g݌7@A!@x 㦦>`Cy;8&y7I('%V4)%H7Wjt7k5TbWx՞IKk"7(Xꬒ$Zu\Nj9T{7?q4/Ϥ%pk/Q)ݜQz^ȰY,T F(q+H([D6c?^#+g{Yo2h2b aYf F!EI'fL {ǪB 4p_>5>%<",BʒX%r9=S5ZUO(`H> qGݪASOuZ#)61* |U 3`_HoF9-BHs@!ÆhW(^~ț *e8ZmyM9$x'Ǡܛ^T-ls(X(L=C{ejXۀf;閎 !J"X3m^KM' $E`Ey3nZǝ)MGL(xXUK׎@WVpFoBv1mL (hoz4B#Zu!לz.:!I@F4`X,l^3* qf%k] _k7PњѰXJ+Rt1^Llrt(-؄E j>qVPưQD`:VA6g6I[9[M9 Ԡ8YkMQRhVICWw)gXOKtӆJV9XK W]e d`KÓH뎑*3A ʹ_h2 V+6w^Oxd?YUn5VQJB]A8?G`_al/sKYR\#jڟѦjE;J`vH]cS)EY3d/l;BDqϯ_@yNY1H\eVX(*D A>YYJ`~Wh/mU-yyq~F'0]#[;n R^._\{F:RQkƎӖO{δ/{Оr;g|ׂ Tѫ 9Z@0;lok6=;U^[. * {R#&S= ,֦mhJb1oSS{ `jG-0jȟ<>tPNy#zԐ >7=<+@d[ԩ"VMV}4Pn˙`xivU Ξxy|v6 vT\pc܂LJ^`{!+ɝqEm,I j8QcXuۇ" h%΢GG*O}1U:Gt8v6:xjL9w5 :dm٘xZ\<)})y O޿xqto$z]3\A[ھ[·[AyMUdmM+"و ~gik,tSa(&u$\{_5-{/6KEYIz,4@]ќIL| 1 ״d֐%?n5ly xzZCBt/_AZI1W ]Ƒ\["VƆWbE:WkoW`-QA2arY5jbMZW6LFVMSUZ=t*[ K꺮`( ͿjS}u .ARؓQ8m(}YѼYs60 ]{Z LY#O>v<*/+WP#6?nj*)c̔ﵞ9!r^q^q"['ZWb*A ѥe] cƞ MO̫ hoGYI|ԯ2lhZobt6t7 [H6>(04Gt>Hi4l<.FX-xm[+qcLbsU-CjET _ҋ'E<[DO!/O2y0^q굚t'L qHEKؚ2H$xLlR%*Ja) RtvH=ZrW*۔`,*p㬽UuvA}&TUoםʗ9nWאhYy\ }M~KT\C鿲pHӪ.]vHpr| Gٲ#JWІ & ˃t'~J+p򹷁D+7wlԠ=?m^D'u1^E[Uu6;lZJvE:VZrG[,*ʑfæȮN B0dLTDŰAfSa^g*%_xNvVQ&anO3;ue5 (v"A_樮7g+ ,RĊJ]eZm՟MU.vhu6/UyO6:&i/\oYk+ȔPt+E>ECAU`3my2/̏Y_`ī09Rne`|cC~($5T+94*E://@=|یo2hYBeK1NT5nRGϖyA-wvx꾪a}44NMFKN;QN.z~\]7:E4M^+Y}󢲥CKfZS Mbj֠+‚8€vUVɻ,3?W%W _э&fkli0IM𓒢${I>)O uÕIF*SlX!;9ygztKJo X M 73)kZ!~~j3t5ruGڣk"5W)P߰Sz±PReU2n'Q6V&q[hO]}<巑\(>:kIw_"7e R+/g=@_TNnB?.TJ`h'7e;:8^˄Y*BzX{/yaǫY6f6Xcw\[Y~ }YI^6kձ Uح1Abz!bnǓrDe瑼$VSMbJľ;kU8h}aݬd7QnoŮmtb6F;縆}nu@L-+(P鱙`EӀS=j7*_B m7r7+rďf[qC\9:ʟ-__+2'񓹘P&HsƃU@(jT09 >8`,:accywGRхH:dY5W БP.4\\ g[-ɮ6H;zAq: Zm*Zo-}RHX +$?ֺ[LujF547cMzߨo} ބ#1on'{}۳8ѱf g}jޘZu Ot )X]-NJ瘜z+/rAkJKqL'CУjG9] x֔<_W=ELj/I^w 1[dȂxW8E/`ȑ1޷ĶY|?ѺIqq9kh ښG$?/,Mh i3 wG5+h/׏$ZV &$\SZoS<gtqҫJwm+7KܹoUEWA,K=@5W$Pra4 O> t/:x?*4ʏ͡$`oנ>Ë0'vGA:z |0\F.UZI嚄:uߢٻV I\-0jR_Oq4 $tCt/d|\xizY۴{UR8 ) 2ܹu./Z+A.{T4!yy`~/*IyF#]lo~&&|g j:]AO@쯘6T/4ڋ4) qY.j݁jj i˅޼}KET<(c A] OMzEۜPZ^ز E `OqN;'f6%zmXVe69Vq mdSo N _מn:v^]ֆJ \Og}^) Ћ'F_jUtBN~s>0]\HX$ ;}Chtx>MAs":-2kL9ݲ&39VQ~&ò}o|؎!7@?پ ;|[})wRGBMB