csensix 发布的文章

IE浏览器下HTTP_REFERER失效

后端语言:PHP。

由于功能需求,后端需要判断链接的http_referer值,PHP保存在 $_SERVER['HTTP_REFERER']中。但经过测试,IE浏览器(包括IE7-IE11)下该值为空,谷歌、火狐等均能正常获取。页面链接是通过window.open(url)的方式打开的。

后得知,IE浏览器只有通过a链接或者form表单提交的方式才能获取到HTTP_REFERER,所以有了下面的解决方案,大致就是先判断浏览器是否是IE浏览器,如果是,就模拟a链接点击跳转到指定页面,如果不是,就使用正常方式跳转。

代码如下:

    // document.all 方法ie11不支持,所以不能简单通过 (document.all) ? true : false 来判断
    function isIE() { //ie?
        if (!!window.ActiveXObject || "ActiveXObject" in window)
            return true;
        else
            return false;
        }

    function referURL(url){
        var isIe=isIE();
        if(isIe) {
           var linka = document.createElement('a');
           linka.href=url;
           linka.target="_blank";
           document.body.appendChild(linka);
           linka.click();
         } else {
             window.open(url);
          }
    }

京东金融与兴业银行联合推出的小金卡真实使用体验如何?

2017年8月18日,兴业银行、京东金融联合宣布,将推出国内首张具备“互联网基因”的借记卡即兴业行京东金融小金卡(下简称小金卡),为持卡人提供储蓄、理财、消费等一体化金融服务。那么,京东小金卡实际使用起来怎么样呢?
要想使用京东小金卡,用户需要在京东APP或者京东金融APP内进行小金卡预约。预约成功后,会提示用户到兴业银行营业点办理。办理过程与普通储蓄卡相同,只是多了一个签约京东小金库的步骤,这也是赋予此卡“互联网基因”的很重要的一步。成功办理之后,可通过柜台、ATM存取款机、转账等多种方式往小金卡内打钱,卡内活期账户资金超过1000元(不含1000元)时,超出部分将于次日转入签约的京东小金库,届时可通过京东金融APP查询到转入金额。

转入京东小金库的资金为持卡人提供高于活期存款利率的投资收益(略高于余额宝收益),当持卡人进行快捷支付、境内POS消费或境内ATM取款等交易时,如果卡内活期账户余额不足,该卡又将自动从京东小金库账户赎回差额部分,连同活期账户余额完成支付。不用人工操作即可实现账户资金在银行活期账户与京东“小金库账户”之间的灵活支配,既满足了持卡人对账户资金的流动性需求,又可获得相对更高的投资收益。假设某京东小金卡持卡人存入41000元,卡内超过1000元的部分(即4万元)次日即自动转入京东小金库申购货币基金,享受高于活期存款利率的投资收益。此外,小金卡每月前五次跨行ATM取款免费。

但是有一点需要注意,京东小金卡在消费时会自动关联京东小金库完成支付,但是,如果用户需要通过小金卡转账,那么就需要先在京东小金库里进行提现,然后再行转账,否则(超过1000元)会提示卡内余额不足。由此可见,京东小金卡是鼓励大家进行理财以及消费的。

记一次网站迁移

背景:公司的部分业务是网站建设,现由于公司业务调整,需要将公司部署在电信IDC机房的四千多个网站迁移至百度云。

迁移开始之前,考虑了几个难点:
1、网站数据较大(150G左右),通过公网传输需要耗费很长时间,但其间不能影响网站正常访问;
2、每个站点都配有FTP账号,迁移后要保证原来的账号仍然可用;
3、所有站点都配有二级域名,当初做的是泛域名解析。

第三个难点相对容易解决,只需要调整一下Apache的匹配规则,然后起用全新的泛域名指向即可。

为了保证第二点中提到的FTP可用,第一点中转移数据的时候,不仅要保证数据完整,而且文件、目录的属性也要和老服务器一致。所以,开始转移数据之前,要在新服务器添加所有对应的用户。补充一点,我们用的是SFTP,所以所有账号都是Linux的一个用户,名称和站点目录一致。

所以迁移步骤大致如下:
1、利用Python脚本生成SFTP用户及对应密码,保存到文本文件,每个账号占一行,后面紧跟一个空格加密码;在新服务器中读取该文本文件,并批量添加用户。之所以可以用这种方法,是由于最开始的SFTP账号是根据站点目录名生成的,所以通过脚本比较方便处理。
2、用Rsync同步数据,配置文件中 use chroot = no 来保证所有文件的所属用户和组都正确,另外 -a 参数保证文件读写权限与旧服务器一致。
3、mysqldump方法导出数据库,同样通过rsync同步到新服务器,导入。我们的结构是一个站点一个数据库,所以虽然数据库数量多,但都很小,毕竟都是企业站居多。导入的方法是:python脚本批量解压.gz的数据库文件,然后把需要导入的库写进一个文件source.sql,该文件的内容格式如下:

source xxx.sql;
source xxx2.sql;
...

至此,网站迁移基本完成。

JS实现类PHP date 时间戳格式化函数

  /**
 * 和PHP一样的时间戳格式化函数
 * @param  {string} format    格式
 * @param  {int}    timestamp 要格式化的时间 默认为当前时间
 * @return {string}           格式化的时间字符串
 */
