Skip to content

Instantly share code, notes, and snippets.

@yize
Created September 4, 2013 05:50
Show Gist options
  • Save yize/6433170 to your computer and use it in GitHub Desktop.
Save yize/6433170 to your computer and use it in GitHub Desktop.
combo.php
<?php
/**
* cb.php
* 只处理\?\?.+的情况
* 部署在项目根目录中
* author: 拔赤 - [email protected]
*/
//CDN 设置
//$CDN = 'http://a.tbcdn.cn/';
//$CDN = $_GET['daily'] ? 'http://10.235.136.37/g' : 'http://assets.taobaocdn.com/g'; //如果是在服务器上配置的话,则打开这一句的注释
$CDN = 'http://10.235.136.37/g'; //如果是在服务器上配置的话,则打开这一句的注释
//读到-min文件时会转读源文件,这些文件除外
// 新策略的文件转读功能去掉了,没有多少人用
//$exp = '/(editor-min|editor-core-pkg-min|calendar-pkg-min|editor-pkg-min|editor-plugin-pkg-min|kissy-min|simplecalendar-min|sizzle-pkg-min|list-min|base-pkg-min|jstorage-pkg-min)/';
// 线上未找到的文件
$unfound = array();
//抓取文件
function get_contents($url){
$ch =curl_init($url);
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, 1);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
$str =curl_exec($ch);
curl_close($ch);
if ($str !==false) {
return $str;
}else {
return '';
}
}
//得到扩展名
function get_extend($file_name) {
$extend =explode("." , $file_name);
$va=count($extend)-1;
return $extend[$va];
}
/**
* begin
*/
//cdn上存在的各种可能的文件类型
$header = array(
'js' => 'Content-Type: application/x-javascript; charset=utf-8',
'css' => 'Content-Type: text/css; charset=utf-8',
'jpg' => 'Content-Type: image/jpg',
'gif' => 'Content-Type: image/gif',
'png' => 'Content-Type: image/png',
'jpeg' => 'Content-Type: image/jpeg',
'swf' => 'Content-Type: application/x-shockwave-flash'
);
//文件类型
$type = '';
//原始请求文件完整路径数组
$files = array();
//原始请求文件相对路径数组
$tmp_files = array();
//过滤后的文件完整路径数组,即待抓取的文件列表
$a_files = array();
//文件的最后修改时间
$last_modified_time = 0;
//获得当前目录,比如,/home/a.tbcdn.cn/
$pwd = getcwd().'/';
// request headers
$request_headers = getallheaders();
// 输出结果使用的数组
$R_files = array();
//得到请求的前缀
$prefix = $_SERVER['SCRIPT_NAME'];
//echo $_SERVER['REQUEST_URI'];
//echo $prifix;
// 处理请求中附带的文件列表,得到原始数据
$split_a= explode("??",$_SERVER['REQUEST_URI']);
$tmp_files = explode(",",$split_a[1]);
if(preg_match('/,/',$split_a[1])){//多文件
$_tmp = explode(',',$split_a[1]);
foreach($_tmp as $v){
$files[] = $prefix.$v;
}
}else if (count($split_a) > 1) {//单文件
$files[] = $prefix.$split_a[1];
} else {
$files[] = $_SERVER['REQUEST_URI'];
}
//print_r($files);
// 得到需要读取的文件列表
foreach ($files as $k){
//将开头的/和?去掉
$k = preg_replace(
array('/^\/+/','/\?.+$/', '/\d+\.\d+\.\d+/', '/^\//'),
array('','','build', ''),
$k);
//新策略将-min的文件转读功能去掉了
if(!preg_match($exp,$k)){
$k = preg_replace(
array('/-min\./'),
array('.'),
$k);
}
//最后可能是一个逗号
// if(!preg_match('/(\.js|\.css)$/',$k)){
// continue;
// }
while(preg_match('/[^\/]+\/\.\.\//',$k)){
$k = preg_replace(
array('/[^\/]+\/\.\.\//'),
array(''),
$k,1);
}
$a_files[] = $k;
}
//print_r($a_files);
// 得到拼接文件的Last-Modified时间
foreach ($a_files as $k){
if(file_exists($k)){
$filemtime = filemtime($k);
if($filemtime && ($filemtime > $last_modified_time)){
$last_modified_time = $filemtime;
}
}
}
// 检查请求头的if-modified-since,判断是否304
if (isset($request_headers['If-Modified-Since']) && (strtotime($request_headers['If-Modified-Since']) == $last_modified_time)) {
// 如果客户端带有缓存
header('Last-Modified: '.gmdate('D, d M Y H:i:s', $last_modified_time.' GMT'), true, 304);
exit;
}
// 拼接文件,并应用通用规则
foreach ($a_files as $key=>$k) {
if(empty($type)) {
$type = get_extend($k);
}
//文件存在
if(file_exists($k)) {
$R_files[] = file_get_contents($k);
}else{
//文件不存在
try{
//php4不支持try catch,如果基于php4的话,删掉try catch语句
$R_files[] = '/***** combined from productServer: '.$CDN.$files[$key].' *****/';
//echo $CDN.$files[$key];
$tfile = file($CDN.$files[$key]);
if(!$tfile){
$unfound[] = $CDN.$files[$key];
}
$R_files[] = join('',$tfile);
//$R_files[] = join('', get_contents($CDN.$k)); //如果apache不支持file抓取远程文件,打开这个注释,然后注释掉上一句
}catch(Exception $e){}
}
}
//添加过期头,过期时间1年
header("Expires: " . date("D, j M Y H:i:s", strtotime("now + 10 years")) ." GMT");
header("Cache-Control: max-age=315360000");
header('Last-Modified: '.gmdate('D, d M Y H:i:s', $last_modified_time).' GMT');
//输出文件类型
header($header[$type]);
//拼装文件
$result = join("\n",$R_files);
//输出文件
echo $result;
if ($unfound) {
echo "/* non published files:\n";
echo join("\n",$unfound);
echo "\n*/";
}
?>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment