2011-03-27

ruby環境構築メモ - fastri編

いまさらだけど、そろそろまじめにrubyの開発環境を整えようと思ったのでとりあえずfastriを導入した。
5分でおわると思いきや地味に苦労したので作業履歴メモ






手順


  1. 最新版のソースをおとす
  2. 展開してsetup.rbを実行
  3. 'fastri-server -b' でインデックス作成
  4. 'fastri-server -B' でフルテキストのインデックス作成
  5. 'fastri-server' で起動
  6. qri Array みたいなかんじで使う

ところがqriを使おうとすると以下のようなエラーが。。


/usr/lib/ruby/1.8/rdoc/ri/ri_paths.rb:61: uninitialized constant Gem::Version (NameError)
from /usr/lib/ruby/1.8/rdoc/ri/ri_paths.rb:57:in `each'
from /usr/lib/ruby/1.8/rdoc/ri/ri_paths.rb:57
from /usr/local/lib/site_ruby/1.8/fastri/util.rb:38:in `require'
from /usr/local/lib/site_ruby/1.8/fastri/util.rb:38
from /usr/bin/qri:6:in `require'
from /usr/bin/qri:6

色々試した結果、以下のようにfastri/util.rbを書き換えたらとりあえず動くようになった。


--- /home/takayuki/tmp/util.rb  2011-03-27 01:25:51.000000000 +0900
+++ /usr/local/lib/site_ruby/1.8/fastri/util.rb 2011-03-27 01:22:55.000000000 +0900
@@ -35,7 +35,7 @@
# don't let rdoc/ri/ri_paths load rubygems.rb, that takes ~100ms !
emulation = $".all?{|x| /rubygems\.rb$/ !~ x} # 1.9 compatibility
$".unshift "rubygems.rb" if emulation
-require 'rdoc/ri/ri_paths'
+#require 'rdoc/ri/ri_paths'
$".delete "rubygems.rb" if emulation
require 'rdoc/ri/ri_writer'

ちなみに環境は以下の通りです. OSはdebian Lenny


% gem --version
1.6.2
% ruby --version
ruby 1.8.7 (2008-08-11 patchlevel 72) [x86_64-linux]




TODO


  • rvmを導入する

2011-03-22

twitterやfacebookのOAuthをつかってrails+omniauthでログイン機能を実装するメモ その2

前回(http://taksatou.blogspot.com/2011/03/twitterfacebookrails.html) 、OAuthで認証するところまでできたので、今回はtwitterアカウントでログインするところを作ります。

omniauthのrailsチュートリアルビデオのpart2に大体対応してますが、ここでの内容はちょっと変えてます。
- http://railscasts.com/episodes/236-omniauth-part-2


rails generate

以下のようにしてdeviseのセットアップとmigrationをします。
emailとpasswordはつかわないので消します。

rails g devise:install
rails g devise user
rails g migration AddOauthTokenAndOauthTokenSecretToAuthentications oauth_token:string oauth_token_secret:string
rails g migration RemoveEmailAndEncryptedPasswordAndPasswordSaltFromUsers email:string encrypted_password:string password_salt:string

rake db:migrate

model

authenrication.rb

class Authentication < ActiveRecord::Base
  belongs_to :user
end
  • attr_accessibleは必要ないので消します

user.rb

class User < ActiveRecord::Base
  has_many :authentications

  devise :database_authenticatable, :registerable,
         :recoverable, :rememberable, :trackable

  def password_required?
    false
  end
end
  • 今回の場合、userは一つのtwitter認証情報をもつだけなのでhas_manyはおかしいと思うかもしれないですが、今後facebookとかとも連動させたくなったときのためにこうしてます。
  • devise の :validatable パラメータは削除してます。

controller

前回編集したAuthenticationsControllerをさらに以下のように編集します。

class AuthenticationsController < ApplicationController

def create
  omniauth = request.env['omniauth.auth']
  authentication = Authentication.find_by_provider_and_uid(omniauth['provider'], omniauth['uid'])

  if authentication
    sign_in_and_redirect(:user, authentication.user)
  elsif current_user          # 既にログインしてるけど、facebookとかの権限も追加するとき
    current_user.authentications.create!(:provider => omniauth['provider'], :uid => omniauth['uid'],
                                         :oauth_token => omniauth['credentials']['token'],
                                         :oauth_token_secret => omniauth['credentials']['secret'])
    redirect_to authentications_url
  else                        # 新規ユーザのとき
    user = User.new
    user.authentications.build(:provider => omniauth['provider'], :uid => omniauth['uid'],
                               :oauth_token => omniauth['credentials']['token'],
                               :oauth_token_secret => omniauth['credentials']['secret'])
    user.save!
    sign_in_and_redirect(:user, user)
  end
end

view

確認用にauthenticateのviewを以下のように編集します

<h1>Authentications</h1>

<% if user_signed_in? %>
  Signed in as <%= current_user.id %>
  Not you ? <%= link_to "Sign out", destroy_user_session_path %>
<% else %>
  <%= link_to "Sign in", "/auth/twitter" %>
<% end %>


<table>
  <tr>
    <th>User</th>
    <th>Provider</th>
    <th>Uid</th>
  </tr>
  <% if @authentications %>
    <% for authentication in @authentications %>
      <tr>
        <td><%= authentication.user_id %></td>
        <td><%= authentication.provider %></td>
        <td><%= authentication.uid %></td>
        <td><%= link_to "Destroy", authentication, :confirm => 'Are you sure?', :method => :delete %></td>
      </tr>
    <% end %>
  <% end %>
</table>

routes.rb

前回設定した内容にdevise用の設定も追加します

devise_for :users
resources :authentications
match '/auth/:provider/callback' => 'authentications#create'
root :to => "authentications#index"

確認

以上でセットアップはおわりです。
実際に http://localhost:3000/authentications にアクセスしてsign in をクリックしてうまくログインできれば成功です。

以下のようにすればコンソールでtwitterにポストの確認ができるはずです。

$ rails c
Twitter.configure do |config|
  config.consumer_key = 'CONSUMER_KEY'
  config.consumer_secret = 'CONSUMER_SECRET'
  config.oauth_token = User.all[0].authentications[0].oauth_token
  config.oauth_token_secret = User.all[0].authentications[0].oauth_token_secret
end
Twitter.update 'hello, omniauth!'

まとめ

  • 結構大変でした
  • deviseとかomniauthとかそもそもrailsのことをよくわかってないので間違ってたら教えてください

twitterやfacebookのOAuthをつかってrails+omniauthでログイン機能を実装するメモ その1

OAuthをつかっていろんなサービスと連動したアプリケーションがつくりたくなったので調査。
railsでやるにはoauth_pluginとかwarden_oauthとか色々なプラグインがあるみたいだけど、omniauthがよさそうです。
rails初心者なので基本的にはomniauthのチュートリアルビデオに沿って作業します。
- https://github.com/intridea/omniauth
- http://railscasts.com/episodes/235-omniauth-part-1

自前でパスワードやメールアドレスは保持せずに、twitterやfacebookのアカウントを使ってログインできるようにするのが目標です。
ちなみにtwitter anywhereを使えば似たようなことはできそうですが、twitterに依存してしまうので今回はパス。

--

インストール

とりあえず以下を追記。nifty-generators, twitterはお好みで。deviseは後半で使います

gem 'devise'
gem 'omniauth'
gem 'nifty-generators', :group => :development
gem 'twitter'

設定

omniauth.rb作成

config/initializers/omniauth.rbを作成

Rails.application.config.middleware.use OmniAuth::Builder do
  privider :twitter, 'CONSUMER_KEY', 'CONSUMER_SECRET'
end

ここでセットするoauthアプリケーションは、twitter側で正しくcallback urlを設定しておく必要があります。
通常は http://yourhost.com/auth/twitter/callback のようなURLです。
そうしておかないと、twitterにリダイレクトされる前で 401 unauthorized となってはじかれます。

controller

ここではnifty generatorをつかいます

rails g nifty:scaffold authentication user_id:integer provider:string uid:string index create destroy

できたAuthenticationsControllerのcreateを以下のように書き換えます。

class AuthenticationsController < ApplicationController

def create
  render :text => request.env['omniauth.auth'].to_yaml
end

request.env['omniauth.auth']の部分は過去のバージョンではrequest.env['rack.auth']なので、うまくいかない場合はバージョンを確認してください。

routes.rb

callback用に以下を追記します

match '/auth/:provider/callback' => 'authentications#create'

ここまでやって http://localhost:3000/auth/twitter にアクセスしてtwitterで認証すると、yamlで認証情報が取得できているのが確認できます。

まとめ

以上でとりあえずOAuthでユーザ情報やcredentialをとることができました。
次回( http://taksatou.blogspot.com/2011/03/twitterfacebookoauthrailsomniauth.html )後半でtwitterでログインするところをつくります。

その他

  • rails 3.0.3, ruby 1.8.7 を使用

2011-03-21

githubでPermission denied (publickey)

自分のgithubリポジトリに複数のホストからpushする方法についてメモ

% git push git@github.com:taksatou/config.git master

とかやって

Permission denied (publickey).
fatal: The remote end hung up unexpectedly

なってしまう場合は公開鍵が登録されていない

のSSH Public Keys からpushしたいホストの公開鍵を登録すればOK

2011-03-19

Javascriptの統計ライブラリ jStatについて

jStat (http://www.jstat.org/) とはJavascriptでMATLABやRのような処理をするためのライブラリです。
(同名のjvm統計データ監視ツールとは関係ないです)

http://www.readwriteweb.com/hack/2011/03/jstat-its-like-r-for-javascript.php
でも紹介されてます。
R開発者のEd Borasky氏によると、これからはJavascriptでビジュアライズするのがトレンドらしいです。

というわけでちょっと遊んでみました。

準備

jStatは以下のものに依存してるけど、http://www.jstat.org/download から依存ライブラリをまとめてパッケージにしたものがダウンロードできるので特になにもしなくてOK。

Bundleをダウンロードして(jstat-1.0.0.zip) 解凍すると、デモがindex.htmlにあるのでそれをブラウザで開くだけ。

デモ

試しにtwitterのpublic timelineに流れてるtweetの文字列長の正規分布のグラフをつくってみました。timelineがかわるとグラフも変わります。



ソースはこんなかんじです


$(document).ready(function() {
$.getJSON("public_timeline.php", function(json) {
            var x = new Array(json.length);
            var sum = 0;
            for (var i = 0; i < json.length; ++i) {
                x[i] = json[i]['text'].length;
                sum += x[i];
            }
            var mean = sum / json.length;
            var sum2 = 0;
            for (var i = 0; i < json.length; ++i) {
                sum2 += Math.pow((mean - x[i]), 2);
            }
            var variance = sum2 / json.length;
            var sd = Math.sqrt(variance);
            var distribution = jstat.dnorm(x, mean, sd);
            jstat.plot(x, distribution, {main: "tweet length distribution", type: "p"});
        });
});

本家サイトにデモがあるのでそっちもみると雰囲気がわかると思います http://www.jstat.org/demonstration

まとめ

  • 導入が簡単でブラウザさえあればいけるので、RやMATLABにくらべると敷居が低い感じ。
  • jQueryがつかえればUI的な面で有利
  • ブラウザでRを使いたい場合はRStudio ( http://www.rstudio.org/ ) とかもあるけど、RStudioは完全に開発者向けなのに対して、jStatはリアルタイムに集計結果を一般ユーザに見せたい、みたいなシチュエーションで重宝しそう

以下気になる点

  • まだドキュメントがない
  • Javascriptで実装されてるので複雑なことをやるにはちょっと非力かも??

2011-03-16

tmuxの複数セッション間でバッファを共有する方法

マルチモニタで開発してる場合、それぞれのモニタ毎に端末アプリをたちあげることになりますが、端末ウィンドウ間でコピペをしたいときにマウスを使用せざるをえなくなってめんどうです。
emacsをサーバモードで起動すれば大抵の場合は事足りるのですが、いつもemacsだけで完結できるとは限りません。
というわけでtmuxとかscreenをつかってコピペするわけですが、tmuxのセッションをまたいでバッファを共有する方法をしらなかったのでメモ。

manをみたところ、セッションをまたいでバッファをやりとりするのに以下コマンドが使えそうです。

copy-buffer [-a src-index] [-b dst-index] [-s src-session] [-t dst-session]
(alias: copyb)
             Copy a session paste buffer to another session.  If no sessions are specified, the current one is used instead.

毎回コマンドをたたくのはめんどうなので、以下のような設定をかきました

unbind C
bind C run-shell "for i in `tmux list-session | awk -F':' '/attached/ {print \$1}'`; do tmux copyb -t \$i  ; done"

dst-sessionは明示しないとだめなので、list-sessionでセッション一覧を取得して、アタッチ中のものすべてにコピーするようにしました。

なお、これも含めてその他もろもろの設定ファイルは以下においてます

2011-03-13

Cのプリプロセッサーについてのtips集


プリプロセッサ処理のみを行う(gccの場合)

gcc -E a.c

マクロの引数について

  • ピリオド3つで可変長引数, __VA_ARGS__でその部分を展開
  • 引数の頭に#をつけると文字列表記(ダブルコーテーションでくくられる)
  • 引数の頭に##をつけると連結
#include <stdio.h> 
#define MY_PRINTF(...) fprintf(stderr, __VA_ARGS__)    
#define DEF(x, y) struct x##_##y##_t { static void f() { MY_PRINTF("%s, %s\n", #x, #y); } }  

DEF(foo, bar); 

int main(void) {     
    foo_bar_t::f();    
} 

みたいなことができる

#ifdefと definedの違い

#if defined(VAX) 

#ifdef VAX
と同じ。
definedの場合は
#if defined(VAX) && !defined(UNIX) 
みたいな書き方ができる

その他

  • マクロの名前は特に制限はない
  • 文字列やコメントの中にあるマクロ名は無視される
  • コンマのはいったを引数を一つにまとめたい場合はカッコでくくる

pragmaとかpredefinedなマクロ

  • 次回に続く(たぶん)




ZenBackWidget