eventletでgearman worker

written by shn, on Jan 16, 2010 8:45:00 PM.

eventletっていうライブラリが僕好みの匂いを発していたので遊んでいる。 gearmanを使って、ブログのURLを与えたらそこのFeedのリストを返すってWorkerを書いていたので、それをeventlet化してみよう。

まず、gearmanのeventlet化、gearmanのライブラリにはpure-pythonのやつを使っていたのでさくっと行けた。

green_gearman.py

from eventlet import patcher
from eventlet.green import socket, select

for mod in ['worker', 'connection']:
    patcher.inject('gearman.%s' % mod, globals(),
        ('socket', socket),
        ('select', select)
    )
del patcher

eventletにはpatcherというざっくりしたモジュールがあって、これを使うと指定したモジュールのglobalsを置き換えてくれる。これでいいんかなっていうぐらい簡単。

試しにecho serverを書いてみる。

from eventlet import coros, api
from green_gearman import GearmanWorker

def echo_func(job):
    print api.getcurrent(), job.arg
    time.sleep(5.)

def main():
    pool = coros.CoroutinePool(max_size=200)
    waiters = []

    for idx in range(10):    
        worker = GearmanWorker(['0.0.0.0:4730'])
        worker.register_function('echo', echo_func)

        waiters.append(pool.execute(worker.work))

    for w in waiters:
        w.wait()
main()

gearmanのprotocolは同時に1ジョブしか処理できないっぽいので、とりあえず10個ぐらい突っ込んでみる。 gearman -h localhost -p 4730 -s -f echo hogehogeを20個ぐらい飛ばしてみた所、

<greenlet.greenlet object at 0x804137f30> hogehoge
<greenlet.greenlet object at 0x804127c90> hogehoge
<greenlet.greenlet object at 0x804137db0> hogehoge
<greenlet.greenlet object at 0x804137e10> hogehoge

ってなってるので、なんかまぁそれっぽくなってる気がする。こんなコードじゃよくわからんな… これで良いのでしょうか。

さて肝心のWorkerもこれにのせたいのだが、urllib2を使っていたので、代わりにfrom eventlet.green import urllib2に代えるだけっぽい(まだやってない…)。

んでつらつらコード見てたんだが、今のところ(eventlet 0.9.2)はlibeventサポートがdisabledされているようで、ちょっとがっかり。

eventlet化する場合、全てのIOをこれにしないとブロックが発生してみんな止まることになっちゃうのだけど、MySQL周りとかをeventlet化するのはめんどくさそうなのでアレでアレしてアレですね。

岡嶋さんの問題を解いたけど、採用されなさそう

written by shn, on Jan 12, 2010 1:13:00 AM.

去年の暮れにほってんとりになってた採用問題が公開されたのでやってみた。

というのも、当初問題文を読み違えていて(格子マップじゃなくて、連続的なマップだと思ってた)、「各経路点を取って最短経路問題ですかねぇ」とか「僕だったら、答えをASCIIにプロットするのに25分かかるなぁ」とか頓珍漢な事を言っていたのが恥ずかしかったからです。 問題文を読んでないので応募以前の問題ですね。ここらへんを参照

んで、最適解法を知った上でコード書いたのに40分かかりました…

さて、僕自身も経営者の端っこなので、いろいろ思う点がありました。

まず、こういうアルゴリズム系の問題を出すことでGoogleの使い方を知らない奴を簡単に落とせるのがわかったのは参考になりました。 というのも、問題解決について求められるのはその時点での知識ではなく、最終的に許される時間の中で解決できるかという点だからです。 また、実務をこなす上でのほとんどの細かい問題に関しては、それまで世界の誰もそんな問題に直面したことが無い!なんて事は殆どないはずなので、そういうった問題を確実に解けるのか?というのは重要なポイントだと思います。

また、話には聞いていましたが、それぐらいの人材が我々のような(勝手に一緒にしちゃいましたが)零細IT企業にふらっと応募することは少ないのだなというのも参考になります。そういう点で、公募する場合には根気よく採用を続ける必要がありそうなので、あんまり応募者をディスると今後の応募に影響してしまいそうな点は気になります。

このエントリに応えると、小さな企業が人を取った場合、新人の戦力が30で今の戦力と足して130だ!、というよりも、新人がしちめんどくせーことやってくれるお陰で俺らが強まって足して150だ!みたいな気概でいるのがいいんじゃないのかなーと思います。

イメージとしてはこんな感じ

以下がコード。Pythonです。:

#!/usr/bin/env python
# -*- coding:utf-8 -*-

import sys

