GAEのgoogle.appengine.ext.webapp.WSGIApplicationはRailsでおなじみの_methodパラメータによるメソッドエミュレーションをやってくれないのでREST厨涙目ということで、そこだけのMiddlewareを書いてみました。

import webob

class MethodEmulator(object):
def __init__(self, application):
self.application = application

def __call__(self, environ, start_response):
req = webob.Request(environ)
if req.params.has_key('_method'):
method = req.params['_method'].upper()
if method in ['PUT', 'DELETE']:
environ['REQUEST_METHOD'] = method
return self.application(environ, start_response)

HEAD, OPTIONS, TRACEやWEBDAVのメソッドも何か怖いので無し。

application = WSGIApplication([('/', MainHandler)])
run_wsgi_app(MethodEmulator(application))

こんな感じで被せて使う。

Rackだとuse MethodEmulatorって感じだったけどRuby始めたばっかりで、

「useなんて予約語無いじゃないか!」

とか迷ったので明示的なのも分かりやすくて良いですね。

REE

$ wget https://packages.endpoint.com/endpoint-rpmsign.pub
$ sudo rpm --import endpoint-rpmsign.pub
$ sudo yum remove ruby*
$ wget https://packages.endpoint.com/rhel/5/os/i386/ruby-enterprise-1.8.7-2.ep.i386.rpm
$ wget https://packages.endpoint.com/rhel/5/os/i386/ruby-enterprise-rubygems-1.3.5-2.ep.i386.rpm
$ sudo rpm -ivh ruby-enterprise-1.8.7-2.ep.i386.rpm
$ sudo rpm -ivh ruby-enterprise-rubygems-1.3.5-2.ep.i386.rpm

RMagick

$ wget http://ftp.kddlabs.co.jp/graphics/ImageMagick/ImageMagick-6.5.8-6.tar.gz
$ tar zxf ImageMagick-6.5.8-6.tar.gz
$ cd ImageMagick-6.5.8-6
$ ./configure --prefix=/usr
$ make
$ sudo make install
$ sudo gem install rmagick
天地創造

何時の間にやらMacでのSnes9xが動くようになってるらしく、無線コントローラーのLogicool Cordless Rumblepad 2と合わせるととても快適。

俺のようなどうせ最新ハードでもレトロゲームやその復刻版か続編しか食手が伸びないおっさんにとってはこういう方が楽しいかも。

Snes9xはC++だそうですが、MacのIntel化の恩恵は少なくなさそうです。

ロジクール Cordless Rumblepad 2(ブラック) GPW-600

svn最新だけをclone

職場はsvnで個人的にgitを使うときなど、全部持ってくると死ぬほど時間がかかるので最新のtrunkだけ持ってくる。

% git svn clone -rHEAD https://svn.example.com/unk/trunk unk

svnから更新分を持ってくる

% git svn rebase

svnへ更新分を反映する

% git svn dcommit

空ディレクトリの削除をsvnにお伝えする

% git svn dcommit --rmdir

svnをgitに変換

-sは--stdlayoutでtrunk/, branches/, tags/構成を想定してブランチに変換してくれるそうです。

% git svn clone -s http://foo.com/repos/project

ブランチ一覧

% git branch

ブランチ作成

% git branch michel-jackson

ブランチ切り替え

% git checkout michel-jackson

ブランチ作りつつ切り替え

% git checkout -b michel-jackson

ブランチをマージ

% git checkout master
% git merge michel-jackson

一個前のコミットをなかった事にする

% git reset HEAD^

昔のコミットを修正する

% git rebase -i 695b1eb
(修正)
% git commit --amend
% git rebase --continue

タグ付ける

% git tag v0.0.1
% git push --tags

タグ情報付きで持ってくる

% git pull --tags

一つ戻して無理やりpush

% git reset --soft HEAD^
% git ci -a -m 'やりなおし'
% git push origin master --force

ブランチをpush

% git push origin win32

localにphp5.3.1とwordpress2.8.6をインストールしてみるが、DBに繋がらないとのエラー。詳細が知りたいのでphp.iniのerror_logを設定してエラーログを見てみるが何も出力されてない。

<?php trigger_error('error raising!'); ?>

