InstagramのURLからコンテンツのIDを調べる
Google Map API で地名から緯度経度を取得する
PHPで実行時間を同時に何箇所も計測する
PHPのプログラムのどこが遅いのかデバッグするときにブレイクポイントみたいに何箇所にも計測地点を設定したいことがある。PEARのBenchmarkというライブラリがあるようなのでそれを使ってもいいのだが、クラスや関数を跨いでいたりすると面倒なのでグローバルスコープの変数を使って、以下の様なパフォーマンス計測の関数を作ってみた。
function setBenchMarker() { if ( !is_array($GLOBALS["benchmark"]) ) { $GLOBALS["benchmark"] = array(); } $trace = debug_backtrace(); $line = $trace[0]["line"]; $file = $trace[0]["file"]; array_push($GLOBALS["benchmark"], array("point"=>$file.":".$line, "time"=>microtime(true))); } function getBenchMarkResult($delim="\n") { $lasttime = 0; $lastpoint = 0; $retbuf = ""; $records = $GLOBALS["benchmark"]; foreach ( $records as $record ) { $point = $record["point"]; $time = $record["time"]; if ( $lasttime != 0 ) { $retbuf .= "[".$lastpoint."]--[".$point."] = ".sprintf('%0.5f', $time-$lasttime).$delim; } $lastpoint = $point; $lasttime = $time; } return $retbuf; }
使い方
class Hoge { public function __construct( ){} public function moge() { setBenchMarker(); } } function foo() { setBenchMarker(); } setBenchMarker(); foo(); $hoge = new Hoge(); $hoge->moge(); setBenchMarker(); $bench = getBenchMarkResult(); echo $bench;最終結果を文字列として返しているのはerror_logなどに出力することもあるだろうなので、 中でechoはしないようにしている。
実行結果例
$ ./test.php [/home/ec2-user/test/test.php:4]--[/home/ec2-user/test/test.php:21] = 0.087116003 [/home/ec2-user/test/test.php:21]--[/home/ec2-user/test/test.php:30] = 0.010456085 [/home/ec2-user/test/test.php:30]--[/home/ec2-user/test/test.php:9] = 0.000001907
人間を宇宙に送り出す必要性
ギズの記事は機能性の高い宇宙服がどのようになっていくかって話だけど、そもそも人間が現段階で宇宙に行かないといけない理由は、人間の身体が宇宙でどんな変化を起こすかという臨床的なデータを出す以外には無いと思っている。
宇宙服のこれまでとこれから。
人間を宇宙に送れば、食事や人間が生活するのに必要な施設が必要になる。高い金払ってまで再利用できない食いモンとかトイレをロケットに載せる必要は無い。
ロボット工学が発展した今日、他の惑星に居住地点を作り上げるところまではロボットに任せて良いだろう。
きっと3Dプリンターで現地の物質を素材に拠点を作っていくのだろう。ロボットだって人型である必要は無い。
paypal月額課金のIPN Listenerの実装
Paypalの月額課金(定期購読)は、管理画面上でボタンのHTMLを作って、それを設置するだけで一応、ユーザから月額課金を実現することは出来るのだが、サービス側でこのユーザがこのコンテンツに月額課金してるよってってのは、IPN Listenerというサーバ側でPaypalのトランザクションを受け取る仕組みを実装してやらないといけない。
Paypal上にこのIPNの仕様のドキュメントはあるはあるのだが、ドキュメントがあまり更新されてなかったり、リンク切れのドキュメントがあったりと、やれやれな状態だが、実装した方法を紹介。
定期購読とIPNのおおまかな流れ
- ユーザが設置した定期購読ボタンを押下する
- Paypalに遷移し、ユーザがPaypal上で決済を行う
- Paypal上の決済完了画面が表示される(自サービスのページには戻されない)
- 非同期でPaypalから自サーバのIPN ListenerのAPIがコールされ、トランザクションを受け取る(トランザクションの種類については後述)
- 自前のIPN Listenerでは、受けたトランザクションがPaypalからのものであるかどうかを確認するためにPaypalに確認のコールを行う
- 妥当性の確認が取れたらユーザの決済が完了したトランザクションを確認したらユーザに課金のサービスを提供できる状態にする(DBの更新とか)
- 定期的(ボタンで設定した決済間隔)にユーザの決済処理が行われて、その結果がトランザクションで飛んで来るので、決済されていればサービス延長する。
- 定期購読の契約がキャンセルされたら同じくキャンセルのトランザクションが飛んでくるので当該サービスを停止処理を行う。
定期購読ボタン作成時の注意
- ボタンを作成する際には提供サービスごとに課金額を変えるような場合は、購読IDを指定するか、カスタム値にサービスのIDを入れる。
- また、Paypalからのトランザクションの戻りがどのユーザの決済かを判断するためにカスタム値にユーザIDなどを入れておく。
サービスIDとユーザIDをボタンのカスタム値としていれる例
<input TYPE="hidden" name="custom" value="<?php echo $serviceid;?>,<?php echo $userid">※何個もカスタム値を入れるなら、わかりやすくするために、serviceid=1,userid=hoge のような入れ方のほうが良いかも。
IPNトランザクションの種類
subscr_signup(購読の契約が行われた)
例えば、初月無料という定期購読にした場合は、このトランザクションを受けたらサービスを有効にしてやる。
subscr_payment(決済が行われた)
subscr_cancel(購読の契約が解除された)
subscr_failed(決済が失敗した)
クレジットカードの有効期限が切れていたりなどで、決済日に決済が行われないとこのトランザクションが流れてくる。こいつがくせ者で、決済に失敗したからといって購読の契約が解除されたわけではない。提供サービスを停止して、再度、定期購読が申し込める画面にアクセスできるようにしてしまうと、ユーザが2重の契約をしてしまう可能性がある。
クレジットカードを登録しなおしたりで決済が可能になると、subscr_paymentが流れてくるので、failed中はサービスを停止ではなくサスペンド状態という扱いにするなどして、契約がキャンセルされたわけじゃないということを意識しておく必要がある。
subscr_failの状態がどれだけ続いたとしても契約は残っている。契約が解除されたら、必ずsubscr_cancelが流れてくるのでそれまではサスペンドする。
subscr_eot(購読の有効期限切れ)
正直、このトランザクションが流れてくるタイミングが良くわからない。購読の有効期限の管理は、自サービス側で行うだろうから、無視しても大丈夫?
StackOverflowでは、以下の様なことやり取りしているが、うーん、わからん。
StackOverflowでは、以下の様なことやり取りしているが、うーん、わからん。
TLS1.2対応が必須になりそう
2016年1月時点では、Paypalの本番環境ではTLS1.0でのコール(phpのcurl関数でトランザクションの確認のコール)ができていたが、今年になって?からテスト環境にはTLS1.0でコールするとエラーが返ってきた。
どうもPHP5.3のcurlではTLS1.2のプロトコルに対応していないようなので、サーバのcurlコマンドをアップデートして、exec関数でコールするようにしてみた。
どうもPHP5.3のcurlではTLS1.2のプロトコルに対応していないようなので、サーバのcurlコマンドをアップデートして、exec関数でコールするようにしてみた。
Paypalから受け取ったデータの妥当性のチェック通信を行なう
function confirmPaypalTransaction($posts) { $paypal_domain = "www.paypal.com"; //テスト環境はsandbox.paypal.com $req = 'cmd=_notify-validate'; foreach ($posts as $key => $value) { $value = urlencode(stripslashes($value)); $req .= "&".$key."=".$value; } $url = 'https://'.$paypal_domain.'/cgi-bin/webscr'; $cmd = '/usr/bin/curl '.$url.' --tlsv1.2 -d "'.$req.'"'; exec($cmd, $res, $ret); /* 以下のcurl関数では叩けなくなった(PHP5.3) $ch = curl_init($url); curl_setopt($ch, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_1); curl_setopt($ch, CURLOPT_POST, 1); curl_setopt($ch, CURLOPT_RETURNTRANSFER,1); curl_setopt($ch, CURLOPT_POSTFIELDS, $req); curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, 1); curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, 2); curl_setopt($ch, CURLOPT_SSLVERSION, CURL_SSLVERSION_TLSv1); curl_setopt($ch, CURLOPT_SSL_CIPHER_LIST, 'rsa_aes_128_sha'); curl_setopt($ch, CURLOPT_FORBID_REUSE, 1); curl_setopt($ch, CURLOPT_HTTPHEADER, array('Connection: Close')); if ( !$res = curl_exec($ch) ) { $this->writeLog("FATAL", "CURL ERROR:". curl_error($ch)); } curl_close($ch);*/ return $res; }