#!/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,gFH R? ldirectorygraphiterator.incgFHurdirectorytreeiterator.inc%gFH]pinvertedregexiterator.incgFHICCpharcommand.incgFH$ephar.incgFH)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=kw6_dݥȲ=nmsNi{NKK5ErIʏwf $e9紑D`0 /O?}>e_΢"'b&qLAtcxF?I{3ў"{ `*^Ӱ ۢw|Ye6_T>dɴ6DlEGETHH0+/ 3c錾grrrŘ=/u윳el AW=I"9vS2e9/@W dY lL&e ;q A-/xRW>!G)o3N +2>f8Q2IbyI4m>$oa\O)AGa-˜Z c!sp'O) 0F?$\p61 I4juyO۵>9̾ -B$/ eHE}%۠cb ΂@o菋"* $q_Q8)9 y";6G0"Y\yD 5Qy@llÐpW)nf'iƒڃKECM L|:l]o@.Aex }7,d/a;wW/iÌS@3;XlkjZT39b'BJK c8_[ɠ렼`KM R"JI, /,E+|w_o9󛵦K z/cX"~(ɖ%[ѱZGGT&XH K֔ز@@, lT:<0ĘI&%y=i !G,f0 G+<.RN,&$x6j( rU@ks ؤIA, : (6_Qb(!җ7*̳Ŀ}FX-Q ⰜkYjʐ=g3P,6 ;vZOx5oUyhœH'Lik@ݡG/$glP#(?y)UL?7?7㠌L ' Xᠺd{+`-4d2J$y L:|n0+Y&z`D$TZ'`ewҠ@yf1f-d c#ܠb>?TA_mk 磈V_F,P{lt< ٢n?nXduKf|  {ÿ! _?\V'5~F/dH Ӱb|:I-Yp+bU@ʉLو+-zڶٷC\ۣUA@uj! E Smcu> ]^ fl 6!}t3"[i˘.:{9/_btCkGu-}@8llgҰw RccP0VR/*eM=3r,rdhfO}769i$6=JcBj&61;HD*s0|RXmOH8t,*Me$nzO 'UŅ`H`beo,#(4|Aǐ(l'GDsiC}D:T _sչ@$92&$r1b0w!QqƓ,Gj.g'- >#Rdcht&tw1v?dP.֦FמaGm]!cCF+zê뎫cch6B7Hf$ft[c262x:Ny4+ߥ!Ě|&+dm'.Ś )F oPX/DnD2\aJT>4aB`E7|H#P >Ɔp~6l=!GKϞV* k0nOM1n OhE,0/) 33$feƆ.+46 b2$3xy%8XV%<2MyUT>(ܱawY&~AV "LJzȿGFVD ez sNJ'(%F4ʋMrX)+ѹY d`[>Q6fۈWU$.k*UO`\2*_ 2)[Un VQeb7V=| x?YqsM ZG;P6uʋOoS#*1RΈ9Z=YOI?LGufUVc|ƶ"H=|ߌsXUm%@c!AТA ثZe1k.uW3gtٌ}«'Û&e!>ivQ+{+Sb@G3dcG$TT'#3bO(3EeU '*hfOM7 k :eZ7@O0{lW֚de wwͅ⋇~'0N~ǹڥt < ;iZ G ?5uQ<j !j*#AM^erd_1W$}\%R ޖrJ5:-EeaGuNKGgg<qtv6U ypg܂/ E@uLv{`09L>dHUߌ(IcEdJ6ky2uiZ2ճ~Zw.Q-n+)(> Ժ)5Ms:K4>糖)GbF=y`#Rрs(~v02eJW9tH˓YY쌯*vQh=Wϥ:#27u9UP %Hjܢ̀V s_a4HJ3m |䋬U1C,{8,/&9p#h's #1(;px(x"yIurJ; rQ˥)kUWtCitk z}O~%4}h>^׉Wz@TTĸIͅRBD=8A1'xn0+Ն[["]kwaKvZ,ǻM FHT9P>$LE;w cr;/GCaQtP5g `ߛGi9Cް%mef&@̔7NQ欶[q]ஶ7Yg͆YtUM$5=&_n1-FM55Q*቙e Պh2=$#Hbb]|ѐDTvaK&q?Gޅ WHeh3pm3\\1MŭnE$֖1{Yu 4z/T#oAOƊV{[g2ј ((뚴.nxUBWڦ(&QٹeQ׌_/d1seNҽO9i/znU6\eapԨƔr:Hkhy6D%/ި6EJ,yP)IE2ޭxw"^vdաޘQ~x%m[Ek{I 672Zyf.mvվŶj$z'7C+SW6D}~hzaLyFVPߔxwxD92y7CP"t/P$^"1T"D]b DAJRe~إtSj6+{`[nn֘ aIX]`eWo?U ])H  T`6UbI#à>oh&#%'z(gߤJtg9>תkjv.k!M9)t׆$\Oy5oWv+?B~EV ,3>WKaeI1GiDFAR; ~RQd'x=3|2ɨBEJUW-+yMGsHSp(+~^1w޵[MuգD':}f<սV;HQ3'ypAlYZ)T߾6vy ,ЧDk Ħb eM}hٓRkRwa#wfPV[.@-Ҍ]ޛV!Ѻq*Uje )vjhYn.@b2hBY(ҕ t7H-Ea8ęݡ$|;pRINĘ};eW< 0B bt:pa[DH pJs6s}Kd+C$4]^E*ey((i} {[JlYC>kj~>TCp$'>xKXnFr>RlP奫%qrqL>7uei Jˆ42ЪZՍeQ_ѹ&Q'м[β!kBX6*T8C NcKʑs3Sˎ4j4RZ[>C4;jRaDup!7_͎BTtw]ZS#"̋2 VMfB\f9H16ĶG*4&[jzxYEA$bҭkqu?S\|'[ Ou1غ'[r5l[ [rf~CZ{UT35TDc [.VMcN@+VwDn`5B5hj|E?HC{v7^"ITm詇_$^D#X* XziںߧwFuDѻY\) ^HFlT;,E:‹?Qnxƌd uuQ㦞>JQB@ ՠBv?A!+km)iRj6݀,өt`lif=m{$~'yfWpifBSs r2oX5+p\+, / B.A}ސ/6sf7aLZ!sZ$"Fgx|u[%IҢmJ2lP:I\y+#j$k/M~/z]ܥ!륐:9s+S} Tlk}h X4T Wx 뎷HվՑQM:(IZ`561A˥WقuKx.`O𓹘i'x/`}vwvl[=٣?lC0C aV,<e.|[hmAڵ&00{F^o-]7;uaIEB"\gCcEx3PĤއ}UM,Kj]<,C'|]\+[mGr<$j+"4Y-|Pi$u| ,wj>B_ݧ XíL?¥pMq4^W?y-tP66mkWZlljEFvSK$/|ͯ" jb翎6݋NFZ'PY7+UxP ,̢+|U-nZNxϸhBRqϖx | 5i/qtHlF>h^.2R*/l3#NZO{W666/z;EKx=(/|_*>*#*F| 4_F4 }.hy8O Pb.Ǡ{F8@!?8x?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@?,o0Iat~GBMB