OSDN Git Service

t30350#:fix destroy in op, p, user
[pettanr/pettanr.git] / spec / models / original_picture_spec.rb
index 2b896d6..8991830 100644 (file)
+# -*- encoding: utf-8 -*-
+#原画
 require 'spec_helper'
 
 describe OriginalPicture do
-  pending "add some examples to (or delete) #{__FILE__}"
+  before do
+    @admin = FactoryGirl.create :admin
+    @user = FactoryGirl.create( :user_yas)
+    @author = FactoryGirl.create :author, :user_id => @user.id
+    @artist = FactoryGirl.create :artist_yas, :author_id => @author.id
+    @other_user = FactoryGirl.create( :user_yas)
+    @other_author = FactoryGirl.create :author, :user_id => @other_user.id
+    @other_artist = FactoryGirl.create :artist_yas, :author_id => @other_author.id
+    @sp = FactoryGirl.create :system_picture
+    @lg = FactoryGirl.create :license_group
+    @license = FactoryGirl.create :license, :license_group_id => @lg.id, :system_picture_id => @sp.id
+  end
+  
+  describe '検証に於いて' do
+    before do
+      @op = FactoryGirl.build :original_picture, :artist_id => @artist.id
+    end
+    
+    context 'オーソドックスなデータのとき' do
+      it '下限データが通る' do
+        @op.ext = 'png' #リストにない拡張子は通らないし
+        @op.width = 1
+        @op.height = 1
+        @op.filesize = 1
+        @op.md5 = 'a'*32
+        @op.should be_valid
+      end
+      it '上限データが通る' do
+        @op.ext = 'jpeg'
+        @op.width = 99999
+        @op.height = 99999
+        @op.filesize = 2000000
+        @op.md5 = 'a'*32
+        @op.should be_valid
+      end
+    end
+    
+    context 'extを検証するとき' do
+      it 'nullなら失敗する' do
+        @op.ext = ''
+        @op.should_not be_valid
+      end
+      it '5文字以上なら失敗する' do
+        @op.ext = 'a'*5
+        @op.should_not be_valid
+      end
+      it 'png,gif,jpeg以外なら失敗する' do
+        @op.ext = 'bmp'
+        @op.should_not be_valid
+      end
+    end
+    context 'widthを検証するとき' do
+      it 'nullなら失敗する' do
+        @op.width = nil
+        @op.should_not be_valid
+      end
+      it '数値でなければ失敗する' do
+        @op.width = 'a'
+        @op.should_not be_valid
+      end
+      it '0なら失敗する' do
+        @op.width = '0'
+        @op.should_not be_valid
+      end
+      it '負でも失敗する' do
+        @op.width = -1
+        @op.should_not be_valid
+      end
+    end
+    context 'heightを検証するとき' do
+      it 'nullなら失敗する' do
+        @op.height = nil
+        @op.should_not be_valid
+      end
+      it '数値でなければ失敗する' do
+        @op.height = 'a'
+        @op.should_not be_valid
+      end
+      it '0なら失敗する' do
+        @op.height = '0'
+        @op.should_not be_valid
+      end
+      it '負でも失敗する' do
+        @op.height = -1
+        @op.should_not be_valid
+      end
+    end
+    context 'filesizeを検証するとき' do
+      it 'nullなら失敗する' do
+        @op.filesize = nil
+        @op.should_not be_valid
+      end
+      it '数値でなければ失敗する' do
+        @op.filesize = 'a'
+        @op.should_not be_valid
+      end
+      it '負なら失敗する' do
+        @op.filesize = '-1'
+        @op.should_not be_valid
+      end
+      it '2MB以上なら失敗する' do
+        @op.filesize = 2000000+1
+        @op.should_not be_valid
+      end
+    end
+    context 'artist_idを検証するとき' do
+      it 'nullなら失敗する' do
+        @op.artist_id = nil
+        @op.should_not be_valid
+      end
+      it '数値でなければ失敗する' do
+        @op.artist_id = 'a'
+        @op.should_not be_valid
+      end
+      it '存在する絵師でなければ失敗する' do
+        @op.artist_id = 0
+        @op.should_not be_valid
+      end
+    end
+    context 'md5を検証するとき' do
+      it 'nullなら失敗する' do
+        @op.md5 = ''
+        @op.should_not be_valid
+      end
+      it '31文字なら失敗する' do
+        @op.md5 = 'a'*31
+        @op.should_not be_valid
+      end
+      it '33文字なら失敗する' do
+        @op.md5 = 'a'*33
+        @op.should_not be_valid
+      end
+    end
+  end
+  
+  describe 'デフォルト値補充に於いて' do
+    it 'defined' do
+      @op = FactoryGirl.build :original_picture, :artist_id => @artist.id
+      @op.supply_default
+    end
+  end
+  
+  describe '上書き補充に於いて' do
+    it '絵師idが設定されている' do
+      @op = FactoryGirl.build :original_picture, :artist_id => nil
+      @op.overwrite @artist
+      @op.artist_id.should eq @artist.id
+    end
+  end
+  
+  describe '作者判定に於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    context 'パラメータが作家のとき' do
+      it '自作の原画ならyes' do
+        @op.own?(@author).should == true
+      end
+      it '他人のならno' do
+        @op.own?(@other_author).should == false
+      end
+    end
+    context 'パラメータが絵師のとき' do
+      it '自作の原画ならyes' do
+        @op.own?(@artist).should == true
+      end
+      it '他人のならno' do
+        @op.own?(@other_artist).should == false
+      end
+    end
+    context 'それ以外のとき' do
+      it 'no' do
+        @op.own?(nil).should == false
+      end
+    end
+  end
+  
+  describe '閲覧許可に於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    it '検査対象が管理者のとき、許可を返す' do
+      OriginalPicture.any_instance.stub(:own?).and_return(false)
+      r = @op.visible?(@admin)
+      r.should == true
+    end
+    it '所持判定を問い合わせ、自分の原画なら許可する' do
+      OriginalPicture.any_instance.stub(:own?).and_return(true)
+      r = @op.visible?(@artist)
+      r.should == true
+    end
+    it '他人の原画なら許可しない' do
+      OriginalPicture.any_instance.stub(:own?).and_return(false)
+      r = @op.visible?(@artist)
+      r.should == false
+    end
+  end
+  
+  describe 'ファイル名に於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    it 'id+拡張子のフォーマットで返す' do
+      r = @op.filename
+      r.should eq "#{@op.id}.png"
+    end
+  end
+  
+  describe 'MimeTypeに於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    it 'image/拡張子のフォーマットで返す' do
+      r = @op.mime_type
+      r.should eq "image/png"
+    end
+  end
+  
+  describe 'ファイルのurlに於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+      OriginalPicture.any_instance.stub(:filename).and_return('3.gif')
+    end
+    it 'ファイル名取得を依頼している' do
+      OriginalPicture.any_instance.should_receive(:filename).exactly(1)
+      @op.url
+    end
+    it '/original_pictures/3.gifのフォーマットで返す' do
+      r = @op.url
+      r.should eq "/original_pictures/3.gif"
+    end
+  end
+  
+  describe 'サムネイル画像タグオプションに於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+      OriginalPicture.any_instance.stub(:url).and_return('/original_pictures/3.gif')
+      PettanImager.stub(:thumbnail_size).with(any_args).and_return([40, 30])
+    end
+    it 'サムネイル画像の幅高さ取得を依頼している' do
+      PettanImager.should_receive(:thumbnail_size).with(any_args).exactly(1)
+      @op.tmb_opt_img_tag
+    end
+    it '戻り値はHashで返す' do
+      r = @op.tmb_opt_img_tag
+      r.is_a?(Hash).should be_true
+    end
+    it 'srcキーを含んでいる' do
+      r = @op.tmb_opt_img_tag
+      r.has_key?(:src).should be_true
+      r[:src].should eq '/original_pictures/3.gif'
+    end
+    it 'widthキーを含んでいる' do
+      r = @op.tmb_opt_img_tag
+      r.has_key?(:width).should be_true
+      r[:width].should eq 40
+    end
+    it 'heightキーを含んでいる' do
+      r = @op.tmb_opt_img_tag
+      r.has_key?(:height).should be_true
+      r[:height].should eq 30
+    end
+  end
+  
+  describe '画像タグオプションに於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+      OriginalPicture.any_instance.stub(:url).and_return('/original_pictures/3.gif')
+    end
+    it '戻り値はHashで返す' do
+      r = @op.opt_img_tag
+      r.is_a?(Hash).should be_true
+    end
+    it 'srcキーを含んでいる' do
+      r = @op.opt_img_tag
+      r.has_key?(:src).should be_true
+      r[:src].should eq '/original_pictures/3.gif'
+    end
+    it 'widthキーを含んでいる' do
+      r = @op.opt_img_tag
+      r.has_key?(:width).should be_true
+      r[:width].should eq @op.width
+    end
+    it 'heightキーを含んでいる' do
+      r = @op.opt_img_tag
+      r.has_key?(:height).should be_true
+      r[:height].should eq @op.height
+    end
+  end
+  
+  describe '未公開に於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    context '実素材がゼロのとき' do
+      it 'Trueを返す' do
+        r = @op.unpublished?
+        r.should be_true
+      end
+    end
+    context '実素材がゼロではないとき' do
+      it 'Falseを返す' do
+        @p = FactoryGirl.create :picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id 
+        r = @op.unpublished?
+        r.should be_false
+      end
+    end
+  end
+
+  describe '停止中に於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    context '未公開ではないが、素材がないとき' do
+      it 'Trueを返す' do
+        @p = FactoryGirl.create :picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id 
+        r = @op.stopped?
+        r.should be_true
+      end
+    end
+    context '未公開のとき' do
+      it 'Falseを返す' do
+        r = @op.stopped?
+        r.should be_false
+      end
+    end
+    context '未公開ではなく、素材もあるとき' do
+      it 'falseを返す' do
+        @p = FactoryGirl.create :picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id 
+        @rp = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id, :picture_id => @p.id
+        r = @op.stopped?
+        r.should be_false
+      end
+    end
+  end
+
+  describe 'ライセンス待ちに於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    context '未公開ではなく、素材もあるが、原画の更新日時が実素材のheadのそれより後のとき' do
+      it 'Trueを返す' do
+        @p = FactoryGirl.create :picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id, :updated_at => Time.now - 1000
+        @rp = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id, :picture_id => @p.id
+        r = @op.unlicensed?
+        r.should be_true
+      end
+    end
+    context '未公開のとき' do
+      it 'Falseを返す' do
+        r = @op.unlicensed?
+        r.should be_false
+      end
+    end
+    context '未公開ではないが、素材がないとき' do
+      it 'Falseを返す' do
+        @p = FactoryGirl.create :picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id 
+        r = @op.unlicensed?
+        r.should be_false
+      end
+    end
+    context '未公開ではなく、素材もあり、実素材のheadの更新日時が原画のそれより後のとき' do
+      it 'Falseを返す' do
+        @p = FactoryGirl.create :picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id, :updated_at => Time.now + 1000
+        @rp = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id, :picture_id => @p.id
+        r = @op.unlicensed?
+        r.should be_false
+      end
+    end
+  end
+
+  describe '公開中に於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    context '未公開ではなく、素材もあり、実素材のheadの更新日時が原画のそれより後のとき' do
+      it 'Trueを返す' do
+        @p = FactoryGirl.create :picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id, :updated_at => Time.now + 1000
+        @rp = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id, :picture_id => @p.id
+        r = @op.published?
+        r.should be_true
+      end
+    end
+    context '未公開のとき' do
+      it 'Falseを返す' do
+        r = @op.published?
+        r.should be_false
+      end
+    end
+    context '未公開ではないが、素材がないとき' do
+      it 'falseを返す' do
+        @p = FactoryGirl.create :picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id 
+        r = @op.published?
+        r.should be_false
+      end
+    end
+    context '未公開ではなく、素材もあるが、原画の更新日時が実素材のheadのそれより後のとき' do
+      it 'falseを返す' do
+        @p = FactoryGirl.create :picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id, :updated_at => Time.now - 1000
+        @rp = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id, :picture_id => @p.id
+        r = @op.published?
+        r.should be_false
+      end
+    end
+  end
+
+  describe '一覧取得に於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    context 'page補正について' do
+      it '文字列から数値に変換される' do
+        OriginalPicture.page('8').should eq 8
+      end
+      it 'nilの場合は1になる' do
+        OriginalPicture.page().should eq 1
+      end
+      it '0以下の場合は1になる' do
+        OriginalPicture.page('0').should eq 1
+      end
+    end
+    context 'page_size補正について' do
+      it '文字列から数値に変換される' do
+        OriginalPicture.page_size('7').should eq 7
+      end
+      it 'nilの場合はOriginalPicture.default_page_sizeになる' do
+        OriginalPicture.page_size().should eq OriginalPicture.default_page_size
+      end
+      it '0以下の場合はOriginalPicture.default_page_sizeになる' do
+        OriginalPicture.page_size('0').should eq OriginalPicture.default_page_size
+      end
+      it 'OriginalPicture.max_page_sizeを超えた場合はOriginalPicture.max_page_sizeになる' do
+        OriginalPicture.page_size('1000').should eq OriginalPicture.max_page_size
+      end
+    end
+  end
+  describe '一覧取得オプションに於いて' do
+    it 'includeキーを含んでいる' do
+      r = OriginalPicture.list_opt
+      r.has_key?(:include).should be_true
+    end
+    it '2つの項目を含んでいる' do
+      r = OriginalPicture.list_opt[:include]
+      r.should have(2).items
+    end
+    it '素材を含んでいる' do
+      r = OriginalPicture.list_opt[:include]
+      r.has_key?(:resource_picture).should be_true
+    end
+    it '実素材を含んでいる' do
+      r = OriginalPicture.list_opt[:include]
+      r.has_key?(:pictures).should be_true
+    end
+  end
+  describe 'json一覧出力オプションに於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+      @p = FactoryGirl.create :picture, :original_picture_id => @op.id, :license_id => @license.id, :artist_id => @artist.id
+      @rp = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :license_id => @license.id, :original_picture_id => @op.id, :picture_id => @p.id
+      @sbt = FactoryGirl.create :speech_balloon_template
+      @comic = FactoryGirl.create :comic, :author_id => @author.id, :visible => 1
+      @panel = FactoryGirl.create :panel, :author_id => @author.id, :publish => 1
+      @story = FactoryGirl.create :story, :author_id => @author.id, :comic_id => @comic.id, :panel_id => @panel.id
+    end
+    it '素材を含んでいる' do
+      r = OriginalPicture.mylist(@artist).to_json OriginalPicture.list_json_opt
+      j = JSON.parse r
+      i = j.first
+      i.has_key?('resource_picture').should be_true
+    end
+    it '実素材を含んでいる' do
+      r = OriginalPicture.mylist(@artist).to_json OriginalPicture.list_json_opt
+      j = JSON.parse r
+      i = j.first
+      i.has_key?('pictures').should be_true
+    end
+  end
+  
+  describe '自分の原画一覧取得に於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    context 'つつがなく終わるとき' do
+      it '一覧取得オプションを利用している' do
+        OriginalPicture.stub(:list_opt).with(any_args).and_return({})
+        OriginalPicture.should_receive(:list_opt).with(any_args).exactly(1)
+        r = OriginalPicture.mylist @artist
+      end
+    end
+    it 'リストを返す' do
+      r = OriginalPicture.mylist @artist
+      r.should eq [@op]
+    end
+    it '時系列で並んでいる' do
+      nc = FactoryGirl.create :original_picture, :artist_id => @artist.id, :updated_at => Time.now + 100
+      r = OriginalPicture.mylist @artist
+      r.should eq [nc, @op]
+    end
+    it '他人の原画は含まない' do
+      nc = FactoryGirl.create :original_picture, :artist_id => @other_artist.id
+      r = OriginalPicture.mylist @artist
+      r.should eq [@op]
+    end
+    context 'DBに5件あって1ページの件数を2件に変えたとして' do
+      before do
+        @op2 = FactoryGirl.create :original_picture, :artist_id => @artist.id, :updated_at => Time.now + 100
+        @op3 = FactoryGirl.create :original_picture, :artist_id => @artist.id, :updated_at => Time.now + 200
+        @op4 = FactoryGirl.create :original_picture, :artist_id => @artist.id, :updated_at => Time.now + 300
+        @op5 = FactoryGirl.create :original_picture, :artist_id => @artist.id, :updated_at => Time.now + 400
+      end
+      it '通常は2件を返す' do
+        r = OriginalPicture.mylist @artist, 1, 2
+        r.should have(2).items 
+      end
+      it 'page=1なら末尾2件を返す' do
+        #時系列で並んでいる
+        r = OriginalPicture.mylist @artist, 1, 2
+        r.should eq [@op5, @op4]
+      end
+      it 'page=2なら中間2件を返す' do
+        r = OriginalPicture.mylist @artist, 2, 2
+        r.should eq [@op3, @op2]
+      end
+      it 'page=3なら先頭1件を返す' do
+        r = OriginalPicture.mylist @artist, 3, 2
+        r.should eq [@op]
+      end
+    end
+    context 'DBに5件あって1ページの件数を0件に変えたとして' do
+      before do
+        @op2 = FactoryGirl.create :original_picture, :artist_id => @artist.id, :updated_at => Time.now + 100
+        @op3 = FactoryGirl.create :original_picture, :artist_id => @artist.id, :updated_at => Time.now + 200
+        @op4 = FactoryGirl.create :original_picture, :artist_id => @artist.id, :updated_at => Time.now + 300
+        @op5 = FactoryGirl.create :original_picture, :artist_id => @artist.id, :updated_at => Time.now + 400
+        OriginalPicture.stub(:default_page_size).and_return(2)
+      end
+      it '通常は全件(5件)を返す' do
+        r = OriginalPicture.mylist @artist, 5, 0
+        r.should have(5).items 
+      end
+    end
+  end
+  
+  describe '更新履歴一覧取得に於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+      @p = FactoryGirl.create :picture, :artist_id => @artist.id, :license_id => @license.id, :revision => 0,
+        :original_picture_id => @op.id
+      @op2 = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    it 'リストを返す' do
+      r = @op.history
+      r.should eq [@p]
+    end
+    it '他の原画の実素材は含んでいない' do
+      @p2 = FactoryGirl.create :picture, :artist_id => @artist.id, :license_id => @license.id, :revision => 0,
+        :original_picture_id => @op2.id
+      r = @op.history
+      r.should eq [@p]
+    end
+    it 'revisionで並んでいる' do
+      @p2 = FactoryGirl.create :picture, :artist_id => @artist.id, :license_id => @license.id, :revision => 1,
+        :original_picture_id => @op.id
+      r = @op.history
+      r.should eq [@p2, @p]
+    end
+  end
+  
+  describe '単体取得に於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    context 'つつがなく終わるとき' do
+      it '単体取得オプションを利用している' do
+        OriginalPicture.stub(:show_opt).with(any_args).and_return({})
+        OriginalPicture.should_receive(:show_opt).with(any_args).exactly(1)
+        r = OriginalPicture.show @op.id, @artist
+      end
+      it '閲覧許可を問い合わせている' do
+        OriginalPicture.any_instance.stub(:visible?).with(any_args).and_return(true)
+        OriginalPicture.any_instance.should_receive(:visible?).with(any_args).exactly(1)
+        r = OriginalPicture.show @op.id, @artist
+      end
+    end
+    it '指定の原画を返す' do
+      pic = OriginalPicture.show @op.id, @artist
+      pic.should eq @op
+    end
+    context '他人の原画を開こうとしたとき' do
+      it '403Forbidden例外を返す' do
+        OriginalPicture.any_instance.stub(:visible?).and_return(false)
+        lambda{
+          pic = OriginalPicture.show @op.id, @other_artist
+        }.should raise_error(ActiveRecord::Forbidden)
+      end
+    end
+    context '存在しない原画を開こうとしたとき' do
+      it '404RecordNotFound例外を返す' do
+        lambda{
+          pic = OriginalPicture.show 0, @artist
+        }.should raise_error(ActiveRecord::RecordNotFound)
+      end
+    end
+  end
+  describe '編集取得に於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    context 'つつがなく終わるとき' do
+      it '単体取得オプションを利用している' do
+        OriginalPicture.stub(:show_opt).with(any_args).and_return({})
+        OriginalPicture.should_receive(:show_opt).with(any_args).exactly(1)
+        r = OriginalPicture.edit @op.id, @artist
+      end
+      it '所持判定を問い合わせている' do
+        OriginalPicture.any_instance.stub(:own?).with(any_args).and_return(true)
+        OriginalPicture.any_instance.should_receive(:own?).with(any_args).exactly(1)
+        r = OriginalPicture.edit @op.id, @artist
+      end
+    end
+    it '指定の原画を返す' do
+      pic = OriginalPicture.edit @op.id, @artist
+      pic.should eq @op
+    end
+    context '他人の原画を開こうとしたとき' do
+      it '403Forbidden例外を返す' do
+        OriginalPicture.any_instance.stub(:own?).and_return(false)
+        lambda{
+          r = OriginalPicture.edit @op.id, @other_artist
+        }.should raise_error(ActiveRecord::Forbidden)
+      end
+    end
+    context '存在しない原画を開こうとしたとき' do
+      it '404RecordNotFound例外を返す' do
+        lambda{
+          r = OriginalPicture.edit 0, @artist
+        }.should raise_error(ActiveRecord::RecordNotFound)
+      end
+    end
+  end
+  describe '単体取得オプションに於いて' do
+    it 'includeキーを含んでいる' do
+      r = OriginalPicture.show_opt
+      r.has_key?(:include).should be_true
+    end
+    it '2つの項目を含んでいる' do
+      r = OriginalPicture.show_opt[:include]
+      r.should have(2).items
+    end
+    it '素材を含んでいる' do
+      r = OriginalPicture.show_opt[:include]
+      r.has_key?(:resource_picture).should be_true
+    end
+    it '実素材を含んでいる' do
+      r = OriginalPicture.show_opt[:include]
+      r.has_key?(:pictures).should be_true
+    end
+  end
+  describe 'json単体出力オプションに於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+      @p = FactoryGirl.create :picture, :original_picture_id => @op.id, :license_id => @license.id, :artist_id => @artist.id
+      @rp = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :license_id => @license.id, :original_picture_id => @op.id, :picture_id => @p.id
+      @sbt = FactoryGirl.create :speech_balloon_template
+      @comic = FactoryGirl.create :comic, :author_id => @author.id, :visible => 1
+      @panel = FactoryGirl.create :panel, :author_id => @author.id, :publish => 1
+      @story = FactoryGirl.create :story, :author_id => @author.id, :comic_id => @comic.id, :panel_id => @panel.id
+    end
+    it '素材を含んでいる' do
+      r = OriginalPicture.show(@op.id, @artist).to_json OriginalPicture.show_json_opt
+      j = JSON.parse r
+      i = j
+      i.has_key?('resource_picture').should be_true
+    end
+    it '実素材を含んでいる' do
+      r = OriginalPicture.show(@op.id, @artist).to_json OriginalPicture.show_json_opt
+      j = JSON.parse r
+      i = j
+      i.has_key?('pictures').should be_true
+    end
+  end
+  describe '作成・更新に於いて' do
+    before do
+      @op = FactoryGirl.build :original_picture, :artist_id => @artist.id
+      @imager = ImagerTest.load "abc\ndef\nghi"
+    end
+    context '事前チェック' do
+      before do
+        #すべての処理が正常パターンで通過すれば、一番深い分岐まで通る。
+        #それで外部のメソッド呼び出しだけに着目してテストする。
+        OriginalPicture.any_instance.stub(:save).with(any_args).and_return(true)
+        PictureIO.original_picture_io.stub(:put).with(any_args).and_return(true)
+      end
+      it '自身に属性をセットしている' do
+        OriginalPicture.any_instance.should_receive(:attributes=).exactly(1)
+        @op.store @imager
+      end
+      it '自身が保存されている' do
+        OriginalPicture.any_instance.should_receive(:save).exactly(1)
+        @op.store @imager
+      end
+      it 'PictureIoに画像データの保存を依頼している' do
+        PictureIO.original_picture_io.should_receive(:put).with(any_args).exactly(1)
+        @op.store @imager
+      end
+    end
+    context 'つつがなく終わるとき' do
+      before do
+      end
+      it '自身に属性をセットしている' do
+        @op.store @imager
+        @op.width.should eq 3
+        @op.height.should eq 3
+        @op.filesize.should eq 9
+      end
+      it '原画モデルが作成されている' do
+        lambda {
+          @op.store @imager
+        }.should change OriginalPicture, :count
+      end
+      it '原画が保存されている' do
+        @op.store @imager
+        OriginalPicture.find(@op).should_not be_nil
+      end
+      it 'Trueを返す' do
+        @op.store(@imager).should eq true
+      end
+    end
+    #以下から例外ケース。処理先頭から失敗させていく
+    context 'imagerが初期化に失敗したとき' do
+      before do
+      end
+      it 'falseを返す' do
+        @op.store(false).should be_false
+      end
+      it '自身の保存は呼ばれていない' do
+        OriginalPicture.any_instance.should_not_receive(:save)
+        @op.store(false)
+      end
+      it '全体エラーメッセージがセットされている' do
+        lambda {
+          @op.store(false)
+        }.should change(@op.errors[:base], :count)
+      end
+    end
+    context '自身の保存に失敗したとき' do
+      before do
+        OriginalPicture.any_instance.stub(:save).with(any_args).and_return(false)
+      end
+      it 'falseを返す' do
+        @op.store(@imager).should be_false
+      end
+      it '更新されていない' do
+        @op.store(@imager)
+        @op.should be_a_new OriginalPicture
+      end
+      it '原画の保存は呼ばれていない' do
+        PictureIO::LocalPicture.any_instance.should_not_receive(:put)
+      end
+    end
+    context '原画の保存に失敗したとき' do
+      before do
+        OriginalPicture.any_instance.stub(:save).with(any_args).and_return(true)
+        PictureIO.original_picture_io.stub(:put).with(any_args).and_raise(PictureIO::Error)
+      end
+      it 'falseを返す' do
+        @op.store(@imager).should be_false
+      end
+      it '更新されていない' do
+        @op.store(@imager)
+        @op.should be_a_new OriginalPicture
+      end
+      it '全体エラーメッセージがセットされている' do
+        lambda {
+          @op.store(@imager)
+        }.should change(@op.errors[:base], :count)
+      end
+    end
+  end
+  
+  describe '削除に於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+    end
+    context '事前チェックしておく' do
+      before do
+        OriginalPicture.any_instance.stub(:destroy).and_return(true)
+        ResourcePicture.any_instance.stub(:unpublish).and_return(true)
+        Picture.any_instance.stub(:unpublish).with(any_args).and_return(true)
+        PictureIO.original_picture_io.stub(:delete).with(any_args).and_return(true)
+      end
+      it '原画モデルに削除を依頼している' do
+        OriginalPicture.any_instance.should_receive(:destroy).exactly(1)
+        r = @op.destroy_with_resource_picture
+      end
+      it '保管庫に原画の画像データ削除を依頼している' do
+        PictureIO.original_picture_io.should_receive(:delete).with(@op.filename).exactly(1)
+        r = @op.destroy_with_resource_picture
+      end
+      context '自身にリンクされた素材があるとき' do
+        it '素材モデルに削除を依頼している' do
+          @p = FactoryGirl.create :picture, :original_picture_id => @op.id, :license_id => @license.id, :artist_id => @artist.id, :revision => 0
+          @rp = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :license_id => @license.id, :original_picture_id => @op.id, :picture_id => @p.id
+          ResourcePicture.any_instance.should_receive(:unpublish).exactly(1)
+          r = @op.destroy_with_resource_picture
+        end
+      end
+      context '自身にリンクされた素材がないとき' do
+        it '素材モデルに削除を依頼しない' do
+          ResourcePicture.any_instance.should_not_receive(:unpublish)
+          r = @op.destroy_with_resource_picture
+        end
+      end
+      context '自身にリンクされた実素材があるとき' do
+        it 'すべての実素材に墨塗を依頼している' do
+          @p = FactoryGirl.create :picture, :original_picture_id => @op.id, :license_id => @license.id, :artist_id => @artist.id, :revision => 0
+          Picture.any_instance.should_receive(:unpublish).with(any_args).exactly(1)
+          r = @op.destroy_with_resource_picture
+        end
+      end
+      context '自身にリンクされた実素材がないとき' do
+        it '実素材に墨塗を依頼しない' do
+          Picture.any_instance.should_not_receive(:unpublish)
+          r = @op.destroy_with_resource_picture
+        end
+      end
+    end
+    context 'つつがなく終わるとき' do
+      before do
+        PictureIO.original_picture_io.stub(:delete).with(any_args).and_return(true)
+        @p = FactoryGirl.create :picture, :original_picture_id => @op.id, :license_id => @license.id, :artist_id => @artist.id, :revision => 0
+        @rp = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :license_id => @license.id, :original_picture_id => @op.id, :picture_id => @p.id
+      end
+      it '自身を削除する' do
+        lambda {
+          r = @op.destroy_with_resource_picture
+        }.should change(OriginalPicture, :count).by(-1)
+        lambda {
+          r = OriginalPicture.find @op.id
+        }.should raise_error
+      end
+      it 'Trueを返す' do
+        r = @op.destroy_with_resource_picture
+        r.should be_true
+      end
+    end
+    context '自身の削除に失敗したとき' do
+      before do
+        OriginalPicture.any_instance.stub(:destroy).with(any_args).and_return(false)
+      end
+      it 'Falseを返す' do
+        r = @op.destroy_with_resource_picture
+        r.should be_false
+      end
+      it 'ロールバックしている' do
+        lambda {
+          r = @op.destroy_with_resource_picture
+        }.should_not change(OriginalPicture, :count)
+      end
+    end
+    context '画像の削除に失敗したとき' do
+      before do
+        PictureIO.original_picture_io.stub(:delete).with(@op.filename).and_raise(PictureIO::Error)
+      end
+      it 'Falseを返す' do
+        r = @op.destroy_with_resource_picture
+        r.should be_false
+      end
+      it 'ロールバックしている' do
+        lambda {
+          r = @op.destroy_with_resource_picture
+        }.should_not change(OriginalPicture, :count)
+      end
+    end
+    context '素材の削除に失敗とき' do
+      before do
+        @p = FactoryGirl.create :picture, :original_picture_id => @op.id, :license_id => @license.id, :artist_id => @artist.id, :revision => 0
+        @rp = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :license_id => @license.id, :original_picture_id => @op.id, :picture_id => @p.id
+        ResourcePicture.any_instance.stub(:unpublish).with(any_args).and_return(false)
+      end
+      it 'Falseを返す' do
+        r = @op.destroy_with_resource_picture
+        r.should be_false
+      end
+      it 'ロールバックしている' do
+        lambda {
+          r = @op.destroy_with_resource_picture
+        }.should_not change(OriginalPicture, :count)
+        lambda {
+          r = @op.destroy_with_resource_picture
+        }.should_not change(ResourcePicture, :count)
+      end
+    end
+    context '実素材の墨塗に失敗とき' do
+      before do
+        @p = FactoryGirl.create :picture, :original_picture_id => @op.id, :license_id => @license.id, :artist_id => @artist.id, :revision => 0
+        @rp = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :license_id => @license.id, :original_picture_id => @op.id, :picture_id => @p.id
+        Picture.any_instance.stub(:unpublish).with(any_args).and_return(false)
+      end
+      it 'Falseを返す' do
+        r = @op.destroy_with_resource_picture
+        r.should be_false
+      end
+      it 'ロールバックしている' do
+        lambda {
+          r = @op.destroy_with_resource_picture
+        }.should_not change(OriginalPicture, :count)
+        lambda {
+          r = @op.destroy_with_resource_picture
+        }.should_not change(ResourcePicture, :count)
+      end
+    end
+  end
+  
+=begin
+  describe 'エクスポートに於いて' do
+    before do
+      @op = FactoryGirl.create :original_picture, :artist_id => @artist.id
+      @p = FactoryGirl.create :picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id, :revision => 0
+      @rp = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :original_picture_id => @op.id, :license_id => @license.id, :picture_id => @p.id
+      #他人の原画排除
+      @other_op = FactoryGirl.create :original_picture, :artist_id => @other_artist.id
+    end
+    context 'つつがなく終わるとき' do
+      it 'ライセンスグループに依頼してリストを取得している' do
+        LicenseGroup.stub(:list).with(any_args).and_return([@lg])
+        LicenseGroup.should_receive(:list).with(any_args).exactly(1)
+        r = OriginalPicture.export(@artist)
+      end
+      it '原画に依頼してリストを取得している' do
+        OriginalPicture.stub(:list).with(any_args).and_return([@op])
+        OriginalPicture.should_receive(:list).with(any_args).exactly(1)
+        r = OriginalPicture.export(@artist)
+      end
+      it 'Hashを返す' do
+        r = OriginalPicture.export(@artist)
+        r.is_a?(Hash).should be_true
+      end
+      it 'Hashはライセンスグループを含んでいる' do
+        r = OriginalPicture.export(@artist)
+        r.include?(:license_groups).should be_true
+      end
+      it 'Hashは原画を含んでいる' do
+        r = OriginalPicture.export(@artist)
+        r.include?(:original_pictures).should be_true
+      end
+      #素材がライセンスされていないケースもある
+      it 'Hashの原画は素材を含んでいる' do
+        r = OriginalPicture.export(@artist)
+        r[:original_pictures].first.resource_picture.should_not be_nil
+      end
+      it 'Hashの原画は実素材を含んでいる' do
+        r = OriginalPicture.export(@artist)
+        r[:original_pictures].first.pictures.should be_nil
+      end
+    end
+    context '実データ単体のとき' do
+      it 'ライセンスは配列構造になっている' do
+        r = OriginalPicture.export(@artist)
+        r[:license_groups].is_a?(Array).should be_true
+      end
+      it 'ライセンスが全件出ている' do
+        r = OriginalPicture.export(@artist)
+        r[:license_groups].size.should eq 1
+        r[:license_groups].first.should eq @lg
+      end
+      it '原画は配列構造になっている' do
+        r = OriginalPicture.export(@artist)
+        r[:original_pictures].is_a?(Array).should be_true
+      end
+      it '原画が全件出ている' do
+        r = OriginalPicture.export(@artist)
+        r[:original_pictures].size.should eq 1
+        r[:original_pictures].first.should eq @op
+      end
+      it '原画に素材が関連付いている' do
+        r = OriginalPicture.export(@artist)
+        i = r[:original_pictures].first
+        i.resource_picture.should eq @rp
+      end
+      it '原画に実素材が関連付いている' do
+        r = OriginalPicture.export(@artist)
+        i = r[:original_pictures].first
+        i.picture.should eq @p
+      end
+    end
+    context '実データ複数のとき' do
+      before do
+        @lg2 = FactoryGirl.create :license_group, :name => 'export test', :url => 'http://export.test/'
+        @license2 = FactoryGirl.create :license, :license_group_id => @lg2.id, :system_picture_id => @sp.id, :name => 'export test license', :url => 'http://export.test/license'
+        @op2 = FactoryGirl.create :original_picture, :artist_id => @artist.id
+        @p2 = FactoryGirl.create :picture, :artist_id => @artist.id, :original_picture_id => @op2.id, :license_id => @license2.id, :revision => 0
+        @rp2 = FactoryGirl.create :resource_picture, :artist_id => @artist.id, :original_picture_id => @op2.id, :license_id => @license2.id, :picture_id => @p2.id
+      end
+      it 'ライセンスは配列構造になっている' do
+        r = OriginalPicture.export(@artist)
+        r[:license_groups].is_a?(Array).should be_true
+      end
+      it 'ライセンスが全件出ている' do
+        r = OriginalPicture.export(@artist)
+        r[:license_groups].size.should eq 2
+        r[:license_groups].first.should eq @lg
+        r[:license_groups].last.should eq @lg2
+      end
+      it '原画は配列構造になっている' do
+        r = OriginalPicture.export(@artist)
+        r[:original_pictures].is_a?(Array).should be_true
+      end
+      it '原画が全件出ている' do
+        r = OriginalPicture.export(@artist)
+        r[:original_pictures].size.should eq 2
+        r[:original_pictures].first.should eq @op
+        r[:original_pictures].last.should eq @op2
+      end
+      it '原画に素材が関連付いている' do
+        r = OriginalPicture.export(@artist)
+        i = r[:original_pictures].first
+        i.resource_picture.should eq @rp
+        i2 = r[:original_pictures].last
+        i2.resource_picture.should eq @rp2
+      end
+      it '原画に実素材が関連付いている' do
+        r = OriginalPicture.export(@artist)
+        i = r[:original_pictures].first
+        i.picture.should eq @p
+        i2 = r[:original_pictures].last
+        i2.picture.should eq @p2
+      end
+    end
+  end
+  
+  describe 'エクスポートオプションに於いて' do
+  end
+  
+  describe 'インポートに於いて' do
+    before do
+      @imports = {:licenses => {}, :artist_id => @artist.id}
+    end
+    context '事前チェックしておく' do
+    end
+  end
+  
+=end
 end