• 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
 
 

カンファレンス会場で最も貴重な資源は時間ですが、二番目はきっと電源です。 電力というのは普段は簡単に安く手に入る資源であるにもかかわらず、人が集まる場所では極端に供給が難しくなります。 このギャップはとても大きいので、ビジネスになるのではないかと思います。

非常に大雑把な計算ですが、 普通のノートPCを1日動かす程度電力が供給できれば良いので、 20000円程度で市販されている外部バッテリを年間100日レンタルできたとして、平均耐用年数が1年だったとしても、1日あたり200円です。

カンファレンス会場に据え置いてしまえば輸送コストはそれほどかかりません。カンファレンスはほとんどデイタイムに開催されるので、夜間の安い電力で一括充電できます。 一晩に一人で1000個のバッテリを充電できるとすると、時給2000円でも8時間で16000円/日。バッテリ一個あたり16円。 人件費を考慮しても十分安価にサービスを提供できるのではないでしょうか。

posted by Png genki on Fri 17 Jul 2009 at 05:33

Gemを作るのが面倒になってきたので、githubから直接requireできたら楽になるかもしれないと思い、試してみました。

   1  def git(uri, sha1, options = {})
   2    require "tmpdir"
   3    basename = File.basename(uri)
   4    outdir = File.join(Dir.tmpdir, basename, sha1)
   5    unless File.exist?(outdir)
   6      sh = proc{|command| IO.popen("#{command} 2>&1"){|io| io.read}}
   7      sh["git clone #{uri} #{outdir}"]
   8      sh["cd #{outdir}; git checkout #{sha1}"]
   9    end
  10    $:.unshift File.join(outdir, 'lib')
  11    require options[:require_as] || basename.split(/\.git$/)[0]
  12  end

Dir.tmpdirを使ってOSのテンポラリディレクトリの下にリポジトリをクローンしてきて、指定したリビジョンをcheckoutし、LOAD_PATHに"lib"を加えて、リポジトリ名のファイルをrequireしています。

意外と普通にgemの代わりに使える感じです。

posted by Png genki on Thu 16 Jul 2009 at 17:40

Gist はちょっとしたコードの切れ端を貼付けておくのに便利です。 先日紹介したEndless RubyのコードもGist上でのみ公開されていました。 Gemを作るまでもないコードをGistに貼付けて、そのままrequireして使えると便利そうだと思ったので、試してみました。

   1  module Gist
   2  module_function
   3    def require(id, sha1 = "HEAD")
   4      Kernel.require "open-uri"
   5      gist = "http://gist.github.com"
   6      eval(if sha1 == "HEAD"
   7        open("#{gist}/#{id}.txt").read
   8      else
   9        Kernel.require "tmpdir"
  10        cache = File.join(Dir.tmpdir, "gist-#{id}-#{sha1}")
  11        if File.exist?(cache)
  12          open(cache).read
  13        else
  14          open("#{gist}/raw/#{id}/#{sha1}") do |i|
  15            code = i.read
  16            open(cache, "w"){|o| o.write code}
  17            code
  18          end
  19        end
  20      end)
  21    end
  22  end
  23  
  24  Gist.require("148479", "a59ea9ec3e865bafd1e4413b43b6ccb7a38d76fc")
  25  #=> Hello, world!

Hash値で内容の変更が検出可能なので、md5のチェックサムを信用している人にとっては同程度の安全性で利用できそうです。

posted by Png genki on Thu 16 Jul 2009 at 17:31
14th Tue

Endless Ruby

HAMLを使うようになってから、endの無いrubyが欲しい今日この頃。 あわや自作する直前、先人の仕事を見つけました。

[ANN] Endless Ruby 0.0.2

endless.rb is a pre-processor for ruby which allows you to use python-ish indentation to delimit scopes, instead of having to type 'end' every time.

良いですね。残念ながら、現時点ではrubylexerのrequireでエラーが出るようですが、一定の制約付きでもかまわないので動くようになるとありがたいです。 JRubyのCharles Nutterも割と 肯定的なコメント を寄せていますね。

Oh dear god no!

Seriously though...neat :)

-- Charlie

JRubyのオプションに--endlessがつく?

posted by Png genki on Tue 14 Jul 2009 at 17:41

