[Rails 2.3.2] acts_as_state_machine から AASMへ変更

こんちには、おがしょーです。
今回は自分の備忘録のためのエントリです。


ユーザー認証などでRestfulAuthenticationを使う場合、ステートマシンとして2種類のモジュールが使える。
1. acts_as_state_machine
2. AASM (rubyist-aasm)


※今回の流れ。
ググって出てくる参考サイトではほとんどacts_as_state_machineを使用していたので、迷い無くこれをチョイス。

プラグインはGitHubからのインストールでまとめたいので探してみる、が見つからない。

実は長い間アップデートされてない。

後継でAASMなるものを発見。

rubyist-aasmをインストールした後、RestfulAuthenticationをAASMに切り替える。

find_in_stateがエラーになる。

そういえばacts_as_state_machineに複数状態を指定できるパッチ当ててたっけ。。


引用元: kaeruspoon

module ScottBarron
module Acts
module StateMachine
module ClassMethods
protected
def with_state_scope(target_states)
target_states = [target_states] unless target_states.is_a?(Array)
raise InvalidState unless target_states.all? {|s| states.include?(s)}
cond = []
cond_param = []
target_states.each do |st|
cond << "#{table_name}.#{state_column} = ?"
cond_param < {:conditions => [cond.join(" OR "), cond_param].flatten} do
yield if block_given?
end
end
end
end
end
end

AASM用に変更してみる・・・が、動かない。AASMがgemだからか??

メソッド名を変えてリトライ。
[ RAILS/config/initializers/patch_aasm.rb ]
module AASM
module Persistence
module ActiveRecordPersistence
module ClassMethods
def find_in_states(number, state, *args)
with_states_scope state do
find(number, *args)
end
end
def count_in_states(state, *args)
with_states_scope state do
count(*args)
end
end
def calculate_in_states(state, *args)
with_states_scope state do
calculate(*args)
end
end
protected
def with_states_scope(target_states)
target_states = [target_states] unless target_states.is_a?(Array)
raise InvalidState unless target_states.all? {|s| aasm_states.include?(s)}
cond = []
cond_param = []
target_states.each do |st|
cond << "?"
cond_param < {:conditions => ["#{table_name}.#{aasm_column} IN (#{cond.join(', ')})", cond_param].flatten} do
yield if block_given?
end
end
end
end
end
end

できた (n’∀’)η


まだgemとかよくわかってないのGitHubでForkして修正して・・・とかやりたかったけど無理でした。
先は長いなぁ〜。


おまけ。
ステートマシンについてはこちらのページが詳しいです。
境界を越える: Rails での拡張

Amazon EC2? それとも・・・

こんにちは、おがしょーです。

今回は開発したアプリを実行させる環境についてですが、レンタルサーバーのホスティング、ハウジング、クラウドの利用などいろいろな選択肢があります。
弊社は現在、コストパフォーマンスと調達期間の関係でAmazon EC2をメインで使ってます。


EC2、安いんです。初期費用とか無いですし。
月の転送料とか、従量制課金の部分をいれても仮想マシン1台あたり月1万もいかないんじゃないでしょうか。
普通のホスティング(専用サーバー)でやろうとすると、スタート月は初期費用込みで10万近くなりますよね、たぶん。