function date(format, timestamp){ 
    var a, jsdate=((timestamp) ? new Date(timestamp*1000) : new Date());
    var pad = function(n, c){
        if((n = n + "").length < c){
            return new Array(++c - n.length).join("0") + n;
        } else {
            return n;
        }
    };
    var txt_weekdays = ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"];
    var txt_ordin = {1:"st", 2:"nd", 3:"rd", 21:"st", 22:"nd", 23:"rd", 31:"st"};
    var txt_months = ["", "January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"]; 
    var f = {
        // Day
        d: function(){return pad(f.j(), 2)},
        D: function(){return f.l().substr(0,3)},
        j: function(){return jsdate.getDate()},
        l: function(){return txt_weekdays[f.w()]},
        N: function(){return f.w() + 1},
        S: function(){return txt_ordin[f.j()] ? txt_ordin[f.j()] : 'th'},
        w: function(){return jsdate.getDay()},
        z: function(){return (jsdate - new Date(jsdate.getFullYear() + "/1/1")) / 864e5 >> 0},
        // Week
        W: function(){
            var a = f.z(), b = 364 + f.L() - a;
            var nd2, nd = (new Date(jsdate.getFullYear() + "/1/1").getDay() || 7) - 1;
            if(b <= 2 && ((jsdate.getDay() || 7) - 1) <= 2 - b){
                return 1;
            } else{
                if(a <= 2 && nd >= 4 && a >= (6 - nd)){
                    nd2 = new Date(jsdate.getFullYear() - 1 + "/12/31");
                    return date("W", Math.round(nd2.getTime()/1000));
                } else{
                    return (1 + (nd <= 3 ? ((a + nd) / 7) : (a - (7 - nd)) / 7) >> 0);
                }
            }
        },
        // Month
        F: function(){return txt_months[f.n()]},
        m: function(){return pad(f.n(), 2)},
        M: function(){return f.F().substr(0,3)},
        n: function(){return jsdate.getMonth() + 1},
        t: function(){
            var n;
            if( (n = jsdate.getMonth() + 1) == 2 ){
                return 28 + f.L();
            } else{
                if( n & 1 && n < 8 || !(n & 1) && n > 7 ){
                    return 31;
                } else{
                    return 30;
                }
            }
        },
        // Year
        L: function(){var y = f.Y();return (!(y & 3) && (y % 1e2 || !(y % 4e2))) ? 1 : 0},
        //o not supported yet
        Y: function(){return jsdate.getFullYear()},
        y: function(){return (jsdate.getFullYear() + "").slice(2)},
        // Time
        a: function(){return jsdate.getHours() > 11 ? "pm" : "am"},
        A: function(){return f.a().toUpperCase()},
        B: function(){
            // peter paul koch:
            var off = (jsdate.getTimezoneOffset() + 60)*60;
            var theSeconds = (jsdate.getHours() * 3600) + (jsdate.getMinutes() * 60) + jsdate.getSeconds() + off;
            var beat = Math.floor(theSeconds/86.4);
            if (beat > 1000) beat -= 1000;
            if (beat < 0) beat += 1000;
            if ((String(beat)).length == 1) beat = "00"+beat;
            if ((String(beat)).length == 2) beat = "0"+beat;
            return beat;
        },
        g: function(){return jsdate.getHours() % 12 || 12},
        G: function(){return jsdate.getHours()},
        h: function(){return pad(f.g(), 2)},
        H: function(){return pad(jsdate.getHours(), 2)},
        i: function(){return pad(jsdate.getMinutes(), 2)},
        s: function(){return pad(jsdate.getSeconds(), 2)},
        //u not supported yet
        // Timezone
        //e not supported yet
        //I not supported yet
        O: function(){
            var t = pad(Math.abs(jsdate.getTimezoneOffset()/60*100), 4);
            if (jsdate.getTimezoneOffset() > 0) t = "-" + t; else t = "+" + t;
            return t;
        },
        P: function(){var O = f.O();return (O.substr(0, 3) + ":" + O.substr(3, 2))},
        //T not supported yet
        //Z not supported yet
        // Full Date/Time
        c: function(){return f.Y() + "-" + f.m() + "-" + f.d() + "T" + f.h() + ":" + f.i() + ":" + f.s() + f.P()},
        //r not supported yet
        U: function(){return Math.round(jsdate.getTime()/1000)}
    };
    return format.replace(/[\\]?([a-zA-Z])/g, function(t, s){
        if( t!=s ){
            // escaped
            ret = s;
        } else if( f[s] ){
            // a date function exists
            ret = f[s]();
        } else{
            // nothing special
            ret = s;
        }
        return ret;
    });
}

使用方法:

console.log(date("Y/m/d H:i:s", 1469771640));

// 输出
"2016/07/29 13:54:00"

Apache Rewrite 排除某个子目录

在做一个网站的时候,使用了apache Rewrite对URL进行重写,但意外发现了一个问题:Ueditor的一些插件,后缀是.html,本不该被重写,结果导致功能异常。虽然有可能可以通过编写更精确的Rewrite规则来避免这个问题,但直接排除后台管理目录的重写显然更容易实现。所以,找到导致插件重写错误的 RewriteRule,在上面添加 RewriteCond,如下:

RewriteEngine on
RewriteCond %{QUERY_STRING} ^(.*)$
#省略...
#下面这条给紧接着的RewriteRule筛选条件,类似if的功能
RewriteCond %{REQUEST_URI} !^/(admin|admin/.*)$
#下面这条导致了上面描述的问题
RewriteRule ^(.*)/([a-z0-9]{1,}).html$ $1/index.php?act=$2
#省略...

如此,所有 admin 子目录的访问都不会被重写。