cas 登录各种虐

原理:

http://steven-wiki.readthedocs.org/en/latest/security/sso/

麻烦一:

出现一个比较奇葩的逻辑。是有一套系统已经成形了。cas 登录是后补充的.

需要由原页面登录页面进行提交。

麻烦二:

cas是其它合作公司处理,竞争公司,你懂的。。

 

解决方式:

1.直接post 提交

1.1 各种逻辑处理问题相对麻烦,特别是对于已经成形的

1.2页面各种跳转不好看

2.ajax 进行提交(前提目标公司要合作。。。。要合作。。。)

2.1目标服务器需要有权限,设置头

参考:http://haibin.info/?p=532

2.2需要考虑兼容问题(ie8+,chrome 这些ok,ie7以下的请无视看3)

参考:http://haibin.info/?p=973

2.3页面不用跳转,可在原逻辑的支持下实现,修改少。

3.隐藏post 表单 js 提交到 隐藏的iframe (通过target 指定对应的提交地址)

3.1 修改相对较少,没有兼容性问题。因为post提交,只要是浏览器都支持.

3.2不过本身就是通过跳来跳去实现的,只是包装到iframe中,再通过外部函数调用。如果原页面逻辑复杂,推荐这种处理方式

4.jsonp登录,类似ajax(前提目标公司要合作。。。。要合作。。。)

4.1参考http://www.cnblogs.com/jifeng/p/3511219.html(页面图大。。。谨慎 ~.~ )

http://www.cnblogs.com/jifeng/p/3511219.html

5. cas/service 手动写入cookie.(需要有服务器的操作权限或配置)

5.1 最简单的实现方式,影响最小,通过模拟登录获取最终CASTCG ,通过客户端GET 直接设置到cookie

6. 模拟登录,登录成功之后 iframe 进行隐藏post提交,即是登录两次。相对修改较少

7.这边的最终解决方案,搞一个同域名下的服务器.让目标公司配合cookie 设置成顶级域名下

1setcookie ( 'name', 'value' ,null,'/','.xxxxxx.com');

 

注意事项:

登录检查

1.http://casservice/cas/login?servce=跳转的目标地址

1.1访问以上地址,会返回对应的ticket ,ticket 只能抓取一次数据

1.2 这种跳转是直接控制top.location ,无法进行iframe 检查

2.http://casservice/cas/remoteLogin?service=跳转的目标地址&loginUrl=跳转的目标地址

2.1登录地址也可以用来进行登录检查判断

2.2只跳转当前location ,可以当前iframe进行跳转

ticket获取用户信息

1 ticket 只能抓取一次数据

2 http://caservice/cas/serviceValidate?service=获取ticket的地址&ticket=返回的ticket

3.补充xml转array代码

01// 过滤标签头
02$result = preg_replace ( '/<cas:/i', '<', $xml );
03// 过滤结束标签头
04$result = preg_replace ( '/</cas:/i', '</', $result );
05$res = @simplexml_load_string ( $result, NULL, LIBXML_NOCDATA );
06if (! $res) {
07    return false;
08}
09$res = json_decode ( json_encode ( $res ), true );
10if (isset ( $res ['authenticationSuccess'] )) {
11    /**
12     * Array
13     * (
14     * _[authenticationSuccess] => Array
15     * _(
16     * __[user] => xxxxx
17     * __[attributes] => Array
18     * __(
19     * ____[login_flag] => xxx
20     * ____[employeeId] => xxx
21     * ____[asHelp] => xxxx
22     * __)
23     * _)
24     * )
25     */
26    //$res ['authenticationSuccess'];
27    return true;
28}

 

修复Yii ar 在win环境下 oci_pconnect 访问慢问题

同样的代码,在linux 环境下的 apache访问速度就正常,一样是链接外部数据库。

在开发环境中win(php5.2)链接测试机linux数据库访问里,速度会慢。还是用oci_pconnect的情况下.

折腾过php的不同版本 5.4 ts nts 这些版本。速度依旧烂。

测试机内部apache访问时,速度不慢。觉得是出在win上。

这块无解。尝试分析各sql访问上的速度问题.

log 补充 trace 打印出sql访问日志,单条进行数据分析

