一个获取google chrome扩展crx文件信息的PHP操作类
此类中实现了从crx文件获取扩展的Appid、获取manifest.json文件内容、将crx文件转换为一般zip文件
代码如下:
<?php
class CrxParserException extends Exception { } /**
* Chrome crx 解析器,用于获取扩展、皮肤ID
*
*/
class CrxParser {
const MAX_PUBLIC_KEY_SIZE = 65535;
const MAX_SIGNATURE_SIZE = 65535;
const HEADER_MAGIC_PREFIX = 'Cr24';
const CURRENT_VERSION = 2; private $fp = null; //文件指针
private $filename = ''; //文件路径
private $_header = array(); //crx 文件的头信息 public function __construct($filename){
$this->parse($filename);
} /**
* 获取此应用的ID
* @return string
*/
public function getAppid() {
$hash = hash('sha256',$this->_key);
$hash = substr($hash,0,32); $length = strlen($hash);
$ascii_0 = ord('0');
$ascii_9 = ord('9');
$ascii_a = ord('a');
$data = '';
for($i=0;$i<$length;$i++) {
$c = ord($hash[$i]); if($c >= $ascii_0 && $c <= $ascii_9) {
$d = chr($ascii_a + $c - $ascii_0);
} else if($c >= $ascii_a && $c < $ascii_a + 6) {
$d = chr($ascii_a + $c - $ascii_a + 10);
} else {
$d = 'a';
}
$data .= $d;
}
return $data;
} /**
* 从crx文件中获取manifest.json文件的配置信息
* @param unknown $zip_file
* @return mixed[]
*/
function getConfig($key=null){
$zip_file=tempnam(dirname($this->filename),'zip');
$manifest_arr=array();
if($this->convertToZip($zip_file)){
$zip=zip_open($zip_file);
if(is_resource($zip)){
while($zip_entry=zip_read($zip)){
$entry_name=zip_entry_name($zip_entry);
if(preg_match('/manifest\.json$/', $entry_name)){
$content=zip_entry_read($zip_entry, zip_entry_filesize($zip_entry));
$content_j=json_decode($content, true);
if(!empty($content_j)){
$manifest_arr=$content_j;
}
}
}
}
zip_close($zip);
unlink($zip_file);
}
return is_null($key) ? $manifest_arr : $manifest_arr[$key];
} /**
* 将文件转换为zip文件
* @param unknown $target_path
*/
public function convertToZip($target_path=null){
$offset=16+$this->_header['key_size']+$this->_header['sig_size'];
$data=$this->getContent($this->filename,$offset);
return !is_null($target_path) ? file_put_contents($target_path, $data) : $data;
} /**
* 开始解析该 crx 文件
*/
private function parse($filename) {
if(strpos($filename, '://')!==false && !file_exists($filename)) {
throw new CrxParserException("parser init: crx file does not exisit");
}
$this->filename=$filename;
$this->fp = fopen($filename, 'r');
$this->parse_header(); // 解析头部信息
$this->parse_key();
$this->parse_sig();
fclose($this->fp);
} /**
* 解析头部信息,并设置 $_header 数组
* @throws CrxParserException 解析错误抛出异常
*/
private function parse_header() {
$data = fread($this->fp, 16); // HEADER 头信息有16个字节
if($data) {
$data = @unpack('C4prefix/Vversion/Vkey_size/Vsig_size',$data);
}else{
throw new CrxParserException("header parse: error reading header");
}
// 前四个字节拼合 prefix
$data['prefix'] = chr( $data['prefix1'] ).chr( $data['prefix2'] ).chr( $data['prefix3'] ).chr( $data['prefix4'] );
unset($data['prefix1'],$data['prefix2'],$data['prefix3'],$data['prefix4']); if($data['prefix'] != self::HEADER_MAGIC_PREFIX) {
throw new CrxParserException("header parse: illegal prefix");
}
if( $data['version'] != self::CURRENT_VERSION ) {
throw new CrxParserException("header parse: illegal version");
}
if(
empty($data['key_size']) || $data['key_size'] > self::MAX_PUBLIC_KEY_SIZE ||
empty($data['sig_size']) || $data['sig_size'] > self::MAX_SIGNATURE_SIZE
){
throw new CrxParserException("header parse: illegal public key size or signature size");
}
$this->_header = $data;
} /**
* 解析key
* @throws CrxParserException
*/
private function parse_key() {
$key = fread($this->fp,$this->_header['key_size']);
if($key) {
$this->_key = $key;
}else{
throw new CrxParserException("key parse: error reading key");
}
} /**
* 解析sig
* @throws CrxParserException
*/
private function parse_sig() {
$sig = fread($this->fp,$this->_header['sig_size']);
if($sig) {
$this->_sig = $sig;
}else{
throw new CrxParserException("sig parse: error reading sig");
}
} /**
* 从文件中获取指定位置及大小的内容
* @param unknown $filename
* @param number $offset
* @param unknown $length
* @return string
*/
private function getContent($filename,$offset=0,$length=-1){
$stream = fopen($filename, 'rb');
$content = stream_get_contents($stream, $length, $offset);
fclose($stream);
return $content;
} }
使用方法:
<?php
$crxParser=new CrxParser('abc.crx');
echo $crxParser->getAppid();
var_dump($crxParser->getConfig());
//$crxParser->convertToZip('abc.zip');
一个获取google chrome扩展crx文件信息的PHP操作类的更多相关文章
- 15个网页设计必备的Google Chrome 扩展
2011年第一篇,翻译自freelancefolder的一篇文章.以下为译文内容: 最近,我将Google Chrome作为了我的主力浏览器,同时,将其作为我设计和开发网页的工具,尽管我还时常会去Fi ...
- Chrome浏览器扩展开发系列之一:初识Google Chrome扩展
1. Google Chrome扩展简介 Google Chrome扩展是一种软件,以增强Chrome浏览器的功能. Google Chrome扩展使用HTML.JavaScript.CS ...
- C# 获取Google Chrome的书签
其实这个很简单,就是读取一个在用户目录里面的一个Bookmarks文件就好了. 先建立几个实体类 public class GoogleChrome_bookMark_meta_info { publ ...
- 解决高版本 Google Chrome 扩展程序强制停用问题 -摘自网络
1]前往这里下载你喜欢的语言的组策略模板 后缀为.adm (其他的文件自己看 https://docs.google.com/viewer?a=v&pid=sites&srcid=Y2 ...
- [转]C# 获取指定目录下所有文件信息、移动目录、拷贝目录
原文:http://blog.csdn.net/vchao13/article/details/6200255 1.获取指定目录下所有文件信息 /// <summary> /// 返回指定 ...
- Google Chrome 扩展程序开发
根据公司的规定,每月八小时,弹性工作制.所以大家平时来的不太准时,如果有事,下班也就早些回去了.所以一个月下来工作时间可能不够,但是公司的考勤日历是这样的: 除了请假和法定节假日外,其他样式显示都是一 ...
- 推荐一个有趣的Chrome扩展程序-查看任意网站的开发技术栈
对于前端开发人员来说,目前的前端框架层出不穷,最受欢迎的莫过于所谓的前端框架三驾马车:Angular, React和Vue.在学习的过程中,肯定好奇现在的互联网公司的网站用的何种前端框架来开发的. C ...
- Chrome安装crx文件的插件时出现“程序包无效”
有趣的事,Python永远不会缺席! 如需转发,请注明出处:小婷儿的python https://www.cnblogs.com/xxtalhr/p/11043453.html 链接:https: ...
- Chrome安装.crx文件
解决"只能通过Chrome网上应用商店安装该程序"的方法 http://chromecj.com/utilities/2015-04/423.html 1.更多工具->扩展程 ...
随机推荐
- start memcached
memcached -m 64 -p 11211 -u fly -l 127.0.0.1 ------------------------------------------------------- ...
- 将世界坐标转成NGUI坐标
将世界坐标转成NGUI坐标,这个中间须要一个屏幕坐标,可參考例如以下代码: /// <summary> /// 将世界坐标转成UI坐标 /// </summary> /// & ...
- 作为iOS程序员,最核心的60%能力有哪些?
作为iOS程序员,最核心的60%能力有哪些? 一个合格的iOS程序员需要掌握多少核心技能?你和专业的开发工程师的差距有多大?你现在的水平能开发一个功能完整性能高效的iOS APP吗?一起来看看下面 ...
- C++类中static修饰的函数的使用
//在C++中应该养成习惯:只用静态成员函数引用静态成员数据,而不引用非静态成员数据 #include <iostream>using namespace std;class st_inf ...
- 还在为开发APP发愁? 这里就有现成通用的代码!
1.开源控件 1)首页: 1.1)首先是下拉刷新数据的 SwipeRefreshLayout 地址:https://github.com/hanks-zyh/SwipeRefreshLayout 1. ...
- ADO直接调用ACESS数据库MDB
1.ADO用ODBC链接不会出现堆栈溢出. 2.直接用ADO链接,因为对象不是NEW出来的,导致其成员变量也是栈上的,数组申请过大,栈溢出. 用VECTOR或者NEW对象,应该能解决.
- crontab -e 定时任务中的 脚本文件 路径
crontab -l 57 */1 * * * python /home/data/crontab_chk_url/personas/trunk/plugins/spider/chk_url_stat ...
- hibernate面试点
1.谈谈你对hibernate的认识和理解 01.全自动的ORM框架 02.子项目 03.面向对象的思想来解决操作数据库 01.hibernate是一个开放源代码的对象关系映射(ORM)框架,它对JD ...
- Form Template Method
<重构>中此方法叫做塑造模板函数,在设计模式中,对应的模式就是模板模式.重构中的很多变动比较大的方法都会导致重构,但重构中有非常多的小重构手法.就好像建筑一个房子,设计模式教你厨房客厅怎么 ...
- 转载-STM32片上FLASH内存映射、页面大小、寄存器映射
原文地址:http://blog.chinaunix.net/uid-20617446-id-3847242.html 本文以STM32F103RBT6为例介绍了片上Flash(Embedded Fl ...