Merbの開発がなかなか進展しないのでSinatraをちょっと触ってみたところ、 DataMapper、Herokuとの組み合わせが非常に簡単でいい感じでした。

blog.rb

   1  require "rubygems"
   2  require "sinatra"
   3  require "dm-core"
   4  require "haml"
   5  
   6  DataMapper::setup(:default, ENV['DATABASE_URL'] || 'sqlite3:db.sqlite3')
   7  
   8  class Post
   9    include DataMapper::Resource
  10    property :id, Serial
  11    property :content, Text
  12    auto_upgrade!
  13  end
  14  
  15  get "/" do
  16    @posts = Post.all(:order => [:id.desc])
  17    haml :index
  18  end
  19  
  20  post "/" do
  21    Post.create(params)
  22    redirect "/"
  23  end
  24  
  25  __END__
  26  @@ index
  27  %h1 Hello, Sinatra!
  28  %ul
  29    - @posts.each do |post|
  30      %li= post.content
  31  %form{:method => :post}
  32    %textarea{:name => :content}
  33    %input{:type => :submit, :value => "Post"}

config.ru

   1  require "blog.rb"
   2  run Sinatra::Application

ローカルで起動するには

   1  % ruby blog.rb

もしくは

   1  % rack

でok. rubyの場合はhttp://localhost:4567、Rackの場合は http://localhost:9292 にアクセス。

これをHerokuにデプロイすると、 http://sinatora-blog.heroku.com/ こんな感じで動きます。

posted by Png genki on Tue 14 Jul 2009 at 15:37 with 2 comments

Tabキーを押すと、Window上に配置されているNSViewのnextKeyViewの間を移動しますが、これを無効にするには、

  1. NSWindowのinitialFirstResponderを適当なView(contentViewなど)に設定する
  2. subviewsのnextKeyViewをself(view自身)にする

これでひとまず目的を実現できるようです。 きっともっと簡単な方法がありそうな気がするのですが、とりあえず。

posted by Png genki on Sat 11 Jul 2009 at 19:33

It's just a demo of Formula :-)


a & = & a + 1 \\
1 & = & 1 + \frac{1}{a} \\
0 & = & \frac{1}{a} \\
a & = & \pm\infty

このブログではデフォルトでeqnarray*環境なので、こんな感じに書けます。

   1  [math]
   2  a & = & a + 1 \\
   3  1 & = & 1 + \frac{1}{a} \\
   4  0 & = & \frac{1}{a} \\
   5  a & = & \pm\infty 
   6  [/math]

posted by Png genki on Sat 11 Jul 2009 at 08:53

RGB値と三刺激値(tristimulus values)XYZの相互変換を行う方法のメモ。

   1  def xyz2rgb(x, y, z)
   2    r =  3.240479 * x - 1.53715  * y - 0.498535 * z;
   3    g = -0.969256 * x + 1.875991 * y + 0.041556 * z;
   4    b =  0.055648 * x - 0.204043 * y + 1.057311 * z;
   5    [r, g, b]
   6  end
   7  
   8  def rgb2xyz(r, g, b)
   9    x = 0.412453 * r + 0.35758  * g + 0.180423 * b;
  10    y = 0.212671 * r + 0.71516  * g + 0.072169 * b;
  11    z = 0.019334 * r + 0.119193 * g + 0.950227 * b;
  12    [x, y, z]
  13  end

posted by Png genki on Mon 6 Jul 2009 at 03:44

Linuxとかではlddコマンドが使えますが、MacOSでは

   1  % otool -L <executable>

とすれば良いようです。

posted by Png genki on Sat 4 Jul 2009 at 10:20

ちょっと趣向をかえて製本の話です。

分厚すぎて持ち歩きにくい技術書などを、章ごとに解体するという事はまれにあると思いますが、いつも困るのが章見出しが見開きの左側に来てる事です。

ss

見出しが見開きの左側に来ている場合、章ごとに本を解体すると、見出しのページの裏に前の章のページがあったりして難しい事になります。 それに対して、見出しが見開きの右側に来ていれば、簡単に章ごとに分解して持ち歩く事が出来るようになります。

posted by Png genki on Sat 27 Jun 2009 at 07:37