#!/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.incf,BH R? ldirectorygraphiterator.incBHurdirectorytreeiterator.inc%BH]pinvertedregexiterator.incBHICCpharcommand.incBH#ڒphar.incBH)k<ko8s+X÷lj. {E#S.#%E6}gHJ"E=@b[Λ_ON_" tz% Qi'͏-}HаE+sdDqT{^a3i2#tX#n $=knF# ۸6ŸN)i*?BBĦ^l{%Yϟ>~ӉMʨ8ePݮ#Rîj|No-۾J}fVoyQa!9rc5׸P.{jKDfa3(??WWlzdײ1p4Kbӄݓ,q%v8FC-El0{=!-#v|pkM!uVgc"iK6H,H<[v3lS8jiXД) 4Biڗ\\/ .$e nDRC_WsmjY+ZAΕd?q?3MyyBK2K}KU9v!M$\d6OjcGd!QqWmʮ3LeM_t/?0g%}*|| Sl$qsK~^5nLſݑY48 B/d+ -?OZYx V+`Y1ꌤ é7;cI @㾭;B'*dB(Diudkf3_K >v+#4+L` }S&$^G~%j+Og% SV-Ja(˦c_3m۔WϨB|X*YK5UAP+TπұG-*PeI#5MI9OJ˃D_. 6"O.ap8y̙kvm~d̲K}\NбFbޘ^]F-MdhcJ"aqt˲-[ lݎ8m }@4N/_(hC/*ش*)?v]R$ 6 5j0d-L|rPߗd/k,Ry"RzfuѝH׵ J)ml$xDJj*%U SR^[1vYֆi76G ӟ3>л`͝՘ "u=v"x:95v'C2Z+K3:yh]&7D#w^T-_r=Di"4J0ڠ笡f2WMR1 915Z0]zQLoZZ4cRL9ZRN0}f> 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=s6?;sR#v\R;uS7LzNGI3EHiA )I<ϴD`X'v~t9_޻ؗis8],d21=Y,S62A96Z,cI"6_,flGdD_"XFq4x/OOQxlv%>5hyƗ<,z yGߟ῿P ?vO}ɞ{ 1e V05z$%43gq9g|ə;0q(ҿyIƏ6%SO-Α B#3MfbZxBկ_rV)q,!S"ϽJWh.0f{D&u¬;@R$MxF{0wh(t?n@τPΧuMX(^軡xXoA%xx\&Q3NIg4&鷰شԴdr!NV~V4[pw[ɠKM)R"JI7W} |?uogzCsBf_k9B%UA떷Q2F8H2XF\n5׈(1j1s-FH1 HJnR$%QD !+A< S0qW RrE7|+5lKn @阂(\NΏ $} !7 +6 "GpC!z Yc.XX1b:PqBP Dc"$:NR@g8_a{l8ABX`2>}^x,@< ]Hno8J^F  U'Gk3 hICq0 _͐b(!<(yz[C`6A"d8\jXw䫌nfϻNd#%".\-$ZPs7fo|oE[ xoHd% ~CHKG7vXG98sr^!FAoDWFgt?S |}c<,?$Z"f ې}x[6e cPX2@KBH~?~KQ:Ԕ*Z%9/z&7R#vm>w}_@^_?ҽs{/5!꥾p_C8Sۺ0\Q9_Q{.y0`h' su:s;RavN {+?~0$DzڔkSW-¿=78N49{pTUρ83n 0 MM9|4ǏV4=Ħ&'P@&\ Ċ1 XJ f@dO+hWNCgMt*IU't狠 |/-[{DmRnK萉C BU~%N~e8f짗q%l/@FMSLqTy<,,3(D()!WL|z+ {-dFS4I-BKb@ˉLլkIV=M;pIK}k6wkIBI䠁hS8(omšV6Wy蟸ӌ8V#9#%镵F[OP.?F^>9 J#=좨ԫuс I 2Ǥb1`f9`([IV4Y沖 |5L^Wwqt4:gk"k+iu{]"7(%HvXb21b5auձp6x^Ů,UR̯^bU$*X4HUEPVV{eˇH\`]g]mzokK >-!Xnх껯T 0ynOM1cn Oha_uw^-w:b^# g"D81ܛvce"8}N0UP  `4>0Bvr&TS(+#N LcBSߝDY>tuqTS>P'0kw:\e d`;K M뮱Xn"cq9DV!,wժQ^P +~ONDXk­**!)A{! }eDZ.<jOWUS#*|P9ZXO@B?JtfE9tCbv"HTbߌG.ˮfl ٱ A_#hч = VZ uŵBSUە\WOyYSBVxx[1L z>21tAr$;Tq<`{҅;S^á=Ne*P59ES0HPE6CH9k]=h~w4Km`h2j⾟/Ia;O]a;4Xa+m\e᧦^wr-̃`2*i>E%OSȾb`'5dM+55#{}ԻBv]U}[œ͐L#=/^]U#&P;d{C/{ާk7c/tcmEdR7iky2qYZrճnZ6"n+): Ԛ9 yO:w”1<}T)AF-l'ι,e<)sv|-8wXN'Gt&V:Um΢tbL:,4"o{QmlsXy6ʂaWYqEmрФzPEw523;" %FGG*O}5q5{՘NE9B/]I: Qr)ж_cYl%ܭjްghHU\ʒ TFO9N9PU14iK`>b*&z4c,zZ@ˠ\-ef_[5r麶!bS]GcIplrX; 1`%!ґ&\k /c2Ud|XUBIKGվl{.nj0,ARVmlMr#u7*BϮ _,Qb7XGkz[\]6_ذyV;_wA`i ѽVVmq|mvkՓ+&T\5Yt)Jrcl[-R' մTy?c.Yų=e83gmPP6i]0,.׋5MV.}ts,Zv2Fm3~J̕qZ]UI{V PTxq7y.AS= ͢}--LX[䋽UmEF,y%q($"KoW76F#l#vZiӛLw/l'6* l\Y.瑽]Zv-jϱAg$vheʺ ŸW9m[/v>4/4n(F38tyS! %n4DԫS$J)6((Uj*TIհC-obtoWL>z tsy+4, Wk,>7M-{2IcOFᔪz }*{B&maTGյv+<;wQyYA7ˆD̤J 01Cq~&"wPq#P 9۳q]EVK9T"0\O̫ް#hoG_XQ{U,=y"\ظRfh6$m}Q0ai Ki4l<.FXO&jG GSF`*%)usN:KX@zg6sS+;<: an:cOW&-qHEڗ5q+ +~rLBA 4Sbj 4:{\2Uz*ZWYT`JUuvA}&lVN[5$.>.䁏&ߌ%H*_p8iUV.;k8M#l~(UC'"4u _" |m `f[75(˛.Ƌܞ ڢViWiTVmg6n\н"Z D H662[WaBW(*YiނTܸX%ʇMp]G`Șȩ2]ŰAfS_g'_x<-MROE,wv˘kvb6(FFKF+jHtj.iWN6UyKg dEPMNRu9!iծ *sk<@+ ߷ߏL)OwRc^4tRl-OQcVn5X0)MY72qZ0#{Jȡ^[|x "ʛJ語5EL#F`"x(kA8wHw:~7򸓸HqxwlGCsiɈN3SnCt𳋞_YkeNj5Bi:qt`V_缨t`tSN~ ]Il5Q_6}SXvXޭj yeUDjMYR Ql4ѿsNe&K ^O5_kL2P;+VXO*7G;g&GGMMJ*}=3yN;6=Z!]]EM3Y@}2O BݗI;(q;QMT2'n>u]BH%~p(+v51XZNuݣwD 'n;]fX?ս;HQq'YpAlYYU߮6vy ^<SgD*Ħbau]Xٓ2k7aa#wfPֻ[6Ҍ/nۻ@^x}Z߻hBh܅~v _CZFqٯnZa9 Y^ -@Veј}n!Z  M#p[7wJ+:cj]2\ )ۚtr\^%U:"m@\M (4-͒įcȎv^tTxIe8‹(r("t+mZd[JlY'[69[~.LڷQgcE^Ir0RpwB@v3] qK/*" #x@ˮr1wDfZ/^PzBg@"[#o ZWB(aPK+̧7e 18E7Fέv|@-[Ҡ)HmݣoCld` URīкMOնz~Rտk@К;ܼ(1}<,:݀lū,ĵ/i^SDkchW(LlYqtIcBLBx .ݦW C;ŷ_dPӻ;Z -!{u}muU|9Z hu7A9>B_L5s^3NE4 ƟjaD>ot|7NxV#D@9(^[iZ޺_7jjηYtsjm~*o4OxU.D5['z\uI1Vm}v P箎(z> < ͝fy\Exgq7ϘtLNtO.*P7'5-ҡ wu 2j6Ȯ 8 _j_GL2Mwk " )@Ʀ&#ɴJ䙕}w\#tq 1eL eOA!=ɼլ sA3G;., }kyM뾬$/TބY36j5vk 11Z7ۛ2I"2Hv)&Jľ;kTh}edݬd7QnĮmt=dF;縆}nueb:;]*.{+= V? 1ֺx:RGE|}lTӺwu]i.~|ew0ۚwswϠ+] RUlApp*(.xKppG\LutZ VKD*MaC>qb9kA҇L)b0iu=7#iw4{B|s>kՕ,ёd.B5ܘ 9 g[=/ɸ67H)<A{: ~ҵ|CBW aYZ7b`Nèb`v(/(Ki/b_^#c*2*)`ugVn;s5cz}4Y-|Pi$Mb լpjB_?h!X-L?"pMq 4^W?y-Wm+WZlmjR5P]H__F>mI~:=ۆV I'[)=&Ԥ2hHlF>^.jLt>i4P :k}d?!R4d^ھn0]\HX$ ;}Chtx>MAs":-2kL9ݲ&39VQ~&ò}o|؎!7@?\mW4ʕD6GBMB