目指せ1級!30代サラリーマンボルダリング日記

都内IT企業で働く30代サラリーマンのクライミング記録

カテゴリ: PHP

Web上で変換サービスはたくさんあるんだけれど、サーバ内で変換して出力するパターンはあまり検索にも引っかからなかったのでメモ。

OSはCentOS6。
OpenOfficeとJODConverterをインストール。
このあたりの記事参照。

OpenOffice.orgをドキュメント変換ツールとして活用する方法
JODConverter
OpenOfficeを使ってPDFを作成

# export DISPLAY=:1.0
# Xvfb :1 &
# Xvfb :1 -screen 0 1024x768x8
# soffice -headless -accept="socket,port=8100;urp;" -nofirststartwizard &
# netstat -anp | grep 8100
# java -jar /path/to/jodconverter-2.2.2/lib/jodconverter-cli-2.2.2.jar /tmp/sample.docx /tmp/sample.pdf


    このエントリーをはてなブックマークに追加

ここ最近というか結構前からhtmlスクレイピングすることが多いのでメモ。
最初はいろいろ試してたんだけど、今使ってるのは2パターン。
ベンチマークとって最速だったからとかじゃなく、単に慣れかも。

scrape_func.php

このオライリーの本のサンプルコードに含まれているファイル。
http://www.oreilly.co.jp/books/4873111870/download.html
始点と終点書いて直感的に取ってこれる。
$_rawData = getURL($url);
$_rawData = mb_convert_encoding($_rawData, "UTF-8", "auto");
$_rawData = cleanString( $_rawData );

$headline = getBlock("<div id=\"headline\">","</div>",$_rawData,false);

$title = getElement("h1", $_rawData);

XPATH

xpath使うのが一般的だとは思うけど、ちっちゃい処理だと上に挙げたライブラリの方が早いんだよね。
でも言語に限らず使えるから便利。
このブログのタイトル取る例。
$res = file_get_contents($url);
$dom = @DOMDocument::loadHTML($res);
$xml = simplexml_import_dom($dom);

//最新記事のタイトル
$title = $xml->xpath("//div[@class='hentry']/h2/a/text()");
echo (string) current($title);

//3番目の記事のタイトル
$title = $xml->xpath("//div[@class='hentry'][3]/h2/a/text()");
echo (string) current($title);
//3番目の記事のリンク
$link = $xml->xpath("//div[@class='hentry'][3]/h2/a/@href")

スクレイピングって地道な作業だよね。
各サイトごとにほぼ手動対応。。
    このエントリーをはてなブックマークに追加

何気にGoogleMapsAPI使うの初めてだったりする。
今回「グルーポンなう」作るときに使いました。
単純にAPI叩いてるだけなのでメモ程度に。

前回の記事 【CakePHP】位置情報の取得と、2点間の距離計算のコンポーネントに追加すると便利かも。

というわけで、

App/controllers/components/geo.php

class GeoComponent extends Object {
  public function addr2geo($addr=""){
    $url = sprintf("http://maps.google.co.jp/maps/geo?q=%s&output=json&key=%s",
                   urlencode($addr),
                   'GoogleMapsAPIKEY');
    $res = file_get_contents($url);
    $data = json_decode($res);
    $geo = $data->Placemark[0]->Point->coordinates;
    return array('lng' => $geo[0], 'lat' => $geo[1]);
  }
}
あ、データ取得できなかった時とかの処理は別途書いてね。。

App/controllers/foo_controller.php

class FooController extends Controller {
  var $name = 'Foo';
  var $components = array('Geo');

  function index () {
    $addr = '東京都新宿区新宿三丁目38-1';
    $geo = $this->Geo->addr2geo($addr);
    echo $geo['lat']; //緯度
    echo $geo['lng']; //経度
  }
}


位置情報をサービスに使うとおもしろいね。
今後もいろいろ作っていければいいなー。
    このエントリーをはてなブックマークに追加

20100630211430 よくありそうなこんなかんじのログイン画面。
かんたんログインボタンと、ID/PASS入力するフォーム。

かんたんログインはいろいろありますけど、とりあえず今回は触れません。
キャリアのゲートウェイのIP制限はしましょうってかんじ。