1'log' => array (
2        'class' => 'CLogRouter',
3        'routes' => array (
4                array (
5                        'class' => 'CFileLogRoute',
6                        'levels' => 'trace, error, warning'
7                )
8        )
9),

 

发现ar 在获取结构时,一次要花费约1s左右。

其它访问速度正常。

ar开启缓存。第一次访问时较慢,后续访问均明显变快,和原来mysql环境下差不多。

1'db' => array (
2              'class'=>'ext.oci8Pdo.OciDbConnection',
3                'connectionString' => 'oci:dbname=//xxx/xx;charset=UTF8;',
4                // 开启表结构缓存(schema caching)提高性能
5                'schemaCachingDuration'=>3600,
6        ),

 

php array cover to xml

参考

http://stackoverflow.com/questions/1397036/how-to-convert-array-to-simplexml

01<?php
02class FileTools {
03    /**
04     * 内部函数
05     *
06     * @param array $arr           
07     * @param SimpleXMLElement $xml        
08     * @return SimpleXMLElement
09     */
10    static function _array_to_xml(array $arr, SimpleXMLElement $xml) {
11        foreach ( $arr as $k => $v ) {
12            $attrArr = array ();
13            $kArray = explode ( ' ', $k );
14            $tag = array_shift ( $kArray );
15 
16            if (count ( $kArray ) > 0) {
17                foreach ( $kArray as $attrValue ) {
18                    $attrArr [] = explode ( '=', $attrValue );
19                }
20            }
21 
22            if (is_array ( $v )) {
23                if (is_numeric ( $k )) {
24                    self::_array_to_xml ( $v, $xml );
25                } else {
26                    $child = $xml->addChild ( $tag );
27                    if (isset ( $attrArr )) {
28                        foreach ( $attrArr as $attrArrV ) {
29                            $child->addAttribute ( $attrArrV [0], $attrArrV [1] );
30                        }
31                    }
32                    self::_array_to_xml ( $v, $child );
33                }
34            } else {
35                $child = $xml->addChild ( $tag, $v );
36                if (isset ( $attrArr )) {
37                    foreach ( $attrArr as $attrArrV ) {
38                        $child->addAttribute ( $attrArrV [0], $attrArrV [1] );
39                    }
40                }
41            }
42        }
43        return $xml;
44    }
45    /**
46     *
47     * @param array $array
48     *          需要转化的数组
49     * @param string $root
50     *          根标签
51     * @param boolean $formatoutput
52     *          是否需要格式化输出
53     * @param boolean $preserveWhiteSpace
54     *          是否过滤标签之间的空白
55     * @throws Exception
56     * @return mixed Ambigous mixed>
57     */
58    static function array_to_xml($array = array(), $root, $formatoutput = false, $preserveWhiteSpace = false) {
59        if (empty ( $root )) {
60            throw new Exception ( '根标签不能为空' );
61        }
62        if (! preg_match ( '/^<w+/?>$/', $root )) {
63            throw new Exception ( '标签格式异常' );
64        }
65        $xml = self::_array_to_xml ( $array, new SimpleXMLElement ( $root ) )->asXML ();
66        if ($formatoutput || $preserveWhiteSpace) {
67            if (! class_exists ( "DOMDocument" ))
68                return $xml;
69            $dom = new DOMDocument ();
70            $dom->preserveWhiteSpace = $preserveWhiteSpace;
71            $dom->loadXML ( $xml );
72            $dom->formatOutput = $formatoutput;
73            $xml = $dom->saveXml ();
74        }
75        return $xml;
76    }
77}

 

dompdf html cover to pdf

http://www.digitaljunkies.ca/dompdf/

常见问题:

1.页面和生成后的pdf各种不对应

需要通过table进行布局。原文件示例代码也是以表格布局为主,否则会引发各种变形。

如果涉及布局太复杂的页面,需要慢慢调试。如果有别的解决方案,请分享

尺寸不是以像素为参考,如pdf 840的页面尺寸,会实际铺满一个1300+的屏幕。所以需要进行特殊设置,不要在页面中尝试精确定位。

 

2.页面尺寸大小

配置文件中有个page_size参数

something like ‘letter’, ‘A4’, ‘legal’, etc. Thee default is ‘letter’,还有其它各种尺寸可以设置,直接源码搜letter即可看到配置文件.