CELLMAX = 9999

def main(input):
    m, start, goal = load_map(input)
    solve(m, start, goal)

    print_result(m, goal)

def load_map(input):
    start, goal = [], None

    d = {
        '*': -1,
        ' ': CELLMAX,
        'S': 0
    }
    m = []
    for l in input:
        m.append([d.get(ch, ch) for ch in l.rstrip()])

    for y in range(len(m)):
        for x in range(len(m[y])):
            if m[y][x] == 0:
                start.append((x, y))
            elif m[y][x] == 'G':
                goal = (x, y)

    return m, start, goal

def solve(m, start, goal):
    cur = 0
    while start:
        next = []

        for c in start:
            m[c[1]][c[0]] = cur

            for d in [(-1, 0), (0, -1), ( 1,  0), ( 0, 1)]:
                if m[c[1]+d[1]][c[0]+d[0]] > cur:
                   next.append((c[0]+d[0], c[1]+d[1]))
        start = next
        cur += 1
        if goal in start:
            m[goal[1]][goal[0]] = cur # syoboi
            return

    if goal not in start:
        raise RuntimeError, u'とけないよ'

def fill(m, c, x):
    filled = []
    for d in [(-1, 0), (0, -1), ( 1,  0), ( 0, 1)]:
        if m[c[1]+d[1]][c[0]+d[0]] > x:
           m[c[1]+d[1]][c[0]+d[0]] = x
           filled.append((c[0]+d[0], c[1]+d[1]))
    return filled

def print_result(m, goal):
    c = goal
    while True:
        check = m[c[1]][c[0]] - 1
        if check == -1:
            break

        m[c[1]][c[0]] = '$'
        for d in [(-1, 0), (0, -1), ( 1,  0), ( 0, 1)]:
            if m[c[1]+d[1]][c[0]+d[0]] == check:
               c = (c[0]+d[0], c[1]+d[1])
               break
        else:
            raise RuntimeError
    m[goal[1]][goal[0]] = 'G' # syoboi
    d = {
       -1: '*',
        0: 'S',
       '$': '$',
       'G': 'G',
    }
    for l in m:
       print ''.join(d.get(ch, ' ') for ch in l)    

if __name__=='__main__':
    main(sys.stdin)

「最短性のチェック」というのは、(コードの外で)アルゴリズムの証明をせよ、という意味だろうか?基本的にダイクストラ法を格子マップでやっているので、最短性はダイクストラ法の証明となると思うのだけど。

って岡嶋さんてスウィフトの関係者なのか!

あとはっちゅう君バジョナップ期待しています!

EC2 US-Westを試す

written by shn, on Jan 3, 2010 12:38:00 AM.

弊社サービスを引っ越ししたりせなあかんなーと思いつつ、US-West初め。 いくつかはまった。

S3 Bucket ExplorerがUS-Westに対応していない。

AMI作るときにはまった。 FirefoxのAddonのアレがUS-Westに対応しておらず、うっかりBucketを作るとUS-Eastに行っちゃう。 コマンドラインから頑張るのが面倒だったので、Bucket Explorerとかいうのを使って作成。

--region us-westを付ける。

API-Toolの引数にus-westを付ける。付けないとeastにデフォルトで行くのかなこれ。 export EC2_URL=https://us-west-1.ec2.amazonaws.comにするとデフォでwest見に行くのかな。試してはいない。

UQ WiMAX契約したよ

written by shn, on Dec 17, 2009 1:18:00 AM.

UQ WiMAXの一日プランに契約した。端末はUD03NA

会社
麻布十番の位置の橋交差点の道路沿い。つながる。電波強度は小〜中
居間も窓際も圏外。場所は港区の有栖川公園から一本入ったところ。
原宿の某社(地上19階)
窓際行くとつながった。電波微弱。でもtwitter/メールには十分な速度
汐留のWINSの麓(イタリア街?)
電波中
その近くの某社
電波弱. 下り0.7Mbps

聞いていた通り、電波弱い。弱くても満足行く通信速度が出るのが救いか

香港・深圳・澳門の旅4日間

written by shn, on Nov 26, 2009 3:48:00 AM.

ホーテンス・S・エンドウ氏にお連れ願い、電脳空間カウボーイズと共に香港・深圳・澳門に食い倒れ、変携帯探しの旅に行って参りました。

写真はこちら

各人の旅行記は以下のあたり

遠藤諭の東京カレー日記:

シン石丸の電脳芸事ニッキ:

Keep Crazy;shi3zの日記:

ガイドをしてくださった山根さんのサイト "山根康宏オフィシャルサイト - 香港携帯情報局" 山根さんありがとうございました! 至高の変ケータイを手に入れた山根さんの嬉しそうな顔が忘れられません。

香港の携帯文化、深圳の圧倒的なエネルギーに関しては皆さんがお書きになっているのでそれ以外の話でも書こうと思います。

金とかの話

2009-11-21 12-22-22

とにかく軍資金が無いとはじまりません。日本で両替しちゃうと札束を持って行くことになり、ちょっとそれは僕にはデンジャラスなので他の方法を探してみました。また今回のは買い出し旅行なのでレートも重要です。

レートから言うと一般的に、日本で両替 > 現地空港で両替 >> 現地市中で両替≒クレジットカードのキャッシング、といわれています。 今回の旅の場合、1HKDあたり市場のレートが11.4に対し、成田空港 13.9 JPY、現地空港 12.5 JPY 街中 11.53 JPYでした。なので成田空港のボッタクリ具合がよくわかると思います。

キャッシングの場合は他と違い、「決済時点での市場のレート」+「引き落としまでの金利」になります。今回の場合、ずっと円高が進行していたので、11.636 - 11.251で決裁されていました。キャッシングの場合はドルコスト平均法的に、日々の為替変動に対してトータルで損しない。という事も言えると思います。金利に関しては引き落としまでが一カ月未満な事を考えると両替手数料と大した差は出ません。

以上の事を考えると、現金を持ち歩かなくて良いという利点もありキャッシングが一番良いという事になります。ATMはそこらじゅう(例えば駅とか)にある上だいたい24時間営業なので、夜中でも現金が手に入る利点があります。

2009-11-21 12-03-49

ただ、今回の場合は両替・インド人・安宿で有名な重慶大厦に行きたかったので、現地で2万円を両替しました。重慶大厦の1,2Fにはインド人経営の両替商がたくさん詰まっており、店によって微妙にレートが違います。入口近くのお店が一番レートが高く、2Fをちょっと行ったところにあるお店が一番良かったです(上の写真のお店)

クレジットカードメインの旅行にする場合、

  • 限度額をあげておく。

現地からでも電話であげられます

  • キャッシング枠があるか確認しておく。

今回はメインのカードに枠を設定してなくて焦りました。日本では借金は禁忌されるので枠を設定していない人も多いと思います。

  • 決済口座の違う、MasterとVISAの2枚を持っておく。

海外ではクレジットカードを、ホテルのチェックイン、自転車のレンタルなどにおいてデポジットに使われる事があります。デポジットとは例えばレンタルするときにあらかじめ引き落としておいて、返却されたときにそのトランザクションをキャンセルすることで、客がバックれても大丈夫なようにすることです。 というわけで善良に生きてるなら特に問題ないのですが、昔、アメリカ旅行をしたときにロードバイクを借りたら1000ドル程デポジットされ、限度額を突破しそのあと面倒な事になったことがあります…

今回の旅では、深圳に行ったのですが、中国ではクレジットカードはあまり使えないようです。次に行く時は銀嶺カードを用意すべきかもしれんと思いました。

あと先達広場ではクレカだと5%ぐらいの手数料を要求されました。

言語とかの話

2009-11-21 10-27-19

香港は広東語、深圳は普通語(北京語)らしいのですが、どっちも等しく何言ってるかわからない。数字ぐらいしかわかりません(麻雀のおかげだ!) 広東語の方が発音が難しいそうです。

英語(根性英語)は香港では通じました。深圳・澳門はさっぱり。 深圳はカウボーイズが携帯を買う事に決めたから「ok! ok!」と言ってるのに、okが通じず延々と営業されて困った、なんて話もありました。

香港でちょっと困ったのは、「Cityview Hotel」という所に泊ったのですが、TAXIの運ちゃんにはそれが通じませんでした。「YMCA」という通名の方を言わないとだめなようです。他のホテルも中国語名の方しかしらない事があると思うので、タクる場合は調べておく必要があるかも。

ガジェットとか

香港の地下鉄(MTR)では、PSPをやっている人をよく見かけました。ちら見したら日本語のゲームをやっていたので驚き。NDSは見ませんでした。

携帯電話だと、Nokiaの細長い端末を良く見かけました。スマートフォンだとiPhone, BlackBerryあたり。Androidは見かけず。香港の地下鉄は電波が入るので驚き。

2009-11-23 14-53-23

MTRの駅にはデジタルサイネージがありました。こういうの増えて行くんでしょうね。もっと驚いたのは深圳で、広場に 5m * 30m ぐらいの巨大なデジタルサイネージが。この国はなんでもスケールがでかい。