標準のAuthコンポーネントと、ktaiライブラリを使ってます。

UIDが設定してあれば「かんたんログイン」できて、してなければID/PASS入力してUID設定することもできるという機能。
今回はトップページで行う設定です。
Cakeのバージョンは1.3

APP/controllers/top_controller.php

class TopController extends AppController
{
  var $name = 'Top';
  var $uses = 'User';
  var $helpers = array('Ktai','Session','Form');
  var $components = array('Ktai','Session','Auth');

  function beforeFilter() {
    $this->Auth->loginAction = '/';
    $this->Auth->autoRedirect = false;
    $this->Auth->authorize = 'controller';
    $this->Auth->allow('index');
    parent::beforeFilter();
  }

  function beforeRender() {
    parent::beforeRender();
  }

  function index () {
    if ($user = $this->Auth->user()) {
      if (isset($this->data['add_uid'])) {
        $user['User']['uid'] = $this->Ktai->get_uid();
        $this->User->save($user['User']);
/* 7/14追記 */
        $user = $this->User->findById($set['id']);
        $this->Auth->login($user);
/* 7/14追記ここまで */
      }
      $this->redirect($this->Auth->redirect());
    } else {
      if (isset($this->data['auto_login'])) {
        if ($user = $this->User->findByUid($this->Ktai->get_uid())) {
          if ($this->Auth->login($user['User'])) {
            $this->redirect($this->Auth->redirect());
          } else {
            $this->Session->setFlash('再度かんたんログイン設定をしてください',null);
          }
        } else {
          $this->Session->setFlash('かんたんログイン設定がされていません',null);
        }
      }
    }
  }

  function isAuthorized () {
    $user = $this->Auth->user();
    $user['User']['last_login'] = date('Y-m-d H:i:s');
    $this->User->save($user['User']);

    return true;
  }
}

やっていることは以下の3点。
・ID/PASSでログイン成功して、「かんたんログイン設定する」にチェックがされていれば、UIDをとってテーブルに突っ込む。
・かんたんログインの場合はUIDをキーにusersテーブルからデータを取ってきてAuth->loginに突っ込んでログインさせる。
・isAuthorized()でログインが成功したらログイン日時を更新。

usersテーブル

+-------------+--------------+------+-----+---------+----------------+
| Field       | Type         | Null | Key | Default | Extra          |
+-------------+--------------+------+-----+---------+----------------+
| id          | int(10)      | NO   | PRI | NULL    | auto_increment |
| username    | varchar(100) | NO   |     | NULL    |                |
| password    | varchar(100) | NO   |     | NULL    |                |
| uid         | varchar(100) | NO   |     | NULL    |                |
| flag        | int(1)       | NO   |     | NULL    |                |
| last_login  | datetime     | NO   |     | NULL    |                |
| created     | datetime     | NO   |     | NULL    |                |
| updated     | datetime     | NO   |     | NULL    |                |
+-------------+--------------+------+-----+---------+----------------+


あ、クッキーが使えない携帯がまだまだたくさんあるため(主にdocomo)、そのための設定も忘れないようにメモ。

APP/config/bootstrap.php

if (!isset($_SESSION)) {
  if ($ini_set) {
    ini_set('session.use_trans_sid', 1);
    ini_set('session.serialize_handler', 'php');
    ini_set('session.use_cookies', 0);
  }
}


2010/07/14追記
ログインの際にUIDをテーブルに追加した場合、
$this->Auth->user()で取れるユーザデータはセッションなので再度ログインしないといけないみたい。
    このエントリーをはてなブックマークに追加

テーブルのカラムのタイプをtinyint(1)にして、saveしようとしたんだけど値がおかしい。

  $set = array('type' => 2);
  $this->Data->create();
  $this->Data->save($set);

typeに2を入れようとしてるんだけど、データベース見てみると1になってる!!
ぐぐったらいっぱいでてきたので、みんなやっぱりハマるんだね。。
ちなみにCakePHP1.3です。

Cakephpはtinyintをbooleanにするらしい。。。なんてこった。

CORE/libs/model/datasources/dbo/dbo_mysql.php line474
    if (($col == 'tinyint' && $limit == 1) || $col == 'boolean') {
      return 'boolean';
    }