しかしEC2って、CPUの処理速度はまぁまぁなんですが転送速度がとにかく遅い。。
SSH中の文字とか、ヌルヌルです(´Д`)
通説で日本ー米国間の物理的なネットワーク遅延が原因だと言われています。
といっても200〜300msくらいの遅延で、そこそこレベルのウェブサービス運用ではそんなに影響することはないと考えてますが。


で、やっぱり国内にサーバーがあればなぁ、ってなります。
10/1付けだけど今日見つけたニュース(Impress EnterpriseWatchより)↓

ITコア、「Amazon EC2より安い」仮想化ホスティングサービス「GS10」



きましたね!国産EC2の対抗が!
GrowServer2010の略称でGS10っていうみたいです。


仮想マシンのラインナップは以下のとおり。(モデル価格表をコピペ)

モデル メモリ CPU ディスク(実効)※1 月額費用S1 1GB 1コア 30GB ¥10,000S2 2GB 2コア 30GB ¥20,000S4 4GB 2コア 30GB ¥30,000S8 8GB 4コア 30GB ¥50,000

  • 初期費は無料です。
  • 契約単位は1ヶ月です。
  • お支払は前払いとなります。
※1 3重化DISK

うーん、できればメモリ2GBのCPU1コアで¥15,000のS1.5(仮)モデルも作ってほしかった。。

仮想化にはVMware vSphereを使ってる模様。
VMの設定はvCenterのUIを直接使えたりするのかな?んなわけないか。
OSのインストールまではどうやってやるのか、とかよくわかってません。


このGS10、転送料無料な上24時間監視やロードバランサーが無料で月1万円は安いと思いますが、どうでしょうか?
さらにダメ押しでSLAは驚異の99.99%(!!)だそうです。


サービスインは11月1日からみたいですが、期待してます。
またEC2との比較テストでもできればまたレポートしてみたいと思います。

[Rails 2.3.2] モデルのカラムにサブクエリの値が使いたいのですよ

こんにちは、最近めっきり(なにが?)なおがしょ〜です。
ここ数ヶ月はずっとRubyOnRailsを勉強してますが、ようやく数%くらいわかってきた気がします。


さて、本題。ウェブアプリケーションで親子関係のあるデータを一覧表示しようとする場合で、子供側のデータを COUNT(*) してみたりしたくなる時があると思います(あるよね?ね??)。深く考えず、はじめにRailsでやってみたのはこんな感じでした↓


# モデル:item.rb
class Item < ActiveRecord::Base
  # col: id
  # col: name
  has_many :sub_item
end
# モデル:sub_item.rb
class SubItem < ActiveRecord::Base
  # col: id
  # col: item_id
  # col: name
  belongs_to :item
end
# アクション:items_controller.rb
def index
   @items = Item.find :all
end
# ビュー:items/index.html.erb
<% @items.each do |item| %>
<%=h item.sub_items.count %><!—  小モデルの合計を表示する —>
<% end %>


上記コードで問題なく実行できましたが、発行されるSQL文は
  親モデルのSELECT文+(小モデルのSELECT文×行数)
となり、非常に無駄な気がします。
サブクエリを使えばSQL文は1つで済みますよね。


プラグインとか他のやり方とか探したんですが、findbysql()をする以外の方法が見つからなかったので、簡単にサブクエリを使ったカラムがモデルに追加できるプラグインを作ってみました。http://github.com/ogasyo/subquerycolumn


以下のコマンドでさくっとインストールして、  $ script/plungin install git://github.com/ogasyo/subquerycolumn.git親モデルとコントローラーのfind()部分を以下の用に修正するだけ。


# モデル:item.rb
class Item < ActiveRecord::Base
  # col: id
  # col: name
  hasmany :sub_item
  sub_query_column :count_sub_items, :type => :integer, :query => “SELECT COUNT(id) FROM sub_items where sub_items.item_id = items.id”
end
# アクション:items_controller.rb
def index
   @items = Item.with_sub_query.find :all
end


ビュー側での値の取得はこんな感じです↓
<%=h item.count_sub_items %><!—  NEW! 小モデルの合計を表示する —>

以上で超簡単にできます。


サクッと簡単に作ったプラグインなのでテストはありませんし、バグもあると思います。
利用する場合はきっちりデバッグを行うようにオヌヌメします。
できればGitHubでForkして改良してくれれば最高です(他力本願
#あと、他の方法でこんなのあるよ、とかもうすでに同じのあるだろ、とかはこっそり教えてくれるとありがたいです。。

Page 1 of 3123