3.页面如果为一个表格,当夹在中间页面时,要报frames的溢出异常。需要将表格调整到当前页。或者直接压到下一页

4.特殊字符无法正常显示,原因不明。google.code上有提供demo实例中有。但是官网本身demo已隐藏相关特殊字符实例。如 欧元符号 €.这边有个项目比较特殊,通过替换成图片进行解决。

php 序列生成工具

01<?php
02/**
03 *
04 * 现在简单实现,可以处理成,根据对应的表,生成对应的序列进行生成,主要同步自增id
05 *
06 */
07class OracleTools {
08    /**
09     * 通过序列获取最新的id
10     * 如果制定表,如果表不存在,则创建新表。并以原表的id+1000的数据添加.
11     * 可以单独指定需要序列的表名和id
12     */
13    public static function getNewId($seqname = "common", $id = "id", $tablename = "", $maxvalue = 0, $startbyzero = false) {
14        $sseqname = $seqname;
15        if (empty ( $tablename ))
16            $tablename = $seqname;
17        $seqname = "O2O_SEQ_{$seqname}";
18 
19        $exists = sql_fetch ( "SELECT
20    count(1)  existsrow
21FROM
22    all_sequences
23WHERE
24    sequence_name = '$seqname'
25AND sequence_owner = '" . Yii::app ()->db->username . "'", 'existsrow' );
26        if (! $exists) {
27            $lastid = 1;
28            if (! $startbyzero) {
29                $lastid = sql_fetch ( "SELECT {$id} FROM {$tablename} WHERE  ROWNUM =1 ORDER BY {$id} DESC ", $id );
30                $lastid += 1000;
31            }
32            if (! $maxvalue) {
33                $maxvalue = '9999999999999999';
34            }
35            sql_execute ( "CREATE SEQUENCE {$seqname} start WITH {$lastid} increment BY 1 MAXVALUE {$maxvalue} minvalue 1 cycle" );
36        }
37        $id = sql_fetch ( "SELECT {$seqname}.nextVal newid FROM dual", 'newid' );
38        return $id;
39    }
40    public static function dropSeq($seqname = "common") {
41        $seqname = "O2O_SEQ_{$seqname}";
42        sql_execute ( "DROP SEQUENCE {$seqname} " );
43    }
44}

 

chinapay 兼容 php5.2-5.5

http://www.zhaoyuanma.com/

001<?php
002define ( "DES_KEY", "SCUBEPGW" );
003define ( "HASH_PAD", "0001ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff003021300906052b0e03021a05000414" );
004bcscale ( 0 );
005$private_key = array ();
006// 兼容 php5.4+
007if (! function_exists ( 'hex2bin' )) {
008    function hex2bin($hexdata) {
009        $bindata = '';
010        if (strlen ( $hexdata ) % 2 == 1) {
011            $hexdata = '0' . $hexdata;
012        }
013        for($i = 0; $i < strlen ( $hexdata ); $i += 2) {
014            $bindata .= chr ( hexdec ( substr ( $hexdata, $i, 2 ) ) );
015        }
016        return $bindata;
017    }
018}
019function padstr($src, $len = 256, $chr = '0', $d = 'L') {
020    $ret = trim ( $src );
021    $padlen = $len - strlen ( $ret );
022    if ($padlen > 0) {
023        $pad = str_repeat ( $chr, $padlen );
024        if (strtoupper ( $d ) == 'L') {
025            $ret = $pad . $ret;
026        } else {
027            $ret = $ret . $pad;
028        }
029    }
030    return $ret;
031}
032function bin2int($bindata) {
033    $hexdata = bin2hex ( $bindata );
034    return bchexdec ( $hexdata );
035}
036function bchexdec($hexdata) {
037    $ret = '0';
038    $len = strlen ( $hexdata );
039    for($i = 0; $i < $len; $i ++) {
040        $hex = substr ( $hexdata, $i, 1 );
041        $dec = hexdec ( $hex );
042        $exp = $len - $i - 1;
043        $pow = bcpow ( '16', $exp );
044        $tmp = bcmul ( $dec, $pow );
045        $ret = bcadd ( $ret, $tmp );
046    }
047    return $ret;
048}
049function bcdechex($decdata) {
050    $s = $decdata;
051    $ret = '';
052    while ( $s != '0' ) {
053        $m = bcmod ( $s, '16' );
054        $s = bcdiv ( $s, '16' );
055        $hex = dechex ( $m );
056        $ret = $hex . $ret;
057    }
058    return $ret;
059}
060function sha1_128($string) {
061    $hash = sha1 ( $string );
062    $sha_bin = hex2bin ( $hash );
063    $sha_pad = hex2bin ( HASH_PAD );
064    return $sha_pad . $sha_bin;
065}
066function mybcpowmod($num, $pow, $mod) {
067    if (function_exists ( 'bcpowmod' )) {
068        return bcpowmod ( $num, $pow, $mod );
069    }
070    return emubcpowmod ( $num, $pow, $mod );
071}
072function emubcpowmod($num, $pow, $mod) {
073    $result = '1';
074    do {
075        if (! bccomp ( bcmod ( $pow, '2' ), '1' )) {
076            $result = bcmod ( bcmul ( $result, $num ), $mod );
077        }
078        $num = bcmod ( bcpow ( $num, '2' ), $mod );
079        $pow = bcdiv ( $pow, '2' );
080    } while ( bccomp ( $pow, '0' ) );
081    return $result;
082}
083function rsa_encrypt($private_key, $input) {
084    $p = bin2int ( $private_key ["prime1"] );
085    $q = bin2int ( $private_key ["prime2"] );
086    $u = bin2int ( $private_key ["coefficient"] );
087    $dP = bin2int ( $private_key ["prime_exponent1"] );
088    $dQ = bin2int ( $private_key ["prime_exponent2"] );
089    $c = bin2int ( $input );
090    $cp = bcmod ( $c, $p );
091    $cq = bcmod ( $c, $q );
092    $a = mybcpowmod ( $cp, $dP, $p );
093    $b = mybcpowmod ( $cq, $dQ, $q );
094    if (bccomp ( $a, $b ) >= 0) {
095        $result = bcsub ( $a, $b );
096    } else {
097        $result = bcsub ( $b, $a );
098        $result = bcsub ( $p, $result );
099    }
100    $result = bcmod ( $result, $p );
101    $result = bcmul ( $result, $u );
102    $result = bcmod ( $result, $p );
103    $result = bcmul ( $result, $q );
104    $result = bcadd ( $result, $b );
105    $ret = bcdechex ( $result );
106    $ret = strtoupper ( padstr ( $ret ) );
107    return (strlen ( $ret ) == 256) ? $ret : false;
108}
109function rsa_decrypt($input) {
110    global $private_key;
111    $check = bchexdec ( $input );
112    $modulus = bin2int ( $private_key ["modulus"] );
113    $exponent = bchexdec ( "010001" );
114    $result = bcpowmod ( $check, $exponent, $modulus );
115    $rb = bcdechex ( $result );
116    return strtoupper ( padstr ( $rb ) );
117}
118function buildKey($key) {
119    global $private_key;
120    if (count ( $private_key ) > 0) {
121        foreach ( $private_key as $name => $value ) {
122            unset ( $private_key [$name] );
123        }
124    }
125    $ret = false;
126    $key_file = parse_ini_file ( $key );
127    if (! $key_file) {
128        return $ret;
129    }
130    $hex = "";
131    if (array_key_exists ( "MERID", $key_file )) {
132        $ret = $key_file ["MERID"];
133        $private_key ["MERID"] = $ret;
134        $hex = substr ( $key_file ["prikeyS"], 80 );
135    } else if (array_key_exists ( "PGID", $key_file )) {
136        $ret = $key_file ["PGID"];
137        $private_key ["PGID"] = $ret;
138        $hex = substr ( $key_file ["pubkeyS"], 48 );
139    } else {
140        return $ret;
141    }
142    $bin = hex2bin ( $hex );
143    $private_key ["modulus"] = substr ( $bin, 0, 128 );
144    $cipher = MCRYPT_DES;
145    $iv = str_repeat ( "x00", 8 );
146    $prime1 = substr ( $bin, 384, 64 );
147    // 兼容 php5.5+
148    // $enc = mcrypt_cbc ( $cipher, DES_KEY, $prime1, MCRYPT_DECRYPT, $iv );
149    $enc = mcrypt_decrypt ( $cipher, DES_KEY, $prime1, 'cbc', $iv );
150    $private_key ["prime1"] = $enc;
151    $prime2 = substr ( $bin, 448, 64 );
152 
153    // $enc = mcrypt_cbc ( $cipher, DES_KEY, $prime2, MCRYPT_DECRYPT, $iv );
154    $enc = mcrypt_decrypt ( $cipher, DES_KEY, $prime2, 'cbc', $iv );
155    $private_key ["prime2"] = $enc;
156    $prime_exponent1 = substr ( $bin, 512, 64 );
157    // $enc = mcrypt_cbc ( $cipher, DES_KEY, $prime_exponent1, MCRYPT_DECRYPT,
158    // $iv );
159    $enc = mcrypt_decrypt ( $cipher, DES_KEY, $prime_exponent1, 'cbc', $iv );
160    $private_key ["prime_exponent1"] = $enc;
161    $prime_exponent2 = substr ( $bin, 576, 64 );
162    // $enc = mcrypt_cbc ( $cipher, DES_KEY, $prime_exponent2, MCRYPT_DECRYPT,
163    // $iv );
164    $enc = mcrypt_decrypt ( $cipher, DES_KEY, $prime_exponent2, 'cbc', $iv );
165    $private_key ["prime_exponent2"] = $enc;
166    $coefficient = substr ( $bin, 640, 64 );
167    // $enc = mcrypt_cbc ( $cipher, DES_KEY, $coefficient, MCRYPT_DECRYPT, $iv
168    // );
169    $enc = mcrypt_decrypt ( $cipher, DES_KEY, $coefficient, 'cbc', $iv );
170    $private_key ["coefficient"] = $enc;
171    return $ret;
172}
173function sign($msg) {
174    global $private_key;
175    if (! array_key_exists ( "MERID", $private_key )) {
176        return false;
177    }
178    $hb = sha1_128 ( $msg );
179    return rsa_encrypt ( $private_key, $hb );
180}
181function signOrder($merid, $ordno, $amount, $curyid, $transdate, $transtype) {
182    if (strlen ( $merid ) != 15)
183        return false;
184    if (strlen ( $ordno ) != 16)
185        return false;
186    if (strlen ( $amount ) != 12)
187        return false;
188    if (strlen ( $curyid ) != 3)
189        return false;
190    if (strlen ( $transdate ) != 8)
191        return false;
192    if (strlen ( $transtype ) != 4)
193        return false;
194    $plain = $merid . $ordno . $amount . $curyid . $transdate . $transtype;
195    return sign ( $plain );
196}
197function verify($plain, $check) {
198    global $private_key;
199    if (! array_key_exists ( "PGID", $private_key )) {
200        return false;
201    }
202    if (strlen ( $check ) != 256) {
203        return false;
204    }
205    $hb = sha1_128 ( $plain );
206    $hbhex = strtoupper ( bin2hex ( $hb ) );
207    $rbhex = rsa_decrypt ( $check );
208    return $hbhex == $rbhex ? true : false;
209}
210function verifyTransResponse($merid, $ordno, $amount, $curyid, $transdate, $transtype, $ordstatus, $check) {
211    if (strlen ( $merid ) != 15)
212        return false;
213    if (strlen ( $ordno ) != 16)
214        return false;
215    if (strlen ( $amount ) != 12)
216        return false;
217    if (strlen ( $curyid ) != 3)
218        return false;
219    if (strlen ( $transdate ) != 8)
220        return false;
221    if (strlen ( $transtype ) != 4)
222        return false;
223    if (strlen ( $ordstatus ) != 4)
224        return false;
225    if (strlen ( $check ) != 256)
226        return false;
227    $plain = $merid . $ordno . $amount . $curyid . $transdate . $transtype . $ordstatus;
228    return verify ( $plain, $check );
229}

 

银联接入

手册
http://console.chinapay.com/NetPayClient/new/NetPayClient_Manual.pdf
无力吐槽啊,居然得用ie来生成私钥,操作界面居然还有兼容问题。
http://console.chinapay.com/newgms/
参考
http://www.phpally.com/tag/unionpay/

使用手册看登录界面右方接入指南,可使用网关,直接查询即可。

目前比较纠结一个问题是,银联的orderid需要满足16位,且包含密钥的中间部分,金额需要从分位算起。
因为原系统已定,所以直接将这边的订单id保存到备注字段中。
将orderid做为第三方id保存。

参数说明:
String MerId 商户号,长度为 15 个字节的数字串,由 ChinaPay 或清算银行分配。
String OrdId 订单号,长度为 16 个字节的数字串,由商户系统生成,失败的订单号允
许重复支付。
String TransAmt 交易金额,长度为 12 个字节的数字串,例如:数字串”000000001234″
表示 12.34 元。
String CuryId 货币代码, 长度为 3 个字节的数字串,目前只支持人民币,取值为”156″ 。
String TransDate 交易日期,长度为 8 个字节的数字串,表示格式为:YYYYMMDD。
String TransType 交易类型,长度为 4 个字节的数字串,取值范围为:”0001″和”0002″,其中”0001″表示消费交易,”0002″表示退货交易。

表 3 ChinaPay 交易状态代码表
状态码 状态信息
1001 消费交易成功
1003 退款提交成功
1005 退款撤销成功
其他
其他均为交易失败,具体信息请登录 ChinaPay 交易控台查询,或查询各银行出
错信息文档。

json dump 支持打印缺损json格式

01/**
02 * 将utf-8编码串转成原字符串
03 * @param string $str utf-8编码
04 * @return string 原字符串
05 */
06function unicode2utf8($str) {
07    if (! $str)
08        return $str;
09    $decode = json_decode ( $str [0] );
10    if ($decode)
11        return $decode;
12    $str = '["' . $str [0] . '"]';
13    $decode = json_decode ( $str );
14    if (count ( $decode ) == 1) {
15        return $decode [0];
16    }
17    return $str;
18}
19/**
20 * 将json串格式化成串输出,对于有缺损的json串同样可以进行格式化
21 * 例:
22 *
23 * {
24 * "枯asdf":{
25 * "adsf":[
26 * "hffe顶替标有"
27 * ],
28 * "0":"asdf"
29 * },
30 * "0":"hahakao"
31 * }
32 *
33 * @param string $json         
34 * @return void
35 */
36function dumpjson($json) {
37    $isnojson = json_decode ( $json, true );
38    $json = preg_replace_callback ( "/(\uw{4})/", "unicode2utf8", $json );
39    $json = preg_replace ( "/\//", '/', $json );
40    $data = indent ( $json );
41    return "<pre>{$data}</pre>";
42}
43function indent($json) {
44    $result = '';
45    $pos = 0;
46    $strLen = mb_strlen ( $json, "utf-8" );
47    $indentStr = '  ';
48    $newLine = "n";
49    $prevChar = '';
50    $outOfQuotes = true;
51    for($i = 0; $i <= $strLen; $i ++) {
52        // Grab the next character in the string.
53        $char = mb_substr ( $json, $i, 1, "utf-8" );
54        // Are we inside a quoted string?
55        if ($char == '"' && $prevChar != '') {
56            $outOfQuotes = ! $outOfQuotes;
57            // If this character is the end of an element,
58            // output a new line and indent the next line.
59        } else if (($char == '}' || $char == ']') && $outOfQuotes) {
60            $result .= $newLine;
61            $pos --;
62            for($j = 0; $j < $pos; $j ++) {
63                $result .= $indentStr;
64            }
65        }
66        // Add the character to the result string.
67        $result .= $char;
68        // If the last character was the beginning of an element,
69        // output a new line and indent the next line.
70        if (($char == ',' || $char == '{' || $char == '[') && $outOfQuotes) {
71            $result .= $newLine;
72            if ($char == '{' || $char == '[') {
73                $pos ++;
74            }
75            for($j = 0; $j < $pos; $j ++) {
76                $result .= $indentStr;
77            }
78        }
79        $prevChar = $char;
80    }
81    $result = preg_replace ( '/\\/', '', $result );
82    $result = preg_replace ( '/\"/', '"', $result );
83    return $result;
84}