これは実際体験してみないと気づかないね。。
無駄に時間とられた。
    このエントリーをはてなブックマークに追加

位置情報取得

位置情報を扱うライブラリはこちらを使いました。
docomoのiエリアデータを利用して土地名も返してくれます。
http://labs.unoh.net/2008/08/phpgeomobilejp_converter.html
これをvendors/Geomobilejp/につっこむ。

2点間の緯度経度から直線距離を求める

こちらを参考。
http://www.pahoo.org/e-soul/webtech/php02/php02-21-01.shtm
上記のライブラリのデータをそのまま突っ込めるように、ちょっとだけ修正。
計算部分はそのままです。


で、これらをCakePHPのコンポーネントで使いたかったのでまとめました。
続きを読む
    このエントリーをはてなブックマークに追加

サイト全体で使いたい定数、たとえばサイトURLとかを
環境によって切り替えたいなと。

本番、開発どちらでも使うものは
APP/config/common.php

環境に依存するものは
dev.php、production.php
みたいなかんじに。

環境を設定するのは
apacheのconfのvirtualHost書いてるようなところ。
CAKE_CONFIG的な名前で・・・。

SetEnv CAKE_CONFIG dev

APP/controller/app_controller.php

ここでconfigを設定した。
で、beforeRendeerでconfigにセットして
viewでは$configで使えるように。

  function beforeFilter() {
    Configure::load('common');
    if (isset($_SERVER['CAKE_CONFIG'])) {
      Configure::load($_SERVER['CAKE_CONFIG']);
      $this->config = array_merge(Configure::read("common"), Configure::read($_SERVER['CAKE_CONFIG'\
]));
    } else {
      $this->config = Configure::read("common");
    }
  }
  function beforeRender() {
    $this->set("config", $this->config);
  }
    このエントリーをはてなブックマークに追加

zoomeの動画をタグで検索

最近のゲーム動画はやっぱりzoomeですよね。
画面サイズ大きいし、容量も大きいし、ビットレートも高い。
ニコ動からもだいぶ流れてきてるように思います。
というわけで、zoomeのタグ検索をPHPで。

require_once 'XML/RPC.php';

$arr = array('tagname' => 'ゲーム',
             'offset' => 0,
             'length' => 10
             );
$params  = new XML_RPC_Value(array(
                                   'tagname' => new XML_RPC_Value($arr['tagname'], 'string'),
                                   'offset' => new XML_RPC_Value($arr['offset'], 'i4'),
                                   'length' => new XML_RPC_Value($arr['length'], 'i4'),
                                   ), "struct");

$msg = new XML_RPC_Message('zoome.tagSearch', array($params));
$cli = new XML_RPC_Client('/xmlrpc', 'up.zoome.jp', 80);
$res = $cli->send($msg);

if (!$res->faultCode()) {
  $data = XML_RPC_decode($res->value());
  /* "Total:". $data['count']; */
  foreach($data['list'] as $key => $val) {
    $zoome = array(
                   'zoomeid' => $val['zoomeid'],
                   'diaryid' => $val['diaryid'],
                   'title' => $val['title'],
                   'url' => $val['diaryurl'],
                   'body' => $val['body'],
                   );
  }
} else {
  exit('CODE:'.$res->faultCode(). ':'. $res->faultString());
}


欲をいえばソートがほしいですね。
デフォルトで新着順だと思うんですけど、再生数のソートくらいはほしいかなー。
他にも投稿とかのメソッドがあるので公式ドキュメントを参照してみてください。

参考:zoome API(XML-RPC)ドキュメント
    このエントリーをはてなブックマークに追加

OpenIDでログインして、ニックネームを登録してもらうような画面でのバリデートメモ。

usersテーブルのnicknameカラムにバリデートをかけたい。


チェック項目は3つ。

  • ユニークなニックネームか
  • 文字数制限内か
  • 空白のみではないか

空白のみの入力をfunction notSpaceでチェック。

APP/model/user.php

