NakajiJapan

餓鬼の超弩級日記


datetime_selectをカスタマイズしたときのあれ

なんか神経使った。Railsはいちいちつまずく。
デフォルトだとロケールも英語になっていてなんの問題もないけど、フォーマット変更したり、日本語にしたりするとさらに追加の設定が必要になったのでめも。

最初はデフォルトのロケールを日本語にしているとエラーが発生してたので調査しました。
yaml側でどうも解析できていないみたいだったのでboot.rbに明示的に記述を足してライブラリを読み込めるように
しました。

# confg/application.rb
config.i18n.default_locale = :ja
# config/boot.rb
require 'yaml' # この行を追加
YAML::ENGINE.yamler= 'syck' # この行を追加

そんでもって日付のフォーマットを変更してみる

# config/locale/ja.yml
ja:
  date:
    order: [ :year, :month, :day ]
  datetime:
    order: [ :year, :month, :day, :hour, :min ]
# views/hoge/_form.html.erb
<%= f.datetime_select :disp_date, :use_month_numbers => true %>

うまくいきました。

■参考URL
http://stackoverflow.com/quest……yamlengine

No Comments »

sorceryを使ってログイン処理を作成してみる

Railsでsorceryを使ってログイン処理を実現してみることにします。

まずはMysqlを理由するのでそれに関連したプラグインもいれてきます。
Gemfileに必要なプラグインを記述

gem 'mysql2'
gem 'sorcery'
gem 'refinerycms'
bundle install

DBの設定を行います

development:
  adapter: mysql2
  database: gifanime
  pool: 5
  timeout: 5000
  encoding: utf8

# Warning: The database defined as "test" will be erased and
# re-generated from your development database when you run "rake".
# Do not set this db to the same as development or production.
test:
  adapter: mysql2
  database: gifanime
  pool: 5
  timeout: 5000
  encoding: utf8

production:
  adapter: mysql2
  database: gifanime
  pool: 5
  timeout: 5000
  encoding: utf8

とりあえず、WEBサーバが動作するようにUnicornの設定をします。(内容はとりあえず、動作する目的で記述しているのであまりつっこまないでください)

# -*- coding: utf-8 -*-
# ワーカーの数
worker_processes 2

# capistrano 用に RAILS_ROOT を指定
app_path = "/Users/nakajimadaichi/develop/gifanime/"
working_directory app_path

# ソケット
listen  '/tmp/unicorn.sock'

# ログ
rails_env = ENV['RAILS_ENV'] || 'production'
if rails_env == 'production'
  stderr_path 'log/unicorn.log'
  stdout_path 'log/unicorn.log'
else
  # stdout
end

# ダウンタイムなくす
preload_app true

before_fork do |server, worker| defined?(ActiveRecord::Base) and ActiveRecord::Base.connection.disconnect!
  old_pid = "#{ server.config[:pid] }.oldbin"
  unless old_pid == server.pid
    begin
      # SIGTTOU だと worker_processes が多いときおかしい気がする
      Process.kill :QUIT, File.read(old_pid).to_i
      rescue Errno::ENOENT, Errno::ESRCH
    end
  end
end

after_fork do |server, worker|
  defined?(ActiveRecord::Base) and ActiveRecord::Base.establish_connection
end

前回の設定をちょっとかえてて、環境によってエラー出力をファイル出力か標準出力ぐらいのことしてます。
最後に起動してこんなふうに出力すればとりあえず完了。。