上記の様なスクリプトを実行してみると確かにエラーログに残るのだが・・・。

% grep -r @mysql .
./wp-includes/wp-db.php: $this->dbh = @mysql_connect($dbhost, $dbuser, $dbpassword, true);
./wp-includes/wp-db.php: if (!@mysql_select_db($db, $this->dbh)) {
./wp-includes/wp-db.php: $this->result = @mysql_query($query, $this->dbh);
./wp-includes/wp-db.php: while ($i < @mysql_num_fields($this->result)) {
./wp-includes/wp-db.php: $this->col_info[$i] = @mysql_fetch_field($this->result);
./wp-includes/wp-db.php: while ( $row = @mysql_fetch_object($this->result) ) {
./wp-includes/wp-db.php: @mysql_free_result($this->result);

なるほど・・・。

これだと原因究明が難しいのでmysql_connectの@を取って実行してみた。

% diff wp-includes/wp-db.php{,.org}
341c341
< $this->dbh = mysql_connect($dbhost, $dbuser, $dbpassword, true); # FIXED by komagata
---
> $this->dbh = @mysql_connect($dbhost, $dbuser, $dbpassword, true);

php_errors.log:

[14-Dec-2009 02:49:02] PHP Warning:  mysql_connect() [<a href='function.mysql-connect'>function.mysql-connect</a>]: [2002] No such file or directory (trying to connect via unix:///tmp/mysql.sock) in /users/komagata/Sites/wordpress/wp-includes/wp-db.php on line 341
[14-Dec-2009 02:49:02] PHP Warning: mysql_connect() [<a href='function.mysql-connect'>function.mysql-connect</a>]: No such file or directory in /users/komagata/Sites/wordpress/wp-includes/wp-db.php on line 341

なるほど、/tmp/mysql.sockにソケットファイルがあると思っているために接続できないらしい。

PHP: 実行時設定 - Manual

mysql.default_socket string

他にソケット名が指定されない場合、ローカルなデータベースサーバに 接続する時のデフォルトのソケット名。

php.iniにデフォルトのMySQLのsocketの場所を設定できる項目があることがわかった。ちなみに接続時のhostにもlocalhost:/path_to/mysql.sockといった指定が可能らしい。

PHPのマニュアルは非常に詳細な内容も記載されていて、日本語訳も充実しているので巷で問題になってたりすることの殆は公式マニュアルで解決することが多い。

macportsで入れた場合のmysqlのデフォルトのsocketの場所である/opt/local/var/run/mysql5/mysqld.sockを指定したら動いた。

問題解決のためにはコード内でエラー抑制はせず、各個人の環境でdisplay_errorsやerror_logを設定すべきだと思うが、php.iniが弄れない共有レンタルサーバーで使われることが多い事や、エラーがズラズラでてもパニックに陥るだけといった理由でこういう作りになっているのかなと思った。

本番環境ではすべきでないが、大抵の共有レンタルサーバーと同じ様にdisplay_errosOnにした場合、エラー抑制子を外すと、データベースに接続できない場合はこんな感じになる。

データベースエラー

たしかにちょっと怖い感じがするかも。初めてのインストールでこれが出たら諦めちゃう人も多いかも。

Macにインストール

% sudo port install php5 +apache2 +pear
% sudo port install php5-mysql php5-mbstring
% cd /opt/local/etc/php5
% sudo cp php.ini-development php.ini
% sudo /opt/local/apache2/bin/apxs -a -e -n "php5" libphp5.so
[activating module `php5' in /opt/local/apache2/conf/httpd.conf]
% sudo vi /opt/local/apache2/conf/httpd.conf
# mode_php5
Include conf/extras-conf/mod_php.conf

Timezoneを設定

/opt/local/etc/php5/php.ini

[Date]
date.timezone = Asia/Tokyo

MySQL設定

/opt/local/etc/php5/php.ini

[MySQL]
mysql.default_socket = /opt/local/var/run/mysql5/mysqld.sock

portsで入れたphpとmysqlそれぞれの見に行くsockの場所のdefaultが違う。

あと半月ちょっとで2010年ですが、想像してたのと随分違うなあ・・・。

ストリートファイター2010

sf2010

ストリートファイター4

ストリートファイター4

関連:PLANET5の厳しさ - p0t

PythonでCSVのFixtureをYAMLに変換(添削希望・・・)などと書いていたんですが、なんとid:perezvonさんが添削してくださいました!

あざーっす!

元のコード:

import sys
import csv

if (len(sys.argv) < 2):
  print 'Usage: %s ' % sys.argv[0]
  quit()

csv_file = sys.argv[1]

filename, suffix = csv_file.split('.')
reader = csv.reader(file(csv_file, 'r'))
i = -1
res = []
for row in reader:
  i += 1
  if i == 0:
    column = row
    continue

  s = filename + str(i) + ":\n"
  for j in range(len(column)):
    s += "  " + column[j] + ": " + row[j]  + "\n"
  res.append(s)

f = open(filename + ".yml", 'w')
f.write("\n".join(res))
f.close()

添削していただいたコード:

import sys
import csv
import yaml

def main(src, dest):
    reader = csv.reader(file(src))
    data = list(reader)

    result = []
    cols = data[0]
    rows = data[1:]
    for row in rows:
        result.append(dict([(k, v) for k, v in zip(cols, row)]))

    yaml.safe_dump(result, file(dest, 'wb'),
                   default_flow_style=False,
                   encoding='utf-8',
                   allow_unicode=True)

if __name__ == '__main__':
    if len(sys.argv) < 2:
        print 'Usage: %s [filename]' % sys.argv[0]
        sys.exit(1)

    src = sys.argv[1]
    dest = src.replace('.csv', '.yml', src.rindex('.'))
    main(src, dest)

ジーザス・・・スマート!

こちらを参考に復習していきたいと思います。

まずこの部分。

if __name__ == '__main__'

なるほど〜Rubyでもやりますね。この辺は無意識にできるようになりたい。

rows = data[1:]

csvの1行目以降を簡単に取っている。どうも簡単なデータ構造を基本的な処理の組み合わせでシンプルに扱うというところを疎かにしがちで冗長になったり、安易にクラスにしたりするので反省していきたい。

そしてこのコードの処理の大半が次の一行に凝縮されている。

result.append(dict([(k, v) for k, v in zip(cols, row)])

zip関数を使うと複数のシーケンスを簡単に回せるんですね!これ、Rubyだとちょっと面倒な処理だと思います。(俺のRubyスキルの低さ含めて)

これはモテる!

yaml.safe_dump

そしてyaml.safe_dump。まさに元エントリーで書かれているように、普通にyaml.dumpして(´・ω・`)となっていました・・・。

dest = src.replace('.csv', '.yml', src.rindex('.'))

ここも素敵。元のコードだとピリオドが一つ以上あるファイル名でおかしなことになりますね。

同じ問題をうまい人にも解いてもらってそれを見るのはとても勉強になりますね。プログラマの教育の話でも目の前で解いて見せるという話がありましたが、とても効率的な気がしました。

日々ブログに恥を垂れ流してますが時々こういう良い事があるので嬉しいですね。ありがとうございます!

参照:PythonでCSVのFixtureをYAMLに変換 - スコトプリゴニエフスク通信

関連:PythonでCSVのFixtureをYAMLに変換(添削希望・・・) - p0t

「やまざき冬のテスト祭りで小皿を当てよう」ということで設定の見直し中。

以前は少し面倒だったautotestのgrowl通知がスマートになってました。

% sudo gem install autotest-growl
% vi ~/.autotest
require "autotest/growl"
GrowlHelperApp

楽になったもんですなー。

Mr.Tブログを復活させるべくWordPress2.8.6を実家サーバーに入れてみる。

インストール

$ sudo apt-get install libapache2-mod-php5 php5-mysql
$ mysqladmin -uroot -p create mrt
$ mysql -uroot -p -e "grant all on mrt.* to mrt@localhost identified by 'unk';"

$ wget http://ja.wordpress.org/wordpress-2.8.6-ja.zip
$ unzip wordpress-2.8.6-ja.zip
$ mv wordpress mrt
$ cd mrt
$ cp wp-config-sample.php wp-config.php
$ sudo chown deployer:www-data wp-content
$ chmod g+w wp-content
Mr.T

できた。

Mr.T