<?php
class User extends AppModel {
    var $name = 'User';
    var $validate = array(
        'nickname' => array(
                            'notSpace' => array(
                                                'rule' => array('notSpace'),
                                                'message' => '空白のみは登録できません。'
                                                ),
                            'between' => array(
                                               'rule' => array('between', 1, 45),
                                               'message' => '1文字以上入力してください。(全角15文字まで)'
                                               ),
                            'isUnique' => array(
                                                'rule' => 'isUnique',
                                                'message' => 'すでに登録されています。'
                                                ),
                            ),
        );
    function notSpace($field=array()) {
      foreach($field as $name => $value){
        if (preg_match("/^( | )+$/", $value)) {
          return false;
        } else {
          return true;
        }
      }
    }
}
?>
    このエントリーをはてなブックマークに追加

前回(アメブロのAtomAPIが倒せない)の記事から数カ月。。
アクセス解析見てもやっぱり同じように困ってる人がいるみたい。

というわけで、久しぶりに調べ直した。
前回のソースと違う部分をピックアップ。
やっぱりWSSE認証がポイントでした。これさえ通ればあとはすんなり。


$nonce = sha1(md5(time()));


ランダム文字生成だから何でもいいとは思うけど。


$pass_digest = base64_encode(pack('H*', sha1($nonce.$created.strtolower(md5($password)))));


ここが全然逆だった。
大文字にするんじゃなくて小文字にする。


あとはgetで記事情報を取得し、その結果からpostするURLをとって記事を投げるみたい。

今回使用したソース。
続きを読む
    このエントリーをはてなブックマークに追加

とりあえずテストでモバイルサイトで作ってみようと、
既存のぼくが管理してるサイト(ネトゲ)のモバイル版作ってみた。

とくにモバイル用に機能追加などはしておらず、
デイリーだったランキングを数時間単位にして表示したくらい。
あとPC版はZendFrameworkで作ってるけど、モバイル版はCakePHPってところが主な違い。

2010年1月18日にPCサイトでモバイル版の告知。
約1週間の計測。

■モバイル版


moPV

moUU



■同期間のPC版。


pcPV

pcUU



■考察


UUをみると、このままでもPC版の1/10くらいのユーザは来そうな感じ。
まーでも、このサイトの9割近くのユーザがネトゲヲタなのでPCの方が使い勝手とか親和性が高いってことか。
ゲームの裏画面でチェックすることが多そうだし。
残りの1割のリア充(笑)が移動中とかに見てるって考えると、この結果は非常に納得。

あ、よく見るとモバイル版のPVが1/23、1/24とそこだけ凹んでる。
この2日間は土日・・・。

要するに外には出ずに、家でずっとネトゲっていうパターンね。


なんかここまで考察に確信が持てるってなかなかないよね。。
    このエントリーをはてなブックマークに追加

前回に引き続き、次はGoogleAnalyticsの導入。
直接書いてもいいんだけど、とりあえずヘルパーに突っ込んた。
vendorにそのままいれるほうがいいのかな。

■APP/views/helpers/ga.php

公式のサンプルソースをヘルパー用に書き換え。

<?php
class GaHelper extends Helper {
  private $GA_ACCOUNT="MO-XXXXXX-X";
  private $GA_PIXEL="/ga.php";

  function getUrl() {
    $url = "";
    $url .= $this->GA_PIXEL . "?";
    $url .= "utmac=" . $this->GA_ACCOUNT;
    $url .= "&utmn=" . rand(0, 0x7fffffff);
    $referer = isset($_SERVER["HTTP_REFERER"]) ? $_SERVER["HTTP_REFERER"] : '';
    $query = $_SERVER["QUERY_STRING"];
    $path = $_SERVER["REQUEST_URI"];
    if (empty($referer)) {
      $referer = "-";
    }

    $url .= "&utmr=" . urlencode($referer);
    if (!empty($path)) {
      $url .= "&utmp=" . urlencode($path);
    }
    $url .= "&guid=ON";
    return str_replace("&", "&", $url);
  }
}
?>

■APP/controllers/app_controller.php

すべてのページで出すから、app_controllerで宣言。

