# coding: utf-8 require 'spec_helper' describe Picky::Query::Token do describe '==' do it 'is equal if the originals are equal' do described_class.processed('similar~', 'Similar~').should == described_class.processed('similar~', 'Similar~') end it 'is not equal if the originals are not equal' do described_class.processed('similar~', 'Similar~').should_not == described_class.processed('similar', 'Similar') end end describe 'categorize' do let(:mapper) do categories = Picky::Index.new :categories @category1 = categories.category :category1 @category2 = categories.category :category2 @category3 = categories.category :category3 Picky::Query::QualifierCategoryMapper.new [categories] end context 'with qualifiers' do let(:token) { described_class.processed 'category1:qualifier', 'category1:Qualifier' } context 'unrestricted' do it 'categorizes correctly' do token.categorize(mapper).should == [@category1] end end context 'restricted' do before(:each) do mapper.restrict_to :category1 end it 'categorizes correctly' do token.categorize(mapper).should == [@category1] end end context 'restricted' do before(:each) do mapper.restrict_to :category2, :category3 end it 'categorizes correctly' do token.categorize(mapper).should == [] end end end context 'without qualifiers' do let(:token) { described_class.processed 'noqualifier', 'NoQualifier' } context 'unrestricted' do it 'categorizes correctly' do token.categorize(mapper).should == nil end end context 'restricted' do before(:each) do mapper.restrict_to :category1 end it 'categorizes correctly' do token.categorize(mapper).should == [@category1] end end context 'restricted' do before(:each) do mapper.restrict_to :category2, :category3 end it 'categorizes correctly' do token.categorize(mapper).should == [@category2, @category3] end end end end describe 'similar_tokens_for' do let(:token) { described_class.processed 'similar~', 'Similar~' } context 'with similar' do before(:each) do @bundle = stub :bundle, :similar => ['array', 'of', 'similar'] @category = stub :category, :bundle_for => @bundle end it 'returns a list of tokens' do token.similar_tokens_for(@category).each do |token| token.should be_kind_of(described_class) end end it 'returns all non-similar tokens' do token.similar_tokens_for(@category).each do |token| token.should_not be_similar end end it 'returns a list of tokens with the right text' do token.similar_tokens_for(@category).map(&:text).should == ['array', 'of', 'similar'] end it 'returns a list of tokens with the right original' do token.similar_tokens_for(@category).map(&:original).should == ['array', 'of', 'similar'] end it 'returns a list of tokens with the right categorization' do token.similar_tokens_for(@category).map(&:predefined_categories).should == [[@category], [@category], [@category]] end end context 'without similar' do before(:each) do @bundle = stub :bundle, :similar => [] @category = stub :category, :bundle_for => @bundle end it 'returns an empty list' do token.similar_tokens_for(@category).should == [] end end end # describe 'to_solr' do # def self.it_should_solr text, expected_result # it "should solrify into #{expected_result} from #{text}" do # described_class.processed(text).to_solr.should == expected_result # end # end # it_should_solr 's', 's' # it_should_solr 'se', 'se' # it_should_solr 'sea', 'sea' # it_should_solr 'sear', 'sear~0.74' # it_should_solr 'searc', 'searc~0.78' # it_should_solr 'search', 'search~0.81' # it_should_solr 'searche', 'searche~0.83' # it_should_solr 'searchen', 'searchen~0.85' # it_should_solr 'searcheng', 'searcheng~0.87' # it_should_solr 'searchengi', 'searchengi~0.89' # it_should_solr 'searchengin', 'searchengin~0.9' # it_should_solr 'searchengine', 'searchengine~0.9' # # it_should_solr 'spec:tex', 'specific:tex' # it_should_solr 'with:text', 'text~0.74' # it_should_solr 'name:', 'name~0.74' # it_should_solr '', '' # it_should_solr 'sp:tex', 'specific:tex' # it_should_solr 'sp:tex~', 'specific:tex' # it_should_solr 'sp:tex"', 'specific:tex' # end describe 'qualify' do def self.it_should_qualify text, expected_result it "should extract the qualifier #{expected_result} from #{text}" do token = described_class.new text token.qualify [token.instance_variable_get(:@qualifiers), token.text].should == expected_result end end it_should_qualify 'spec:qualifier', [['spec'], 'qualifier'] it_should_qualify 'with:qualifier', [['with'], 'qualifier'] it_should_qualify 'without qualifier', [nil, 'without qualifier'] it_should_qualify 'name:', [['name'], ''] it_should_qualify ':broken qualifier', [[], 'broken qualifier'] # Unsure about that. Probably should recognize it as text. it_should_qualify '', [nil, ''] it_should_qualify 'sp:text', [['sp'], 'text'] it_should_qualify '""', [nil, '""'] it_should_qualify 'name:', [['name'], ''] it_should_qualify 'name:hanke', [['name'], 'hanke'] it_should_qualify 'g:gaga', [['g'], 'gaga'] it_should_qualify ':nothing', [[], 'nothing'] it_should_qualify 'hello', [nil, 'hello'] it_should_qualify 'a:b:c', [['a'], 'b:c'] it_should_qualify 'a,b:c', [['a','b'], 'c'] it_should_qualify 'a,b,c:d', [['a','b','c'], 'd'] it_should_qualify ':', [[], ''] it_should_qualify 'vorname:qualifier', [['vorname'], 'qualifier'] end describe 'qualifier' do it 'returns the right thing' do token = described_class.processed 'b' token.qualifiers.should == [] end it 'returns the right thing' do token = described_class.processed 'a:b' token.qualifiers.should == ['a'] end it 'returns the right thing' do token = described_class.processed 'a,b:c' token.qualifiers.should == ['a', 'b'] end end describe 'processed' do it 'should return a new token' do described_class.processed('some text', 'SOME TEXT').should be_kind_of(described_class) end it 'generates a token' do described_class.processed('some text', 'SOME TEXT').class.should == described_class end end describe 'process' do let(:token) { described_class.new 'any_text' } it 'returns itself' do token.process.should == token end it 'should have an order' do token.should_receive(:qualify).once.ordered token.should_receive(:partialize).once.ordered token.should_receive(:similarize).once.ordered token.should_receive(:remove_illegals).once.ordered token.process end end describe 'partialize' do context 'token explicitly partial' do let(:token) { described_class.new 'a*' } it 'is afterwards partial' do token.partialize token.should be_partial end end context 'token explicitly nonpartial' do let(:token) { described_class.new 'a"' } it 'is afterwards not partial' do token.partialize token.should_not be_partial end end context 'token nonpartial' do let(:token) { described_class.new 'a' } it 'is afterwards not partial' do token.partialize token.should_not be_partial end end context 'special case' do let(:token) { described_class.new 'a*"' } it 'is afterwards not partial - last one wins' do token.partialize token.should_not be_partial end end context 'special case' do let(:token) { described_class.new 'a"*' } it 'is afterwards partial - last one wins' do token.partialize token.should be_partial end end end describe 'similarize' do context 'token explicitly similar' do let(:token) { described_class.new 'a~' } it 'is afterwards similar' do token.similarize token.should be_similar end end context 'token explicitly nonsimilar' do let(:token) { described_class.new 'a"' } it 'is afterwards not similar' do token.similarize token.should_not be_similar end end context 'token nonsimilar' do let(:token) { described_class.new 'a' } it 'is afterwards not similar' do token.similarize token.should_not be_similar end end end describe 'symbolize!' do before(:each) do @token = described_class.processed 'string', 'String' end it 'is not symbolized' do @token.text.should == 'string' end it 'can be symbolized' do @token.symbolize! @token.text.should == :string end end describe 'partial?' do context 'similar, partial' do before(:each) do @token = described_class.processed 'similar~', 'Similar~' @token.partial = true end it 'should be false' do @token.partial?.should == false end end context 'similar, not partial' do before(:each) do @token = described_class.processed 'similar~', 'Similar~' end it 'should be false' do @token.partial?.should == false end end context 'not similar, partial' do before(:each) do @token = described_class.processed 'not similar', 'NOT SIMILAR' @token.partial = true end it 'should be true' do @token.partial?.should == true end end context 'not similar, not partial' do before(:each) do @token = described_class.processed 'not similar', 'NOT SIMILAR' end it 'should be nil' do @token.partial?.should == nil end end end describe 'similar' do it 'should not change the original with the text' do token = described_class.processed "bla~", 'BLA~' token.text.should_not == token.original end def self.it_should_have_similarity text, expected_similarity_value it "should have #{ "no" unless expected_similarity_value } similarity for '#{text}'" do described_class.processed(text, text.upcase).similar?.should == expected_similarity_value end end it_should_have_similarity 'name:', nil it_should_have_similarity 'name:hanke', nil it_should_have_similarity 'g:gaga', nil it_should_have_similarity ':nothing', nil it_should_have_similarity 'hello', nil it_should_have_similarity 'a:b:c', nil it_should_have_similarity '""', false it_should_have_similarity '"exact~"', false # The tilde will just be removed it_should_have_similarity '~', true it_should_have_similarity '"hello"~', true # overrides the "" it_should_have_similarity 'philippe~', true end describe 'special cases' do it 'should be blank on ""' do token = described_class.processed '""', '""' token.should be_blank end end describe "original" do it "should keep the original text even when processed" do token = described_class.processed "I'm the processed text.", "I'm the original text." token.original.should == "I'm the original text." end end describe "blank?" do it "should be blank if the token text itself is blank" do token = described_class.processed '' token.blank?.should be_true end it "should be non-blank if the token text itself is non-blank" do token = described_class.processed 'not empty' token.blank?.should be_false end end describe "to_s" do describe "with qualifier" do it "should display qualifier and text combined with a ':'" do token = described_class.processed('sp:qualifier') token.to_s.should == 'Picky::Query::Token(qualifier, ["sp"])' end end describe "without qualifier" do it "should display just the text" do token = described_class.processed('text') token.to_s.should == 'Picky::Query::Token(text)' end end end describe 'qualifiers' do context 'with qualifier' do let(:token) { described_class.processed 'sp:qualifier' } it 'should return the qualifier' do token.qualifiers.should == ['sp'] end end context 'with incorrect qualifier' do let(:token) { described_class.processed 'specific:qualifier' } it 'should return the qualifier' do token.qualifiers.should == ['specific'] end end context 'with multiple qualifiers' do let(:token) { described_class.processed 'sp,spec:qualifier' } it 'should return the qualifier' do token.qualifiers.should == ['sp', 'spec'] end end context 'without qualifier' do let(:token) { described_class.processed 'noqualifier' } it 'is correct' do token.qualifiers.should == [] end end end describe 'partial=' do context 'partial nil' do let(:token) { described_class.new 'text' } it 'should set partial' do token.partial = true token.should be_partial end it 'should set partial' do token.partial = false token.should_not be_partial end end context 'partial not nil' do let(:token) { described_class.processed 'text"' } it 'should not set partial' do token.partial = true token.should_not be_partial end end context 'partial not nil' do let(:token) { described_class.processed 'text*' } it 'should not set partial' do token.instance_variable_set :@partial, false token.partial = true token.should_not be_partial end it 'should not set partial' do token.partial = false token.should be_partial end end end describe 'partialize!' do it 'should not partialize a token if the text ends with "' do token = described_class.processed 'text"' token.should_not be_partial end it 'should partialize a token if the text ends with *' do token = described_class.processed 'text*' token.should be_partial end it 'should not partialize a token if the text ends with ~' do token = described_class.processed 'text~' token.should_not be_partial end it 'lets the last one win' do token = described_class.processed 'text"*' token.should be_partial end it 'lets the last one win' do token = described_class.processed 'text*"' token.should_not be_partial end end describe "processed" do it 'should remove *' do token = described_class.processed 'text*' token.text.should == 'text' end it 'should remove ~' do token = described_class.processed 'text~' token.text.should == 'text' end it 'should remove "' do token = described_class.processed 'text"' token.text.should == 'text' end it "should pass on a processed text" do described_class.processed('text').text.should == 'text' end end end