あくまで中立な実況。
ポーニョポーニョポニョさかなの子ー♪
一級ブロリスト。
あくまで中立な実況。
ポーニョポーニョポニョさかなの子ー♪
一級ブロリスト。
__init__と__call__ってどう違うのかなと思ったけどわかった。
class Foo:
def __init__(self):
print 'init'
def __call__(self):
print 'call'
Foo()()
init
call
__init__はクラスがインスタンス化される時に呼ばれて、__call__はインスタンスが呼ばれた時に動くんですね。
この間書いた、RubyのFoo()とか使ってると混乱するわ!
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なんて予約語無いじゃないか!」
とか迷ったので明示的なのも分かりやすくて良いですね。
$ 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
$ 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化の恩恵は少なくなさそうです。
![]() |
|
職場はsvnで個人的にgitを使うときなど、全部持ってくると死ぬほど時間がかかるので最新のtrunkだけ持ってくる。
% git svn clone -rHEAD https://svn.example.com/unk/trunk unk
% git svn rebase
% git svn dcommit
% git svn dcommit --rmdir
-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
% git reset --soft HEAD^
% git ci -a -m 'やりなおし'
% git push origin master --force
% git push origin win32localに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: 実行時設定 - Manualmysql.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_errosをOnにした場合、エラー抑制子を外すと、データベースに接続できない場合はこんな感じになる。
たしかにちょっと怖い感じがするかも。初めてのインストールでこれが出たら諦めちゃう人も多いかも。
% 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
/opt/local/etc/php5/php.ini
[Date]
date.timezone = Asia/Tokyo
/opt/local/etc/php5/php.ini
[MySQL]
mysql.default_socket = /opt/local/var/run/mysql5/mysqld.sock
portsで入れたphpとmysqlそれぞれの見に行くsockの場所のdefaultが違う。
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('.'))
ここも素敵。元のコードだとピリオドが一つ以上あるファイル名でおかしなことになりますね。
同じ問題をうまい人にも解いてもらってそれを見るのはとても勉強になりますね。プログラマの教育の話でも目の前で解いて見せるという話がありましたが、とても効率的な気がしました。
日々ブログに恥を垂れ流してますが時々こういう良い事があるので嬉しいですね。ありがとうございます!