<?php
class AppController extends Controller {
  var $helpers = array('ga');

■APP/views/layouts/main.ctp

メインのレイアウトのbodyタグ上に書く。

<html>
<head></head>
<body>
   ・
   ・
<?php $googleAnalyticsImageUrl = $ga->getUrl(); ?>
<img src="<?php echo $googleAnalyticsImageUrl; ?>" />
</body>
</html>

■APP/webroot/ga.php

あとはGAの管理画面からDLできるga.phpをwebrootに置いて準備完了。
ちゃんと動いてるみたい。

    このエントリーをはてなブックマークに追加

■xml宣言

<?xml version="1.0" encoding="Shift_JIS"?>

これだけで書くとエラーになる。
たぶんみんな最初にこれでつまづくんだろね。
ぐぐったらすぐでてきた。

解決方法はいくつかあって、
・echoでだす。
・apache
とか。

今回はechoで出します。
<?php echo'<?xml version="1.0" encoding="Shift_JIS"?>'."\n"; ?>

■DOCTYPEの指定

<!DOCTYPE html PUBLIC "-//WAPFORUM//DTD XHTML Mobile 1.0//EN" "http://www.wapforum.org/DTD/xhtml-mobile10.dtd">

・3キャリア別にDOCTYPE分ける場合のメモ
http://ma-san.org/2009/10/htmlphpdoctype.html

■Content-Typeヘッダの指定

<meta http-equiv="Content-Type" content="application/xhtml+xml ; charset=Shift_JIS" />

今回はapacheでの設定ではなくレン鯖とか汎用的にいけるようにソースに。
app_controller.phpのbeforerenderで出力。
*)継承先のbeforerender()でparent:beforerender()を忘れずに。

APP/controller/app_controller.php

<?php
class AppController extends Controller {
  function beforerender() {
    header('Content-Type: application/xhtml+xml');
  }
}

■エミュレータ

docomo
http://www.nttdocomo.co.jp/service/imode/make/content/browser/html/tool2/index.html

softbank
http://creation.mb.softbank.jp/web/web_tool.html

au
http://www.au.kddi.com/ezfactory/tool/ue/
http://web.archive.org/web/*/developer.openwave.com/ja/tools_and_sdk/openwave_mobile_sdk/SDK62K/Openwave_SDK_62K.exe

・fireMobileSimulator
http://firemobilesimulator.org/
    このエントリーをはてなブックマークに追加

ちょっと調べることがあったので、メモとして残しておきます。

日本語のtweetの抽出方法はいろいろあるとおもうんだけど、
今回はフォロワー取得APIでユーザの最新のステータスが取れることから、
特定の有名人のフォロワーのtweetを集めて平均値を出してみようかと。

たぶん統計学的にどのくらいのtweetの平均値をとれば統計として信用できる値とかってあるはずよね。。
習ったような習わなかったような気がするけど、とりあえず適当な数とってみました。

ターゲットとした有名人は勝間さん。
勝間さんのフォロワーをAPIで取得して平均値をだしました。

ちょっと古いAPIの仕様見てたらpage=1とかでページ数指定するって書いてたんだけど、公式のドキュメント見たらcursorになってた。
無駄にハマった。

初期値はcursor=-1を設定して、レスポンスにnext_cursorが返ってくるので次のページを見るときはその値を設定する。
(要するにcursor=1,2,3とかじゃないみたい。)

下に書いてある簡単なバッチ作ってみたところ、
対象Tweet6587tweet
トータル文字数234153文字
平均Tweet文字数35.5文字
以上のような結果に。
*)RTで始まってるtweetは除きました。

何度かやってみてもだいたい30文字台だったので、
日本人は30〜40文字でつぶやいているんでしょう、たぶん。

tweetLength.php

<?php
$username = "USERNAME";
$password = "PASSWORD";
$target = 'kazuyo_k';

$tweet=0;
$length=0;
$cursor = -1;
for ($i=0;$i<150;$i++) {
  if ($cursor == 0) break 1;

  $url = sprintf("http://twitter.com/statuses/followers.json?cursor=%d&screen_name=%s",$cursor,$target);
  $json = request($url,$username,$password);
  if (!is_array($json->users)) continue 1;
  foreach ($json->users as $key => $val) {
    if(isset($val->status)) {
      //RTで始まるtweetは飛ばす
      if(preg_match("/^RT(.+)/",$val->status->text)) continue 1;

      if (mb_detect_encoding($val->status->text) != 'UTF-8') continue 1;
      $tweet++;
      $length+=mb_strlen($val->status->text, 'utf-8');
    }
  }
  $cursor = $json->next_cursor;
}
echo "tweet  : " . $tweet . "\n";
echo "length : " . $length . "\n";
echo "avg    : " . $length / $tweet . "\n";

