-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathTUTORIAL
231 lines (166 loc) · 9.03 KB
/
TUTORIAL
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
= Quick Introduction to Ferret
The simplest way to use Ferret is through the Ferret::Index::Index class.
This is now aliased by Ferret::I for quick and easy access. Start by including
the Ferret module.
require 'ferret'
include Ferret
=== Creating an index
To create an in memory index is very simple;
index = Index::Index.new()
To create a persistent index;
index = Index::Index.new(:path => '/path/to/index')
Both of these methods create new Indexes with the StandardAnalyzer. An
analyzer is what you use to divide the input data up into tokens which you can
search for later. If you'd like to use a different analyzer you can specify it
here, eg;
index = Index::Index.new(:path => '/path/to/index',
:analyzer => Analysis::WhiteSpaceAnalyzer.new)
For more options when creating an Index refer to Ferret::Index::Index.
=== Adding Documents
To add a document you can simply add a string or an array of strings. This will
store all the strings in the "" (ie empty string) field (unless you specify the
default field when you create the index).
index << "This is a new document to be indexed"
index << ["And here", "is another", "new document", "to be indexed"]
But these are pretty simple documents. If this is all you want to index you
could probably just use SimpleSearch. So let's give our documents some fields;
index << {:title => "Programming Ruby", :content => "blah blah blah"}
index << {:title => "Programming Ruby", :content => "yada yada yada"}
Note the way that all field-names are Symbols. Although Strings will work,
this is a best-practice in Ferret. Or if you are indexing data stored in a
database, you'll probably want to store the id;
index << {:id => row.id, :title => row.title, :date => row.date}
So far we have been storing and tokenizing all of the input data along with
term vectors. If we want to change this we need to change the way we setup the
index. You must create a FieldInfos object describing the index:
field_infos = FieldInfos.new(:store => :no,
:index => :untokenized_omit_norms,
:term_vector => :no)
The values that you set FieldInfos to have will be used by default by all
fields. If you want to change the properties for specific fields, you need to
add a FieldInfo to field_infos.
field_infos.add_field(:title, :store => :yes, :index => :yes, :boost => 10.0)
field_infos.add_field(:content, :store => :yes,
:index => :yes,
:term_vector => :with_positions_offsets)
If you need to add a field to an already open index you do so like this:
index.field_infos.add_field(:new_field, :store => :yes)
=== Searching
Now that we have data in our index, how do we actually use this index to
search the data? The Index offers two search methods, Index#search and
Index#search_each. The first method returns a Ferret::Index::TopDocs object.
The second we'll show here. Lets say we wanted to find all documents with the
phrase "quick brown fox" in the content field. We'd write;
index.search_each('content:"quick brown fox"') do |id, score|
puts "Document #{id} found with a score of #{score}"
end
But "fast" has a pretty similar meaning to "quick" and we don't mind if the
fox is a little red. Also, the phrase could be in the title so we'll search
there as well. So we could expand our search like this;
index.search_each('title|content:"quick|fast brown|red fox"') do |id, score|
puts "Document #{id} found with a score of #{score}"
end
What if we want to find all documents entered on or after 5th of September,
2005 with the words "ruby" or "rails" in any field. We could type something like;
index.search_each('date:( >= 20050905) *:(ruby OR rails)') do |id, score|
puts "Document #{index[id][:title]} found with a score of #{score}"
end
Ferret has quite a complex query language. To find out more about Ferret's
query language, see Ferret::QueryParser. You can also construct even more
complex queries like Ferret::Search::Spans by hand. See Ferret::Search::Query
for more information.
=== Highlighting
Ferret now has a super-fast highlighting method. See
Ferret::Index::Index#highlight. Here is an example of how you would use it
when printing to the console:
index.search_each('date:( >= 20050905) content:(ruby OR rails)') do |id, score|
puts "Document #{index[id][:title]} found with a score of #{score}"
highlights = index.highlight("content:(ruby OR rails)", 0,
:field => :content,
:pre_tag = "\033[36m",
:post_tag = "\033[m")
puts highlights
end
And if you want to highlight a whole document, set :excerpt_length to :all:
puts index.highlight(query, doc_id,
:field => :content,
:pre_tag = "\033[36m",
:post_tag = "\033[m",
:excerpt_length => :all)
=== Accessing Documents
You may have noticed that when we run a search we only get the document id
back. By itself this isn't much use to us. Getting the data from the index is
very straightforward. For example if we want the :title field form the 3rd
document type;
index[2][:title]
Documents are lazy loading so if you try this:
puts index[2]
You will always get an empty Hash. To load all fields, call the load method:
puts index[2].load
NOTE: documents are indexed from 0. You can also use array-like index
parameters to access index. For example
index[1..4]
index[10, 10]
index[-5]
The default field is :id (although you can change this with index's
:default_create_field parameter);
index << "This is a document"
index[0][:id]
Let's go back to the database example above. If we store all of our documents
with an id then we can access that field using the id. As long as we called
our id field :id we can do this
index["89721347"]["title"]
Pretty simple huh? You should note though that if there are more then one
document with the same *id* or *key* then only the first one will be returned
so it is probably better that you ensure the key is unique somehow. By setting
Index's :key attribute to :id, Ferret will do this automatically for you. It
can even handle multiple field primary keys. For example, you could set to
:key to [:id, :model] and Ferret would keep the documents unique for that pair
of fields.
=== Modifying and Deleting Documents
What if we want to change the data in the index. Ferret doesn't actually let
you change the data once it is in the index. But you can delete documents so
the standard way to modify data is to delete it and re-add it again with the
modifications made. It is important to note that when doing this the documents
will get a new document number so you should be careful not to use a document
number after the document has been deleted. Here is an example of modifying a
document;
index << {:title => "Programing Rbuy", :content => "blah blah blah"}
doc_num = nil
index.search_each('title:"Programing Rbuy"') {|id, score| doc_id = id}
return unless doc_id
doc = index[doc_id]
index.delete(doc_id)
# modify doc. It is just a Hash after all
doc[:title] = "Programming Ruby"
index << doc
If you set the :key parameter as described in the last section there is no
need to delete the document. It will be automatically deleted when you add
another document with the same key.
Also, we can use the id field, as above, to delete documents. This time though
every document that matches the id will be deleted. Again, it is probably a
good idea if you somehow ensure that your *ids* are kept unique.
id = "23453422"
index.delete(id)
=== Onwards
This is just a small sampling of what Ferret allows you to do. Ferret, like
Lucene, is designed to be extended, and allows you to construct your own query
types, analyzers, and so on. Going onwards you should check out the following
documentation:
* Ferret::Analysis: for more information on how the data is processed when it
is tokenized. There are a number of things you can do with your data such as
adding stop lists or perhaps a porter stemmer. There are also a number of
analyzers already available and it is almost trivial to create a new one
with a simple regular expression.
* Ferret::Search: for more information on querying the index. There are a
number of already available queries and it's unlikely you'll need to create
your own. You may however want to take advantage of the sorting or filtering
abilities of Ferret to present your data the best way you see fit.
* Ferret::QueryParser: if you want to find out more about what you can do with
Ferret's Query Parser, this is the place to look. The query parser is one
area that could use a bit of work so please send your suggestions.
* Ferret::Index: for more advanced access to the index you'll probably want to
use the Ferret::Index::IndexWriter and Ferret::Index::IndexReader. This is
the place to look for more information on them.
* Ferret::Store: This is the module used to access the actual index storage
and won't be of much interest to most people.