深圳のPCパーツショップは、インクジェットプリンタ用の詰め替えインクがボトルで売ってたりして面白かったのですが、PCパーツ自体はあまり面白いのはありませんでした。日本で売ってる物がちょっと安いぐらい。MBは半額ぐらいだったかも。ハイエンドのVGAなんかも見かけましせんでした。やたらフォトフレーム売ってましたね。SANYOのものもあったけど本物なのかしら

澳門

2009-11-23 22-14-32

カウボーイズたちは前回行ったそうなので僕一人だけいきました。感想は「場末のラスベガス」。マカオグランプリの後片付けが終わっておらず、道路がレース場だったのがちょっとおもしろかったですね。 カジノは金をシュレッダーにかけているような感じでした。

CSSツールをうpした。 - スプライト作成、ネスト、変数

written by shn, on Nov 14, 2009 4:35:00 PM.

Python Hack-a-thonに来たので、昔作ったツールのメンテをしたりしてました。 んで、githubにうpした。

githubのプロジェクトはこちら: github cssc.git

  • YAMLで段組を定義して、CSSスプライトを生成する make_sprite.py
  • IE6のためにPNGをなるたけ綺麗にGIFにする to_gif.py
  • CSSの記述に、ネストや、変数などを使えるようにするコンバータ cssc.py

で構成されてます。

make_sprite.py

YAMLの定義ファイルからCSSスプライトを生成します。 CSSスプライトっていうのは、GoogleのCSSスプライトのような奴で、複数の画像を一枚にまとめて、リクエスト回数、ファイルサイズを節約します。

CSSスプライトを使用すると、タグの代わりに<div>などの要素にbackgroundを指定することのなります。 CSSスプライトの中で画像の座標を調べるのは面倒なので、そこらへんの生成はcssc.pyを通して自動化しています。

使い方

make_sprite.py -o sprite.png -c sprite.json -b ./images/ sprite.yaml
  • -o sprite.png 出力CSSスプライト名
  • -c sprite.json 座標定義ファイル。あとでcssc.pyで使います。
  • -b ./images/ 画像のベースディレクトリ
  • sprite.yaml 画像の段組を記載したyaml

yamlの記法

段組はyamlで記述します。

冗長に書くと、以下のような感じです。

direction: vertical
images:
  - direction: horizontal
    images:
      - left-arrow.png
      - google.png
      - right-arrow.png
  - direction: horizontal
    images:
      - google-large.png
      - direction: vertical
        images:
          - down.png
          - up.png
          - close.png

基本的に directionとimagesのマップのネストで段組を表現します。directionは horizontal もしくは vertical 、imagesは画像ファイル名もしくは、マップのリストになります。

directionを省略して、直接子イメージのリストを書くことも出来ます。その場合の方向は、親の方向と逆になります。 この省略記法を利用すると、上のyamlは以下のようになります。

direction: vertical
images:
    - [left-arrow.png, google.png, right-arrow.png]
    - [google-large.png, [down.png, up.png, close.png]]

to_gif.py

pngをgifにします。

使い方は to_gif.py in.png out.gif 

cssc.py

CSS手書き派は日々、ネストができないことや、変数が使えないことに憤慨しています。

つまりこういう書き方がしたい!

{% set header_height = '24px' %}
{% set main_color = '#001122' %}

#site-header {
    height: {{ header_height }};

    h1#logo {
        font-size: 24px;
        color: {{ main_color }};
    }

    a {
        color: #aabbcc;

        _:hover {
            border: 1px solid #ffeedd;
        }
    }
}

OK! それなら、そうすればよいわ

./cssc.py -o out.css in.cssc

in.csscにみょうちきりんなcssもどきを書くと、in.cssにちゃんとしたものをはいてくれます。

上のmake_sprite.pyで、座標ファイルを出力した場合は以下のような感じで使います。

./cssc.py --coords main,http://static.com/main.png -o out.css in.css

--coordsオプションに
スプライト名:スプライトのURLを指定します。 座標はスプライト名.jsonから読み込みます。

csscファイルの方は、以下のように。冗長

h1#logo {
    {{ sprite_background('main', 'logo.png') }}
}

sprite_background関数の第1引数にスプライト名(コマンドラインで指定する奴)、第2引数に元画像ファイル名を指定します。

cssc詳細