function request($host='',$username='',$password=''){
  $ch = curl_init();
  curl_setopt($ch, CURLOPT_URL, $host);
  curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
  curl_setopt($ch, CURLOPT_USERPWD, "$username:$password");
  curl_setopt($ch, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_1);
  $result = curl_exec($ch);
  curl_close($ch);

  return json_decode($result);
}
?>
    このエントリーをはてなブックマークに追加

各ページによって読み込むjsを変えたいときに。
簡単なことだけど、ふと忘れそうなので。

/path/to/app/webroot/js/foo.jsを読み込みたい場合。

・controller

コントローラではヘルパーの呼び出し。
<?php
class HogeController extends AppController {
  var $name = 'Hoge';
  var $helpers = array('Javascript');

  public function index() {
  }


・view

ビューではaddScriptで<head>タグ内に記述される。
($scripts_for_layoutの部分)
<?php $this->addScript($javascript->link('foo')); ?>

その場所に出力したい場合はそのまま
<?php echo $javascript->link('foo'); ?>

・layout

レイアウトはとりあえずこんなかんじで$scripts_for_layoutを書く。
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-type" content="text/html; charset=utf-8" />
<?php echo $scripts_for_layout; ?>
</head>
<body>
    このエントリーをはてなブックマークに追加

どうTwitterを楽しむべきか、
なにが楽しいのかいまいちよくわからんので
自分で興味が持てるようにAPI使ってサイト作りました。
あとCakePHPの練習も兼ねて。

いちおうECサイト担当っぽく「欲しい」をキーワードに集めてみました。

とりあえずぶっぱなしぎみで作ったので中身ごり押しな感じです。


欲望なう

欲望なう

キャッシュだったり、もうちょっとリアルタイム性だしたりと改善点は多々あると思うのでぼちぼち作ります。
    このエントリーをはてなブックマークに追加

前回のlivedoorBlogにAtomAPIで投稿に引き続きアメブロもやっちゃうぜーと思ったらハマった。
同じAtomAPIだしコード同じで行けるだろうと思ったら違うらしい。

いろいろ調べてみるとWSSE認証が独自ぽい。
でもいくら調べても同じように困ってる人はいても、解決してる人が見つからない。。

追記)倒しました!!(2010-02-03)  >>こちらの記事へ。



これまで調べてlivedoorBlogと違うと思われる部分がpass_digestの作成部分。

livedoorBlog
$pass_digest = base64_encode(pack('H*', sha1($nonce.$created.$password)));

アメブロ
$pass_digest = base64_encode(pack('H*', sha1($nonce.$created.strtoupper(md5($password)))));

パスワードをmd5したあとに大文字化するらしいとの情報。
デマかホントかわかりません。。

返ってくるレスポンスは
CurlResponse Object
(
    [body] => 
X-WSSE authentication required
    [headers] => Array
        (
            [Http-Version] => 1.1
            [Status-Code] => 401
            [Status] => 401 Unauthorized
            [Set-Cookie] => BIGipServerPool_comment=3395622060.20480.0000; expires=Sat, 17-Oct-2009 01:32:13 GMT; path=/
            [Date] => Fri, 16 Oct 2009 02:18:33 GMT
            [Server] => Apache
            [Content-Length] => 83
            [Content-Type] => application/x.atom+xml
        )

)

だめだー、なんか違うとこで間違ってるのか・・・。
今回使用したコードは以下。

続きを読む
    このエントリーをはてなブックマークに追加

app/vendors/shells/以下にファイルを作成。

/path/to/app/vendors/shells/foo.php
<?php
class FooShell extends Shell {

  var $uses = array('Hoge');

  function bar(){
    $this->args[0]; /* コマンドラインからの引数 */
    /* 処理 */
  }
}
?>