$ unicorn_rails -c config/unicorn.rb -E development -p 5000
I, [2012-02-12T19:35:14.296704 #4448]  INFO -- : unlinking existing socket=/tmp/unicorn.sock
I, [2012-02-12T19:35:14.297074 #4448]  INFO -- : listening on addr=/tmp/unicorn.sock fd=5
I, [2012-02-12T19:35:14.298091 #4448]  INFO -- : listening on addr=0.0.0.0:5000 fd=6
I, [2012-02-12T19:35:14.298290 #4448]  INFO -- : Refreshing Gem list
I, [2012-02-12T19:35:16.681432 #4448]  INFO -- : master process ready
I, [2012-02-12T19:35:16.694516 #4477]  INFO -- : worker=0 ready
I, [2012-02-12T19:35:16.698188 #4478]  INFO -- : worker=1 ready

次はsorceryのインストールです。

# デフォルトのインストール
rails generate sorcery:install
# ユーザのリソースを作成
rails g scaffold User username:string email:string crypted_password:string salt:string
# DBは婦負
rake db:migrate
# セッション用のコントローラ作成
rails g controller UserSessions new create destroy

あとはsorceryが公開されているgithubのwikiをみて写経していきます。

# views/user_sessions/new.html.erb
<%= form_for(@user) do |f| %>
  <% if @user.errors.any? %>

<%= pluralize(@user.errors.count, "error") %> prohibited this user from being saved:

    <% @user.errors.full_messages.each do |msg| %>
  • <%= msg %>
  • <% end %>
<% end %>
<%= f.label :username %> <%= f.text_field :username %>
<%= f.label :email %> <%= f.text_field :email %>
<%= f.label :password %> <%= f.password_field :password %>
<%= f.label :password_confirmation %> <%= f.password_field :password_confirmation %>
<%= f.submit %>
<% end %>
# models/user.rb
class User < ActiveRecord::Base
  authenticates_with_sorcery!
  #attr_accessible :email, :password, :password_confirmation

  validates_length_of       :password, :minimum => 3, :message => "password must be at least 3 characters long", :if => :password
  validates_confirmation_of :password, :message => "should match confirmation", :if => :password
end
# controllers/user_sessions_controller.rb
class UserSessionsController < ApplicationController
  before_filter :require_login
  skip_before_filter :require_login, :o nly => [:index, :new, :create]

  def new
    @user = User.new
  end

  def create
    respond_to do |format|
      if @user = login(params[:username],params[:password])
        format.html { redirect_back_or_to(:users, :notice => 'Login successful.') }
        format.xml { render :x ml => @user, :status => :created, :location => @user }
      else
        format.html { flash.now[:alert] = "Login failed."; render :action => "new" }
        format.xml { render :x ml => @user.errors, :status => :unprocessable_entity }
      end
    end
  end

  def destroy
    logout
    redirect_to(:users, :notice => 'Logged out!')
  end
end
# views/user_sessions/new.html.erb

Login

<%= render 'form' %> <%= link_to 'Back', user_sessions_path %>
# views/user_sessions/_form.html.erb

<%= form_tag user_sessions_path, :method => :post do %>
<%= label_tag :username %> <%= text_field_tag :username %>
<%= label_tag :password %> <%= password_field_tag :password %>
<%= submit_tag "Login" %>
<% end %>
# config/routes.rb
  root :to => 'users#index'
  resources :user_sessions
  resources :users

  match 'login' => 'user_sessions#new', :as => :login
  match 'logout' => 'user_sessions#destroy', :as => :logout
# views/layouts/application.html.erb


  

      <%= stylesheet_link_tag    "application" %>
      <%= javascript_include_tag "application" %>
      <%= csrf_meta_tags %>
  
  

<%= notice %>

<%= alert %>

<%= yield %>
# controllers/users_controller.rb
class UsersController < ApplicationController
  before_filter :require_login
  skip_before_filter :require_login, :except => [:destroy]
end

これでログイン可能になります。

このプラグインには他にもTwitter認証やfacebook認証もできるみたいで「external」をサブモジュールに
指定すればできるらしいです。

■参照URL
・Simple Password Authentication
https://github.com/NoamB/sorce……entication

・外部連携してログインする方法
https://github.com/NoamB/sorce……i/External

No Comments »

「Unicorn + Rails + Nginx」で構築してみる

勉強がてら構築してみました。以前はrailsのWebサーバはThinで構築していたのですが、
最近流行している「Unicorn」、さらにはフロントにリバースプロキシとHTTP用のサーバ「nginx」
をおいて構築してみることにします。

仕組みについてはこちらのサイトがよくまとめられていたので概要つかむによいでしょう。

まずは、nginxのインストールを行います。インストールはあらかじめrpm をインストールしてからyumでインストールし直します。

rpm -ivh nginx-release-centos-6-0.el6.ngx.noarch.rpm
yum install nginx

一度、デフォルトでnginxで起動するかどうかの確認を行います。「service nginx start」で http://hoge.moge にアクセスしてnginx用の画面が表示されていればとりあえずはインストール完了です。

以下、Unicornがnginx経由でアクセスできるようにするための設定ファイルになります。

upstream hoge.moge {
    server unix:/var/www/test/tmp/sockets/unicorn.sock;
}

server {
  listen 80;
  server_name hoge.moge;

  root /var/www/test/public/;
  access_log log/access.log;
  error_log  log/error.log;

  location / {
    if (-f $request_filename) {
        break;
    }
    # ファイルが存在しなければunicornにproxyする
    proxy_set_header X-Real-IP  $remote_addr;
    proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    proxy_set_header Host $http_host;
    proxy_pass http://hoge.moge;
  }

  # この記述がないと静的ファイルが表示されなかった
  location ~* \.(ico|css|js|gif|jpe?g|png)(\?[0-9]+)?$ {
    expires 1y;
  }
}

なんか書き方はPerlbalよりわかりやすいようなイメージしました。気のせいかもしれませんが。。。

次は、Unicornのインストールです。gemでパッケージをインストールします。

gem install unicorn

これでおしまい。

その後、Railsで展開されているコンフィグファイルに設定情報を記述します。

vi config/unicorn.conf
# -*- coding: utf-8 -*-
# ワーカーの数
worker_processes 2

# capistrano 用に RAILS_ROOT を指定
working_directory "/var/www/test/"

# ソケット
#listen '/tmp/unicorn_of_example.sock'
#listenFile = working_directory << 'tmp/sockets/unicorn.sock'
listen  '/var/www/test/tmp/sockets/unicorn.sock'

# ログ
log = '/var/log/rails/unicorn.log'
stderr_path File.expand_path('log/unicorn.log', ENV['RAILS_ROOT'])
stdout_path File.expand_path('log/unicorn.log', ENV['RAILS_ROOT'])

# ダウンタイムなくす
preload_app true

before_fork do |server, worker|
    defined?(ActiveRecord::Base) and ActiveRecord::Base.connection.disconnect!

    old_pid = "#{ server.config[:pid] }.oldbin"
    unless old_pid == server.pid
        begin
            # SIGTTOU だと worker_processes が多いときおかしい気がする
            Process.kill :QUIT, File.read(old_pid).to_i
        rescue Errno::ENOENT, Errno::ESRCH
        end
    end
end

after_fork do |server, worker|
    defined?(ActiveRecord::Base) and ActiveRecord::Base.establish_connection
end

最後にそれぞれを起動します。

unicorn_rails -c config/unicorn.rb -E production -p 5000 -D
service nginx start

指定のURLからアクセスして画面がみれれば完成でし!

□参考URL
http://nginx.org/
http://techracho.bpsinc.jp/bps......07_28/2208

1 Comment »

Rails + Thinで構成してみる in Scientific Linux

さくらのVPSを借りることになったので早速Ruby+Rails+Thinで構築してみることにします。

「Thin」てどういうものかって勝手にまとめてみると軽量で高速なRails専用のWebサーバらしいです。

the Mongrel parser, the root of Mongrel speed and security
Event Machine, a network I/O library with extremely high scalability, performance and stability
Rack, a minimal interface between webservers and Ruby frameworks

RVMのインストール

# bash -s stable < <(curl -s https://raw.github.com/wayneeseguin/rvm/master/binscripts/rvm-installer)

設定情報を追加

[[ -s "$HOME/.rvm/scripts/rvm" ]] && . "$HOME/.rvm/scripts/rvm"
# rvm  install 1.9.3
# rvm  install 1.9.2
# rvm --default use 1.9.3

RubyGemsのインストール

sudo /usr/local/rvm/bin/ruby setup.rb
gem update --system

Mysqlのインストール

gem install mysql2

これの前提条件として

yum install mysql mysql-server mysql-devel

が必要になる。

Thinのインストール:Ruby専用サーバ

gem install execjs
gem install therubyracer
gem install thin

これでブラウザで画面がみれるようになれば完成です。しかし、Thinとは別にまたUnicornらしきものがでていることを知ったので勉強がてらいれてみることにします。それは後ほど。

■リファレンス
http://www.ruby-lang.org/ja/
http://rubygems.org/
http://beginrescueend.com/
http://code.macournoyer.com/thin/

No Comments »

[入門]railsで簡単なヘルパーを作成してみる

viewで使うようのhelperを作成します。

style="display: none"<%end%>> <%= render @cart %>

これじゃスマートじゃないよね。というかもっと簡素化します。

    <%= hidden_div_if(@cart.line_items.empty?, id: 'cart') do %>
      <%= render @cart %>
    <%end%>

実際のヘルパーは「app/helpers/appliction_helper.rb」に保存されて以下のように実装すると完了です。

module ApplicationHelper
  def hidden_div_if(condition, attributes = {}, &block)
    if condition
      attributes["style"] = "display: none"
    end

    content_tag("div", attributes, &block)
  end
end
No Comments »