123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478 |
- <?php
- /**
- * ****************************************************
- * Android APK File Parser
- * Author: Katana
- * Version: v0.1
- * Web: http://www.win-ing.cn
- * 功能:解析安卓apk包中的压缩XML文件,还原和读取XML内容
- * 依赖功能:需要PHP的ZIP包函数支持。
- * ****************************************************
- */
- class ApkParser {
- // ----------------------
- // 公共函数,供外部调用
- // ----------------------
- public function open($apk_file, $xml_file = 'AndroidManifest.xml') {
- $zip = new ZipArchive();
- if ($zip->open($apk_file) === true) {
- $xml = $zip->getFromName($xml_file);
- $zip->close();
- if ($xml) {
- try {
- return $this->parseString($xml);
- } catch (Exception $e) {
- }
- }
- }
- return false;
- }
- public function parseString($xml) {
- $this->xml = $xml;
- $this->length = strlen($xml);
- $this->root = $this->parseBlock(self::AXML_FILE);
- return true;
- }
- public function getXML($node = null, $lv = -1) {
- if ($lv == -1) {
- $node = $this->root;
- }
- if (!$node) {
- return '';
- }
- if ($node['type'] == self::END_TAG) {
- $lv--;
- }
- $xml = @($node['line'] == 0 || $node['line'] == $this->line) ? '' : "\n".str_repeat(' ', $lv);
- $xml .= $node['tag'];
- $this->line = @$node['line'];
- foreach ($node['child'] as $c) {
- $xml .= $this->getXML($c, $lv + 1);
- }
- return $xml;
- }
- public function getPackage() {
- return $this->getAttribute('manifest', 'package');
- }
- public function getVersionName() {
- return $this->getAttribute('manifest', 'android:versionName');
- }
- public function getVersionCode() {
- return $this->getAttribute('manifest', 'android:versionCode');
- }
- public function getAppName() {
- return $this->getAttribute('manifest/application', 'android:name');
- }
- public function getMainActivity() {
- for ($id = 0; true; $id++) {
- $act = $this->getAttribute("manifest/application/activity[{$id}]/intent-filter/action", 'android:name');
- if (!$act) {
- break;
- }
- if ($act == 'android.intent.action.MAIN') {
- return $this->getActivity($id);
- }
- }
- return null;
- }
- public function getActivity($idx = 0) {
- $idx = intval($idx);
- return $this->getAttribute("manifest/application/activity[{$idx}]", 'android:name');
- }
- public function getAttribute($path, $name) {
- $r = $this->getElement($path);
- if (is_null($r)) {
- return null;
- }
- if (isset($r['attrs'])) {
- foreach ($r['attrs'] as $a) {
- if ($a['ns_name'] == $name) {
- return $this->getAttributeValue($a);
- }
- }
- }
- return null;
- }
- // ----------------------
- // 类型常量定义
- // ----------------------
- const AXML_FILE = 0x00080003;
- const STRING_BLOCK = 0x001C0001;
- const RESOURCEIDS = 0x00080180;
- const START_NAMESPACE = 0x00100100;
- const END_NAMESPACE = 0x00100101;
- const START_TAG = 0x00100102;
- const END_TAG = 0x00100103;
- const TEXT = 0x00100104;
- const TYPE_NULL = 0;
- const TYPE_REFERENCE = 1;
- const TYPE_ATTRIBUTE = 2;
- const TYPE_STRING = 3;
- const TYPE_FLOAT = 4;
- const TYPE_DIMENSION = 5;
- const TYPE_FRACTION = 6;
- const TYPE_INT_DEC = 16;
- const TYPE_INT_HEX = 17;
- const TYPE_INT_BOOLEAN = 18;
- const TYPE_INT_COLOR_ARGB8 = 28;
- const TYPE_INT_COLOR_RGB8 = 29;
- const TYPE_INT_COLOR_ARGB4 = 30;
- const TYPE_INT_COLOR_RGB4 = 31;
- const UNIT_MASK = 15;
- private static $RADIX_MULTS
- = array(
- 0.00390625,
- 3.051758E-005,
- 1.192093E-007,
- 4.656613E-010
- );
- private static $DIMENSION_UNITS
- = array(
- "px",
- "dip",
- "sp",
- "pt",
- "in",
- "mm",
- "",
- ""
- );
- private static $FRACTION_UNITS
- = array(
- "%",
- "%p",
- "",
- "",
- "",
- "",
- "",
- ""
- );
- private $xml = '';
- private $length = 0;
- private $stringCount = 0;
- private $styleCount = 0;
- private $stringTab = array();
- private $styleTab = array();
- private $resourceIDs = array();
- private $ns = array();
- private $cur_ns = null;
- private $root = null;
- private $line = 0;
- // ----------------------
- // 内部私有函数
- // ----------------------
- private function getElement($path) {
- if (!$this->root) {
- return null;
- }
- $ps = explode('/', $path);
- $r = $this->root;
- foreach ($ps as $v) {
- if (preg_match('/([^\[]+)\[([0-9]+)\]$/', $v, $ms)) {
- $v = $ms[1];
- $off = $ms[2];
- } else {
- $off = 0;
- }
- foreach ($r['child'] as $c) {
- if ($c['type'] == self::START_TAG && $c['ns_name'] == $v) {
- if ($off == 0) {
- $r = $c;
- continue 2;
- } else {
- $off--;
- }
- }
- }
- // 没有找到节点
- return null;
- }
- return $r;
- }
- private function parseBlock($need = 0) {
- $o = 0;
- $type = $this->get32($o);
- if ($need && $type != $need) {
- throw new Exception('Block Type Error', 1);
- }
- $size = $this->get32($o);
- if ($size < 8 || $size > $this->length) {
- throw new Exception('Block Size Error', 2);
- }
- $left = $this->length - $size;
- $props = false;
- switch ($type) {
- case self::AXML_FILE :
- $props = array(
- 'line' => 0,
- 'tag' => '<?xml version="1.0" encoding="utf-8"?>'
- );
- break;
- case self::STRING_BLOCK :
- $this->stringCount = $this->get32($o);
- $this->styleCount = $this->get32($o);
- $o += 4;
- $strOffset = $this->get32($o);
- $styOffset = $this->get32($o);
- $strListOffset = $this->get32array($o, $this->stringCount);
- $styListOffset = $this->get32array($o, $this->styleCount);
- $this->stringTab = $this->stringCount > 0 ? $this->getStringTab($strOffset, $strListOffset) : array();
- $this->styleTab = $this->styleCount > 0 ? $this->getStringTab($styOffset, $styListOffset) : array();
- $o = $size;
- break;
- case self::RESOURCEIDS :
- $count = $size / 4 - 2;
- $this->resourceIDs = $this->get32array($o, $count);
- break;
- case self::START_NAMESPACE :
- $o += 8;
- $prefix = $this->get32($o);
- $uri = $this->get32($o);
- if (empty($this->cur_ns)) {
- $this->cur_ns = array();
- $this->ns[] = &$this->cur_ns;
- }
- $this->cur_ns[$uri] = $prefix;
- break;
- case self::END_NAMESPACE :
- $o += 8;
- $prefix = $this->get32($o);
- $uri = $this->get32($o);
- if (empty($this->cur_ns)) {
- break;
- }
- unset($this->cur_ns[$uri]);
- break;
- case self::START_TAG :
- $line = $this->get32($o);
- $o += 4;
- $attrs = array();
- $props = array(
- 'line' => $line,
- 'ns' => $this->getNameSpace($this->get32($o)),
- 'name' => $this->getString($this->get32($o)),
- 'flag' => $this->get32($o),
- 'count' => $this->get16($o),
- 'id' => $this->get16($o) - 1,
- 'class' => $this->get16($o) - 1,
- 'style' => $this->get16($o) - 1,
- 'attrs' => &$attrs
- );
- $props['ns_name'] = $props['ns'].$props['name'];
- for ($i = 0; $i < $props['count']; $i++) {
- $a = array(
- 'ns' => $this->getNameSpace($this->get32($o)),
- 'name' => $this->getString($this->get32($o)),
- 'val_str' => $this->get32($o),
- 'val_type' => $this->get32($o),
- 'val_data' => $this->get32($o)
- );
- $a['ns_name'] = $a['ns'].$a['name'];
- $a['val_type'] >>= 24;
- $attrs[] = $a;
- }
- // 处理TAG字符串
- $tag = "<{$props['ns_name']}";
- foreach ($this->cur_ns as $uri => $prefix) {
- $uri = $this->getString($uri);
- $prefix = $this->getString($prefix);
- $tag .= " xmlns:{$prefix}=\"{$uri}\"";
- }
- foreach ($props['attrs'] as $a) {
- $tag .= " {$a['ns_name']}=\"".$this->getAttributeValue($a).'"';
- }
- $tag .= '>';
- $props['tag'] = $tag;
- unset($this->cur_ns);
- $this->cur_ns = array();
- $this->ns[] = &$this->cur_ns;
- $left = -1;
- break;
- case self::END_TAG :
- $line = $this->get32($o);
- $o += 4;
- $props = array(
- 'line' => $line,
- 'ns' => $this->getNameSpace($this->get32($o)),
- 'name' => $this->getString($this->get32($o))
- );
- $props['ns_name'] = $props['ns'].$props['name'];
- $props['tag'] = "</{$props['ns_name']}>";
- if (count($this->ns) > 1) {
- array_pop($this->ns);
- unset($this->cur_ns);
- $this->cur_ns = array_pop($this->ns);
- $this->ns[] = &$this->cur_ns;
- }
- break;
- case self::TEXT :
- $o += 8;
- $props = array(
- 'tag' => $this->getString($this->get32($o))
- );
- $o += 8;
- break;
- default :
- throw new Exception('Block Type Error', 3);
- break;
- }
- $this->skip($o);
- $child = array();
- while ($this->length > $left) {
- $c = $this->parseBlock();
- if ($props && $c) {
- $child[] = $c;
- }
- if ($left == -1 && $c['type'] == self::END_TAG) {
- $left = $this->length;
- break;
- }
- }
- if ($this->length != $left) {
- throw new Exception('Block Overflow Error', 4);
- }
- if ($props) {
- $props['type'] = $type;
- $props['size'] = $size;
- $props['child'] = $child;
- return $props;
- } else {
- return false;
- }
- }
- private function getAttributeValue($a) {
- $type = &$a['val_type'];
- $data = &$a['val_data'];
- switch ($type) {
- case self::TYPE_STRING :
- return $this->getString($a['val_str']);
- case self::TYPE_ATTRIBUTE :
- return sprintf('?%s%08X', self::_getPackage($data), $data);
- case self::TYPE_REFERENCE :
- return sprintf('@%s%08X', self::_getPackage($data), $data);
- case self::TYPE_INT_HEX :
- return sprintf('0x%08X', $data);
- case self::TYPE_INT_BOOLEAN :
- return ($data != 0 ? 'true' : 'false');
- case self::TYPE_INT_COLOR_ARGB8 :
- case self::TYPE_INT_COLOR_RGB8 :
- case self::TYPE_INT_COLOR_ARGB4 :
- case self::TYPE_INT_COLOR_RGB4 :
- return sprintf('#%08X', $data);
- case self::TYPE_DIMENSION :
- return $this->_complexToFloat($data).self::$DIMENSION_UNITS[$data & self::UNIT_MASK];
- case self::TYPE_FRACTION :
- return $this->_complexToFloat($data).self::$FRACTION_UNITS[$data & self::UNIT_MASK];
- case self::TYPE_FLOAT :
- return $this->_int2float($data);
- }
- if ($type >= self::TYPE_INT_DEC && $type < self::TYPE_INT_COLOR_ARGB8) {
- return (string)$data;
- }
- return sprintf('<0x%X, type 0x%02X>', $data, $type);
- }
- private function _complexToFloat($data) {
- return (float)($data & 0xFFFFFF00) * self::$RADIX_MULTS[($data >> 4) & 3];
- }
- private function _int2float($v) {
- $x = ($v & ((1 << 23) - 1)) + (1 << 23) * ($v >> 31 | 1);
- $exp = ($v >> 23 & 0xFF) - 127;
- return $x * pow(2, $exp - 23);
- }
- private static function _getPackage($data) {
- return ($data >> 24 == 1) ? 'android:' : '';
- }
- private function getStringTab($base, $list) {
- $tab = array();
- foreach ($list as $off) {
- $off += $base;
- $len = $this->get16($off);
- $mask = ($len >> 0x8) & 0xFF;
- $len = $len & 0xFF;
- if ($len == $mask) {
- if ($off + $len > $this->length) {
- throw new Exception('String Table Overflow', 11);
- }
- $tab[] = substr($this->xml, $off, $len);
- } else {
- if ($off + $len * 2 > $this->length) {
- throw new Exception('String Table Overflow', 11);
- }
- $str = substr($this->xml, $off, $len * 2);
- $tab[] = mb_convert_encoding($str, 'UTF-8', 'UCS-2LE');
- }
- }
- return $tab;
- }
- private function getString($id) {
- if ($id > -1 && $id < $this->stringCount) {
- return $this->stringTab[$id];
- } else {
- return '';
- }
- }
- private function getNameSpace($uri) {
- for ($i = count($this->ns); $i > 0;) {
- $ns = $this->ns[--$i];
- if (isset($ns[$uri])) {
- $ns = $this->getString($ns[$uri]);
- if (!empty($ns)) {
- $ns .= ':';
- }
- return $ns;
- }
- }
- return '';
- }
- private function get32(&$off) {
- $int = unpack('V', substr($this->xml, $off, 4));
- $off += 4;
- return array_shift($int);
- }
- private function get32array(&$off, $size) {
- if ($size <= 0) {
- return null;
- }
- $arr = unpack('V*', substr($this->xml, $off, 4 * $size));
- if (count($arr) != $size) {
- throw new Exception('Array Size Error', 10);
- }
- $off += 4 * $size;
- return $arr;
- }
- private function get16(&$off) {
- $int = unpack('v', substr($this->xml, $off, 2));
- $off += 2;
- return array_shift($int);
- }
- private function skip($size) {
- $this->xml = substr($this->xml, $size);
- $this->length -= $size;
- }
- }
- ?>
|