cronで使用するケースが多いと思うので、下記のように-appにappディレクトリのパスを設定。
/path/to/cake/console/cake foo bar -app /path/to/app/
    このエントリーをはてなブックマークに追加

前回の記事に引き続きAtomAPI用。
PHPとcURL使用。

<?php
require_once 'cURL.php';

$atomapi_url = "http://cms.blog.livedoor.com/atom/";
$livedoor_id = "your livedoor id"; /* livedoorID */
$password = "your password"; /* パスワード */
$category = "1"; /* カテゴリ */
$title = "sample"; /* 記事タイトル */
$text = "content"; /* 記事本文 */

$created = date('Y-m-d\TH:i:s\Z');
$nonce = pack('H*', sha1(md5(time())));
$pass_digest = base64_encode(pack('H*', sha1($nonce.$created.$password)));
$wsse =
  'UsernameToken Username="'.$livedoor_id.'", '.
  'PasswordDigest="'.$pass_digest.'", '.
  'Nonce="'.base64_encode($nonce).'", '.
  'Created="'.$created.'"';

$text64= base64_encode($text);

$rawdata =
  '<?xml version="1.0"?>'.
  '<entry xmlns="http://purl.org/atom/ns#" xmlns:dc="http://purl.org/dc/elements/1.1/">'.
  '<title type="text/html" mode="escaped">'.$title.'</title>'.
  '<dc:subject type="text/html" mode="escaped">'.$category.'</dc:subject>'.
  '<content type="application/xhtml+xml" mode="base64">'.$text64.'</content>'.
  '</entry>';

$headers = array(
              "X-WSSE" => $wsse,
              "Content-Type" => "application/x.atom+xml",
              "Cache-Control" => "no-cache",
              );

$curl = new Curl();
$curl->headers = $headers;
$curl->options = array("POSTFIELDS" => $rawdata);
$res = $curl->post($atomapi_url);
?>
    このエントリーをはてなブックマークに追加

cakePHPを使ってみる。
PHPカンファレンス2009(前回の記事)でも使用者が一番多かったフレームワーク。
MVCのフレームワーク使ったことある人にはわかりやすいディレクトリ構成。

慣れるまでのメモをつらつら。

基本構成(post_typesテーブルの場合)

コントローラURL: (複数)/post_types/メソッド名/パラメータ
コントローラ: (複数)PostTypesController (extends AppController)
モデル: (単数)PostType (extends AppModel)
ビュー: (複数)app/views/post_types/各メソッド.ctp

モデル

  • テーブルの主キーは、"id"というフィールドで、primary、auto_increment で作るのが基本。
  • "モデル名_id"というフィールドを作るだけで、他のテーブルモデルへの外部キーとして設定。 例:users テーブルの中に、 "post_id" を作成  ⇒ posts テーブルの id を参照。 "post_type_id" を作成 ⇒ post_types テーブルの id を参照。
  • "created" (DATETIME)というフィールドを作成するだけで、新しいデータが入ると時間が自動的に記録。
  • "modified" (DATETIME)というフィールドを作成するだけで、データが更新されるたびに時間が自動的に記録。
  • "updated" (DATETIME)というフィールドを作成するだけで、データが更新されるたびに時間が自動的に記録。
  • "title"というフィールドを作ると、そのページの標準のタイトル名として利用される。
  • "name"や"title" というフィールドを作ると、外部から参照された場合に(指定がなければ)その値がセレクトボックスなどの「選択項目名」として自動的に利用される。(言い換えると、generateList()の引数が指定されていない場合にはこのように動作するということになります。)どちらもなければ、idの値がインデックスと項目名の両方に入ります。(このあたりの処理は、model_php4.phpと model_php5.php に書かれています。)

ビューを出力する必要がないメソッドの場合

<?php
$this->autoRender = false;
?>
あれ、絶対テーブル必要なの・・・!?って思ってしまったから これもメモ。

テーブル未使用モデル、モデル未使用コントローラ

・DBを使用しないモデル⇒$useTable = false;
・モデルを使用しないコントローラ⇒ var $uses = array();、または var $uses = nullでOK。
    このエントリーをはてなブックマークに追加

このページのトップヘ