{% set width = '100px' %} /* 変数などはJinja2そのまんまなので[Jinja2のドキュメント](http://jinja.pocoo.org/2/documentation/)を見るのが良いと思います。 */
// C++スタイルのコメントも使えます
{# jinja2スタイルも使えるぜ... #} 

div {
    /* 普通にcssのスタイル定義を書きます */
    border: 1px solid black;
    margin: 4px 8px;

    width: {{ width }}; // これはjinja2

    span {
        /* ネストすると "div span" になります。 */
        font-weight: bold;
    }

    a {
        text-decoration: none;

        _:hover {
            // 親要素とスペース無しで続け対場合は '_' を使います。
            // この場合、このブロックは "div a:hover" になります。
            text-decoration: underline;
        }
    }
}

あとがき

  • いまどきCSSを手書きしている奴はいるのだろうか
  • でもYUI compressorとか使うならMakeで処理できるようにしておきたいよね。
  • みんなここらへんの地味な所どうしてんだろう。
  • PyParsingはじめてトライしてみたんだけど遅い… たぶん文法定義が悪い

WiiDiet 0.1

written by shn, on Nov 11, 2009 9:55:00 PM.

WiiFitのボードに乗ったら自動的に体重ログってTwitterに書いてくれるツールが見当たらないので作りました。 Mac用です。 WiiRemoteFrameworkを使わせていただきました。

WiiDiet 0.1 Screenshot

Download WiiDiet 0.1

使い方

  1. "Bluetooth設定アシスタント"でバランスボードを登録する。 初めてWiiリモコンを仕様する場合を参考にするとよいかも

  2. "WiiDiet"を起動させる。

  3. "Discoveringうんたら"と出てたら、バランスボードのSyncボタンを押す

  4. "Balanceboard found"とか出たら、上に乗る。

  5. Send tweetボタンでTwitterに発言

設定から"Automatically"的なのをチェックしておくと、体重が安定した時点で自動的に送信します。

アレげな所

  • 体重がちょっとアバウト(+5kgぐらい多めにでたり)
  • なんで毎回Sync押さないといけないの? だれか良い方法ありましたら教えてください。

Rackspace Cloudを使ってみた。

written by shn, on Aug 25, 2009 8:02:00 PM.

メモリ1GBプラン?を借りる $0.06/h なので、ひと月$45ぐらいかな

pingは150msぐらい。

ダメスギはこんな感じ

[    0.000000] Linux version 2.6.24-23-xen (buildd@crested) (gcc version 4.2.3 (Ubuntu 4.2.3-2ubuntu7)) #1 SMP Mon Jan 26 03:09:12 UTC 2009 (Ubuntu 2.6.24-4.6-generic)
[    0.000000] Command line: root=/dev/sda1 ro xencons=tty console=tty1 clocksource=jiffies 
[    0.000000] BIOS-provided physical RAM map:
[    0.000000]  Xen: 0000000000000000 - 0000000040800000 (usable)
[    0.000000] Entering add_active_range(0, 0, 264192) 0 entries of 256 used
[    0.000000] end_pfn_map = 264192
[668358.384872] Entering add_active_range(0, 0, 264192) 0 entries of 256 used
[668358.387987] Zone PFN ranges:
[668358.387989]   DMA             0 ->     4096
[668358.387992]   DMA32        4096 ->  1048576
[668358.387993]   Normal    1048576 ->  1048576
[668358.387995] Movable zone start PFN for each node
[668358.387996] early_node_map[1] active PFN ranges
[668358.387998]     0:        0 ->   264192
[668358.388001] On node 0 totalpages: 264192
[668358.393477]   DMA zone: 56 pages used for memmap
[668358.393483]   DMA zone: 0 pages reserved
[668358.393485]   DMA zone: 4040 pages, LIFO batch:0
[668358.393902]   DMA32 zone: 3556 pages used for memmap
[668358.393905]   DMA32 zone: 256540 pages, LIFO batch:31
[668358.421618]   Normal zone: 0 pages used for memmap
[668358.421627]   Movable zone: 0 pages used for memmap
[668358.421659] No mptable found.
[668358.424600] PERCPU: Allocating 22368 bytes of per cpu data
[668358.424663] Built 1 zonelists in Zone order, mobility grouping on.  Total pages: 260580
[668358.424669] Kernel command line: root=/dev/sda1 ro xencons=tty console=tty1 clocksource=jiffies 
[668358.425368] Initializing CPU#0
[668358.425568] PID hash table entries: 4096 (order: 12, 32768 bytes)
[668358.425616] Xen reported: 1994.998 MHz processor.
[    0.093560] console [tty1] enabled
[    0.093602] Console: colour dummy device 80x25
[    0.093606] console [tty1] enabled
[    0.094084] Dentry cache hash table entries: 131072 (order: 8, 1048576 bytes)
[    0.094496] Inode-cache hash table entries: 65536 (order: 7, 524288 bytes)
[    0.094627] Software IO TLB disabled
[    0.103167] Memory: 946140k/1056768k available (2533k kernel code, 101908k reserved, 1331k data, 220k init)
[    0.169046] Calibrating delay using timer specific routine.. 4001.13 BogoMIPS (lpj=8002266)
[    0.169114] Security Framework initialized
[    0.169125] SELinux:  Disabled at boot.
[    0.169135] AppArmor: AppArmor initialized
[    0.169141] Failure registering capabilities with primary security module.
[    0.169165] Mount-cache hash table entries: 256
[    0.169325] CPU: L1 I Cache: 64K (64 bytes/line), D cache 64K (64 bytes/line)
[    0.169333] CPU: L2 Cache: 512K (64 bytes/line)
[    0.169341] CPU: Physical Processor ID: 0
[    0.169344] CPU: Processor Core ID: 1
[    0.169364] SMP alternatives: switching to UP code
[    0.170091] Early unpacking initramfs... done
[    0.252310] Brought up 1 CPUs
[    0.252328] CPU0 attaching NULL sched-domain.
[    0.252986] net_namespace: 120 bytes
[    0.253002] failed to set up cpufreq notifier
[    0.270099] Time: 165:165:165  Date: 165/165/65
[    0.270190] NET: Registered protocol family 16
[    0.271737] CPU0 attaching NULL sched-domain.
[    0.287158] SMP alternatives: switching to SMP code
[    0.287791] CPU0 attaching sched-domain:
[    0.287795]  domain 0: span 03
[    0.287796]   groups:Initializing CPU#1
[    0.287803]  01 02
[    0.287811] CPU1 attaching sched-domain:
[    0.287813]  domain 0: span 03
[    0.287814]   groups: 02 01
[    0.000059] CPU: L1 I Cache: 64K (64 bytes/line), D cache 64K (64 bytes/line)
[    0.000073] CPU: L2 Cache: 512K (64 bytes/line)
[    0.000080] CPU: Physical Processor ID: 0
[    0.000084] CPU: Processor Core ID: 1
[    0.288058] CPU0 attaching NULL sched-domain.
[    0.288060] CPU1 attaching NULL sched-domain.
[    0.298986] CPU0 attaching sched-domain:
[    0.298987]  domain 0: span 07
[    0.298989]   groups: 01 02 04
[    0.298991] CPU1 attaching sched-domain:
[    0.298992]  domain 0: span 07
[    0.298994]   groups: 02 04 01Initializing CPU#2
[    0.299001] 
[    0.299003] CPU2 attaching sched-domain:
[    0.299006]  domain 0: span 07
[    0.299007]   groups: 04 01 02
[    0.000072] CPU: L1 I Cache: 64K (64 bytes/line), D cache 64K (64 bytes/line)
[    0.000081] CPU: L2 Cache: 512K (64 bytes/line)
[    0.000087] CPU: Physical Processor ID: 0
[    0.000091] CPU: Processor Core ID: 1
[    0.299180] CPU0 attaching NULL sched-domain.
[    0.299182] CPU1 attaching NULL sched-domain.
[    0.299184] CPU2 attaching NULL sched-domain.
[    0.310812] CPU0 attaching sched-domain:
[    0.310814]  domain 0: span 0f
[    0.310815]   groups:Initializing CPU#3
[    0.310818]  01 02 04 08
[    0.310823] CPU1 attaching sched-domain:
[    0.310824]  domain 0: span 0f
[    0.310826]   groups: 02 04 08 01
[    0.310829] CPU2 attaching sched-domain:
[    0.310830]  domain 0: span 0f
[    0.310832]   groups: 04 08 01 02
[    0.310839] CPU3 attaching sched-domain:
[    0.310840]  domain 0: span 0f
[    0.310841]   groups: 08 01 02 04
[    0.000053] CPU: L1 I Cache: 64K (64 bytes/line), D cache 64K (64 bytes/line)
[    0.000060] CPU: L2 Cache: 512K (64 bytes/line)
[    0.000065] CPU: Physical Processor ID: 0
[    0.000068] CPU: Processor Core ID: 1
[    0.310886] Brought up 4 CPUs
[    0.311247] PCI: Fatal: No config space access function found
[    0.311255] PCI: setting up Xen PCI frontend stub
[    0.312098] ACPI: Interpreter disabled.
[    0.312108] Linux Plug and Play Support v0.97 (c) Adam Belay
[    0.312142] pnp: PnP ACPI: disabled
[    0.312362] xen_mem: Initialising balloon driver.
[    0.314883] Setting mem allocation to 1048576 kiB
[    0.315062] PCI: System does not support PCI
[    0.315069] PCI: System does not support PCI
[    0.322580] NET: Registered protocol family 8
[    0.322588] NET: Registered protocol family 20
[    0.322698] AppArmor: AppArmor Filesystem Enabled
[    0.323172] NET: Registered protocol family 2
[    0.358129] IP route cache hash table entries: 32768 (order: 6, 262144 bytes)
[    0.358500] TCP established hash table entries: 131072 (order: 9, 2097152 bytes)
[    0.359437] TCP bind hash table entries: 65536 (order: 8, 1048576 bytes)
[    0.359943] TCP: Hash tables configured (established 131072 bind 65536)
[    0.359952] TCP reno registered
[    0.370028] checking if image is initramfs... it is
[    0.460664] Freeing initrd memory: 76412k freed
[    0.185337] audit: initializing netlink socket (disabled)
[    0.185369] audit(1251197576.406:1): initialized
[    0.185902] VFS: Disk quotas dquot_6.5.1
[    0.185944] Dquot-cache hash table entries: 512 (order 0, 4096 bytes)
[    0.186056] io scheduler noop registered
[    0.186062] io scheduler anticipatory registered
[    0.186066] io scheduler deadline registered
[    0.186098] io scheduler cfq registered (default)
[    0.189239] Xen virtual console successfully installed as tty1
[    0.189319] Event-channel device installed.
[    0.196374] Successfully initialized TPM backend driver.
[    0.213651] netfront: Initialising virtual ethernet driver.
[    0.534309] xen-vbd: registered block device major 8
[    0.534359] blkfront: sda1: barriers enabled
[    0.253875] blkfront: sda2: barriers enabled
[    0.566569] rtc: IRQ 8 is not free.
[    0.566664] Linux agpgart interface v0.102
[    0.567264] RAMDISK driver initialized: 16 RAM disks of 65536K size 1024 blocksize
[    0.567337] input: Macintosh mouse button emulation as /devices/virtual/input/input0
[    0.567446] PNP: No PS/2 controller found. Probing ports directly.
[    0.568305] i8042.c: No controller found.
[    0.574007] mice: PS/2 mouse device common for all mice
[    0.574050] cpuidle: using governor ladder
[    0.574186] NET: Registered protocol family 1
[    0.574262] registered taskstats version 1
[    0.574281] XENBUS: Device with no driver: device/console/0
[    0.574301]   Magic number: 1:252:3141
[    0.574423] /build/buildd/linux-2.6.24/debian/build/custom-source-xen/drivers/rtc/hctosys.c: unable to open rtc device (rtc0)
[    0.574437] Freeing unused kernel memory: 220k freed
[    1.101172] Serial: 8250/16550 driver $Revision: 1.90 $ 4 ports, IRQ sharing enabled
[    1.689595] loop: module loaded
[    1.734227] fuse init (API version 7.9)
[    1.508563] thermal: Unknown symbol acpi_processor_set_thermal_limit
[    1.553625] device-mapper: uevent: version 1.0.3
[    1.553703] device-mapper: ioctl: 4.12.0-ioctl (2007-10-02) initialised: dm-devel@redhat.com
[    1.594329] md: linear personality registered for level -1
[    1.606941] md: multipath personality registered for level -4
[    1.619622] md: raid0 personality registered for level 0
[    1.645283] md: raid1 personality registered for level 1
[    1.641772] xor: automatically using best checksumming function: generic_sse
[    1.658048]    generic_sse:  6129.000 MB/sec
[    1.658055] xor: using function: generic_sse (6129.000 MB/sec)
[    1.658888] async_tx: api initialized (async)
[    1.725313] raid6: int64x1   2055 MB/s
[    1.792632] raid6: int64x2   2592 MB/s
[    1.859932] raid6: int64x4   2603 MB/s
[    1.926625] raid6: int64x8   2004 MB/s
[    1.993496] raid6: sse2x1    2655 MB/s
[    2.060584] raid6: sse2x2    4740 MB/s
[    2.128145] raid6: sse2x4    4399 MB/s
[    2.128174] raid6: using algorithm sse2x2 (4740 MB/s)
[    2.128185] md: raid6 personality registered for level 6
[    2.128189] md: raid5 personality registered for level 5
[    2.128193] md: raid4 personality registered for level 4
[    2.207428] md: raid10 personality registered for level 10
[   12.425011] kjournald starting.  Commit interval 5 seconds
[   12.713647] EXT3-fs: mounted filesystem with ordered data mode.
[   13.559085] usbcore: registered new interface driver usbfs
[   13.559117] usbcore: registered new interface driver hub
[   13.559152] usbcore: registered new device driver usb
[   14.201891] udev: starting version 141
[   14.611203] evbug.c: Connected device: "Macintosh mouse button emulation", <NULL>
[   15.676061] SCSI subsystem initialized
[   15.890948] EXT3 FS on sda1, internal journal
[   16.296229] Adding 2097144k swap on /dev/sda2.  Priority:-1 extents:1 across:2097144k
[   16.582728] ip_tables: (C) 2000-2006 Netfilter Core Team
[   16.673512] nf_conntrack version 0.5.0 (16384 buckets, 65536 max)
[   16.841378] NET: Registered protocol family 10
[   16.841659] lo: Disabled Privacy Extensions
[   17.181698] NET: Registered protocol family 17
[   27.340132] eth0: no IPv6 routers present
[   32.485839] eth1: no IPv6 routers present
[  706.329680] udev: starting version 141

FAPWS3でSet-Cookieが出来ないっていう

written by shn, on Aug 10, 2009 2:58:00 AM.

今日は残ってる仕事をしに会社に行ったはずが、Zineのpluginとか作って遊んでたら一日が終わった。 そんななかで子一時間はまったのが、FAPWS3がSet-Cookieヘッダを食っちゃうって話。

FAPWS3駆動にしたZineでadmin panelにログインできないなー、おかしいなーと奮闘してたらコレが原因だった。

Unable to set cookie with Fapws3-0.2 not Set-Cookie headers send :(

fapws.baseのstart_responseが、Set-Cookieを食っちゃってるのが原因なので、cookie関連のコードを全部消しちまえばちゃんとzineが動くようになる。

なんでstart_response.set_cookie()なんてメソッドが用意されてんのかよくわからないけど、作者の勘違いかな。

以下パッチ:

--- base.py 2009-08-09 19:37:49.587425000 +0900
+++ fapws_base.py   2009-08-09 18:38:08.873789000 +0900
@@ -1,17 +1,13 @@
 # -*- coding: utf-8 -*-

 import datetime
-from Cookie import SimpleCookie, CookieError
 try:
     import cStringIO as StringIO
 except ImportError:
     import StringIO
 import traceback, sys, string

 status_reasons = {
     100: 'Continue',
@@ -90,7 +86,6 @@
         self.status_reasons = "OK"
         self.response_headers = {}
         self.exc_info = None
-        self.cookies = SimpleCookie()
         # NEW -- sent records whether or not the headers have been send to the
         # client
         self.sent= False
@@ -111,36 +106,11 @@
         key=str(key)
         val=str(val)
         self.response_headers[key]=val
-    def set_cookie(self, key, value='', max_age=None, expires=None, path='/', domain=None, secure=None):
-        self.cookies[key] = value
-        self.response_headers['Set-Cookie'] = self.cookies
-        if max_age:
-            self.cookies[key]['max-age'] = max_age
-        if expires:
-            if isinstance(expires, str):
-                self.cookies[key]['expires'] = expires
-            elif isinstance(expires, datetime.datetime):
-                expires = expires.strftime('%a, %d %b %Y %H:%M:%S GMT')
-            else:
-                raise CookieError, 'expires must be a datetime object or a string'
-            self.cookies[key]['expires'] = expires
-        if path:
-            self.cookies[key]['path'] = path
-        if domain:
-            self.cookies[key]['domain'] = domain
-        if secure:
-            self.cookies[key]['secure'] = secure
-    def delete_cookie(self, key):
-        if self.cookies:
-            self.cookies[key] = ''
-        self.cookies[key]['max-age'] = "0"
+
     def __str__(self):
         res = "HTTP/1.1 %s %s\r\n" % (self.status_code, self.status_reasons)
         for key, val in self.response_headers.items():
-            if key.upper() != "SET-COOKIE":
-                res += '%s: %s\r\n' % (key,val)
-        if self.cookies:
-            res+=str(self.cookies)+"\r\n"
+            res += '%s: %s\r\n' % (key,val)
         res += "\r\n"
         return str(res)

SQLAlchemyメモ

written by shn, on Aug 5, 2009 1:49:00 PM.

毎回ググるので、メモっておく。 このブログがSEOされて、このエントリが上位に来れば、次ググる時楽になる寸法だ!

INSERT IGNORE INTO ...

insert(..., prefixes=['IGNORE'])

SELECT ... FOR UPDATE

select(..., for_update=True)

随時更新したい。