ferret 0.9.6 → 0.10.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/MIT-LICENSE +1 -1
- data/README +12 -24
- data/Rakefile +38 -54
- data/TODO +14 -17
- data/ext/analysis.c +982 -823
- data/ext/analysis.h +133 -76
- data/ext/array.c +96 -58
- data/ext/array.h +40 -13
- data/ext/bitvector.c +476 -118
- data/ext/bitvector.h +264 -22
- data/ext/compound_io.c +217 -229
- data/ext/defines.h +49 -0
- data/ext/document.c +107 -317
- data/ext/document.h +31 -65
- data/ext/except.c +81 -36
- data/ext/except.h +117 -55
- data/ext/extconf.rb +2 -9
- data/ext/ferret.c +211 -104
- data/ext/ferret.h +22 -11
- data/ext/filter.c +97 -82
- data/ext/fs_store.c +348 -367
- data/ext/global.c +226 -188
- data/ext/global.h +44 -26
- data/ext/hash.c +474 -391
- data/ext/hash.h +441 -68
- data/ext/hashset.c +124 -96
- data/ext/hashset.h +169 -20
- data/ext/helper.c +56 -5
- data/ext/helper.h +7 -0
- data/ext/inc/lang.h +29 -49
- data/ext/inc/threading.h +31 -0
- data/ext/ind.c +288 -278
- data/ext/ind.h +68 -0
- data/ext/index.c +5688 -0
- data/ext/index.h +663 -616
- data/ext/lang.h +29 -49
- data/ext/libstemmer.c +3 -3
- data/ext/mem_pool.c +84 -0
- data/ext/mem_pool.h +35 -0
- data/ext/posh.c +1006 -0
- data/ext/posh.h +1007 -0
- data/ext/priorityqueue.c +117 -194
- data/ext/priorityqueue.h +135 -39
- data/ext/q_boolean.c +1305 -1108
- data/ext/q_const_score.c +106 -93
- data/ext/q_filtered_query.c +138 -135
- data/ext/q_fuzzy.c +206 -242
- data/ext/q_match_all.c +94 -80
- data/ext/q_multi_term.c +663 -0
- data/ext/q_parser.c +667 -593
- data/ext/q_phrase.c +992 -555
- data/ext/q_prefix.c +72 -61
- data/ext/q_range.c +235 -210
- data/ext/q_span.c +1480 -1166
- data/ext/q_term.c +273 -246
- data/ext/q_wildcard.c +127 -114
- data/ext/r_analysis.c +1720 -711
- data/ext/r_index.c +3049 -0
- data/ext/r_qparser.c +433 -146
- data/ext/r_search.c +2934 -1993
- data/ext/r_store.c +372 -143
- data/ext/r_utils.c +941 -0
- data/ext/ram_store.c +330 -326
- data/ext/search.c +1291 -668
- data/ext/search.h +403 -702
- data/ext/similarity.c +91 -113
- data/ext/similarity.h +45 -30
- data/ext/sort.c +721 -484
- data/ext/stopwords.c +361 -273
- data/ext/store.c +556 -58
- data/ext/store.h +706 -126
- data/ext/tags +3578 -2780
- data/ext/term_vectors.c +352 -0
- data/ext/threading.h +31 -0
- data/ext/win32.h +54 -0
- data/lib/ferret.rb +5 -17
- data/lib/ferret/document.rb +130 -2
- data/lib/ferret/index.rb +577 -26
- data/lib/ferret/number_tools.rb +157 -0
- data/lib/ferret_version.rb +3 -0
- data/test/test_helper.rb +5 -13
- data/test/unit/analysis/tc_analyzer.rb +513 -1
- data/test/unit/analysis/{ctc_tokenstream.rb → tc_token_stream.rb} +23 -0
- data/test/unit/index/tc_index.rb +183 -240
- data/test/unit/index/tc_index_reader.rb +312 -479
- data/test/unit/index/tc_index_writer.rb +397 -13
- data/test/unit/index/th_doc.rb +269 -206
- data/test/unit/query_parser/tc_query_parser.rb +40 -33
- data/test/unit/search/tc_filter.rb +59 -71
- data/test/unit/search/tc_fuzzy_query.rb +24 -16
- data/test/unit/search/tc_index_searcher.rb +23 -201
- data/test/unit/search/tc_multi_searcher.rb +78 -226
- data/test/unit/search/tc_search_and_sort.rb +93 -81
- data/test/unit/search/tc_sort.rb +23 -23
- data/test/unit/search/tc_sort_field.rb +7 -7
- data/test/unit/search/tc_spans.rb +51 -47
- data/test/unit/search/tm_searcher.rb +339 -0
- data/test/unit/store/tc_fs_store.rb +1 -1
- data/test/unit/store/tm_store_lock.rb +3 -3
- data/test/unit/tc_document.rb +81 -0
- data/test/unit/ts_analysis.rb +1 -1
- data/test/unit/ts_utils.rb +1 -1
- data/test/unit/utils/tc_bit_vector.rb +288 -0
- data/test/unit/utils/tc_number_tools.rb +117 -0
- data/test/unit/utils/tc_priority_queue.rb +106 -0
- metadata +140 -301
- data/CHANGELOG +0 -9
- data/ext/dummy.exe +0 -0
- data/ext/field.c +0 -408
- data/ext/frtio.h +0 -13
- data/ext/inc/except.h +0 -90
- data/ext/index_io.c +0 -382
- data/ext/index_rw.c +0 -2658
- data/ext/lang.c +0 -41
- data/ext/nix_io.c +0 -134
- data/ext/q_multi_phrase.c +0 -380
- data/ext/r_doc.c +0 -582
- data/ext/r_index_io.c +0 -1021
- data/ext/r_term.c +0 -219
- data/ext/term.c +0 -820
- data/ext/termdocs.c +0 -611
- data/ext/vector.c +0 -637
- data/ext/w32_io.c +0 -150
- data/lib/ferret/analysis.rb +0 -11
- data/lib/ferret/analysis/analyzers.rb +0 -112
- data/lib/ferret/analysis/standard_tokenizer.rb +0 -71
- data/lib/ferret/analysis/token.rb +0 -100
- data/lib/ferret/analysis/token_filters.rb +0 -86
- data/lib/ferret/analysis/token_stream.rb +0 -26
- data/lib/ferret/analysis/tokenizers.rb +0 -112
- data/lib/ferret/analysis/word_list_loader.rb +0 -27
- data/lib/ferret/document/document.rb +0 -152
- data/lib/ferret/document/field.rb +0 -312
- data/lib/ferret/index/compound_file_io.rb +0 -338
- data/lib/ferret/index/document_writer.rb +0 -289
- data/lib/ferret/index/field_infos.rb +0 -279
- data/lib/ferret/index/fields_io.rb +0 -181
- data/lib/ferret/index/index.rb +0 -675
- data/lib/ferret/index/index_file_names.rb +0 -33
- data/lib/ferret/index/index_reader.rb +0 -503
- data/lib/ferret/index/index_writer.rb +0 -534
- data/lib/ferret/index/multi_reader.rb +0 -377
- data/lib/ferret/index/multiple_term_doc_pos_enum.rb +0 -98
- data/lib/ferret/index/segment_infos.rb +0 -130
- data/lib/ferret/index/segment_merge_info.rb +0 -49
- data/lib/ferret/index/segment_merge_queue.rb +0 -16
- data/lib/ferret/index/segment_merger.rb +0 -358
- data/lib/ferret/index/segment_reader.rb +0 -412
- data/lib/ferret/index/segment_term_enum.rb +0 -169
- data/lib/ferret/index/segment_term_vector.rb +0 -58
- data/lib/ferret/index/term.rb +0 -53
- data/lib/ferret/index/term_buffer.rb +0 -83
- data/lib/ferret/index/term_doc_enum.rb +0 -291
- data/lib/ferret/index/term_enum.rb +0 -52
- data/lib/ferret/index/term_info.rb +0 -37
- data/lib/ferret/index/term_infos_io.rb +0 -321
- data/lib/ferret/index/term_vector_offset_info.rb +0 -20
- data/lib/ferret/index/term_vectors_io.rb +0 -553
- data/lib/ferret/query_parser.rb +0 -312
- data/lib/ferret/query_parser/query_parser.tab.rb +0 -928
- data/lib/ferret/search.rb +0 -50
- data/lib/ferret/search/boolean_clause.rb +0 -100
- data/lib/ferret/search/boolean_query.rb +0 -299
- data/lib/ferret/search/boolean_scorer.rb +0 -294
- data/lib/ferret/search/caching_wrapper_filter.rb +0 -40
- data/lib/ferret/search/conjunction_scorer.rb +0 -99
- data/lib/ferret/search/disjunction_sum_scorer.rb +0 -205
- data/lib/ferret/search/exact_phrase_scorer.rb +0 -32
- data/lib/ferret/search/explanation.rb +0 -41
- data/lib/ferret/search/field_cache.rb +0 -215
- data/lib/ferret/search/field_doc.rb +0 -31
- data/lib/ferret/search/field_sorted_hit_queue.rb +0 -184
- data/lib/ferret/search/filter.rb +0 -11
- data/lib/ferret/search/filtered_query.rb +0 -130
- data/lib/ferret/search/filtered_term_enum.rb +0 -79
- data/lib/ferret/search/fuzzy_query.rb +0 -154
- data/lib/ferret/search/fuzzy_term_enum.rb +0 -247
- data/lib/ferret/search/hit_collector.rb +0 -34
- data/lib/ferret/search/hit_queue.rb +0 -11
- data/lib/ferret/search/index_searcher.rb +0 -200
- data/lib/ferret/search/match_all_query.rb +0 -104
- data/lib/ferret/search/multi_phrase_query.rb +0 -216
- data/lib/ferret/search/multi_searcher.rb +0 -261
- data/lib/ferret/search/multi_term_query.rb +0 -65
- data/lib/ferret/search/non_matching_scorer.rb +0 -22
- data/lib/ferret/search/phrase_positions.rb +0 -55
- data/lib/ferret/search/phrase_query.rb +0 -214
- data/lib/ferret/search/phrase_scorer.rb +0 -152
- data/lib/ferret/search/prefix_query.rb +0 -54
- data/lib/ferret/search/query.rb +0 -140
- data/lib/ferret/search/query_filter.rb +0 -51
- data/lib/ferret/search/range_filter.rb +0 -103
- data/lib/ferret/search/range_query.rb +0 -139
- data/lib/ferret/search/req_excl_scorer.rb +0 -125
- data/lib/ferret/search/req_opt_sum_scorer.rb +0 -70
- data/lib/ferret/search/score_doc.rb +0 -38
- data/lib/ferret/search/score_doc_comparator.rb +0 -114
- data/lib/ferret/search/scorer.rb +0 -91
- data/lib/ferret/search/similarity.rb +0 -278
- data/lib/ferret/search/sloppy_phrase_scorer.rb +0 -47
- data/lib/ferret/search/sort.rb +0 -112
- data/lib/ferret/search/sort_comparator.rb +0 -60
- data/lib/ferret/search/sort_field.rb +0 -91
- data/lib/ferret/search/spans.rb +0 -12
- data/lib/ferret/search/spans/near_spans_enum.rb +0 -304
- data/lib/ferret/search/spans/span_first_query.rb +0 -79
- data/lib/ferret/search/spans/span_near_query.rb +0 -108
- data/lib/ferret/search/spans/span_not_query.rb +0 -130
- data/lib/ferret/search/spans/span_or_query.rb +0 -176
- data/lib/ferret/search/spans/span_query.rb +0 -25
- data/lib/ferret/search/spans/span_scorer.rb +0 -74
- data/lib/ferret/search/spans/span_term_query.rb +0 -105
- data/lib/ferret/search/spans/span_weight.rb +0 -84
- data/lib/ferret/search/spans/spans_enum.rb +0 -44
- data/lib/ferret/search/term_query.rb +0 -128
- data/lib/ferret/search/term_scorer.rb +0 -183
- data/lib/ferret/search/top_docs.rb +0 -36
- data/lib/ferret/search/top_field_docs.rb +0 -17
- data/lib/ferret/search/weight.rb +0 -54
- data/lib/ferret/search/wildcard_query.rb +0 -26
- data/lib/ferret/search/wildcard_term_enum.rb +0 -61
- data/lib/ferret/stemmers.rb +0 -1
- data/lib/ferret/stemmers/porter_stemmer.rb +0 -218
- data/lib/ferret/store.rb +0 -5
- data/lib/ferret/store/buffered_index_io.rb +0 -190
- data/lib/ferret/store/directory.rb +0 -141
- data/lib/ferret/store/fs_store.rb +0 -381
- data/lib/ferret/store/index_io.rb +0 -245
- data/lib/ferret/store/ram_store.rb +0 -286
- data/lib/ferret/utils.rb +0 -8
- data/lib/ferret/utils/bit_vector.rb +0 -123
- data/lib/ferret/utils/date_tools.rb +0 -138
- data/lib/ferret/utils/number_tools.rb +0 -91
- data/lib/ferret/utils/parameter.rb +0 -41
- data/lib/ferret/utils/priority_queue.rb +0 -120
- data/lib/ferret/utils/string_helper.rb +0 -47
- data/lib/ferret/utils/thread_local.rb +0 -28
- data/lib/ferret/utils/weak_key_hash.rb +0 -60
- data/lib/rferret.rb +0 -37
- data/rake_utils/code_statistics.rb +0 -106
- data/test/benchmark/tb_ram_store.rb +0 -76
- data/test/benchmark/tb_rw_vint.rb +0 -26
- data/test/functional/thread_safety_index_test.rb +0 -81
- data/test/functional/thread_safety_test.rb +0 -137
- data/test/longrunning/tc_numbertools.rb +0 -60
- data/test/longrunning/tm_store.rb +0 -19
- data/test/unit/analysis/ctc_analyzer.rb +0 -532
- data/test/unit/analysis/data/wordfile +0 -6
- data/test/unit/analysis/rtc_letter_tokenizer.rb +0 -20
- data/test/unit/analysis/rtc_lower_case_filter.rb +0 -20
- data/test/unit/analysis/rtc_lower_case_tokenizer.rb +0 -27
- data/test/unit/analysis/rtc_per_field_analyzer_wrapper.rb +0 -39
- data/test/unit/analysis/rtc_porter_stem_filter.rb +0 -16
- data/test/unit/analysis/rtc_standard_analyzer.rb +0 -20
- data/test/unit/analysis/rtc_standard_tokenizer.rb +0 -20
- data/test/unit/analysis/rtc_stop_analyzer.rb +0 -20
- data/test/unit/analysis/rtc_stop_filter.rb +0 -14
- data/test/unit/analysis/rtc_white_space_analyzer.rb +0 -21
- data/test/unit/analysis/rtc_white_space_tokenizer.rb +0 -20
- data/test/unit/analysis/rtc_word_list_loader.rb +0 -32
- data/test/unit/analysis/tc_token.rb +0 -25
- data/test/unit/document/rtc_field.rb +0 -28
- data/test/unit/document/tc_document.rb +0 -47
- data/test/unit/document/tc_field.rb +0 -98
- data/test/unit/index/rtc_compound_file_io.rb +0 -107
- data/test/unit/index/rtc_field_infos.rb +0 -127
- data/test/unit/index/rtc_fields_io.rb +0 -167
- data/test/unit/index/rtc_multiple_term_doc_pos_enum.rb +0 -83
- data/test/unit/index/rtc_segment_infos.rb +0 -74
- data/test/unit/index/rtc_segment_term_docs.rb +0 -17
- data/test/unit/index/rtc_segment_term_enum.rb +0 -60
- data/test/unit/index/rtc_segment_term_vector.rb +0 -71
- data/test/unit/index/rtc_term_buffer.rb +0 -57
- data/test/unit/index/rtc_term_info.rb +0 -19
- data/test/unit/index/rtc_term_infos_io.rb +0 -192
- data/test/unit/index/rtc_term_vectors_io.rb +0 -108
- data/test/unit/index/tc_term.rb +0 -27
- data/test/unit/index/tc_term_voi.rb +0 -18
- data/test/unit/search/rtc_similarity.rb +0 -37
- data/test/unit/search/rtc_sort_field.rb +0 -14
- data/test/unit/search/tc_multi_searcher2.rb +0 -126
- data/test/unit/store/rtc_fs_store.rb +0 -62
- data/test/unit/store/rtc_ram_store.rb +0 -15
- data/test/unit/store/rtm_store.rb +0 -150
- data/test/unit/store/rtm_store_lock.rb +0 -2
- data/test/unit/ts_document.rb +0 -2
- data/test/unit/utils/rtc_bit_vector.rb +0 -73
- data/test/unit/utils/rtc_date_tools.rb +0 -50
- data/test/unit/utils/rtc_number_tools.rb +0 -59
- data/test/unit/utils/rtc_parameter.rb +0 -40
- data/test/unit/utils/rtc_priority_queue.rb +0 -62
- data/test/unit/utils/rtc_string_helper.rb +0 -21
- data/test/unit/utils/rtc_thread.rb +0 -61
- data/test/unit/utils/rtc_weak_key_hash.rb +0 -25
- data/test/utils/number_to_spoken.rb +0 -132
data/ext/filter.c
CHANGED
@@ -7,66 +7,76 @@
|
|
7
7
|
*
|
8
8
|
***************************************************************************/
|
9
9
|
|
10
|
-
void
|
10
|
+
void filt_destroy_i(Filter *filt)
|
11
11
|
{
|
12
|
-
|
13
|
-
|
14
|
-
|
12
|
+
h_destroy(filt->cache);
|
13
|
+
free(filt->name);
|
14
|
+
free(filt);
|
15
|
+
}
|
16
|
+
void filt_deref(Filter *filt)
|
17
|
+
{
|
18
|
+
if (--(filt->ref_cnt) == 0) {
|
19
|
+
filt->destroy_i(filt);
|
20
|
+
}
|
15
21
|
}
|
16
22
|
|
17
|
-
BitVector *filt_get_bv(Filter *
|
23
|
+
BitVector *filt_get_bv(Filter *filt, IndexReader *ir)
|
18
24
|
{
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
25
|
+
CacheObject *co = h_get(filt->cache, ir);
|
26
|
+
|
27
|
+
if (!co) {
|
28
|
+
BitVector *bv;
|
29
|
+
if (!ir->cache) {
|
30
|
+
ir_add_cache(ir);
|
31
|
+
}
|
32
|
+
bv = filt->get_bv_i(filt, ir);
|
33
|
+
co = co_create(filt->cache, ir->cache, filt, ir,
|
34
|
+
(free_ft)&bv_destroy, (void *)bv);
|
24
35
|
}
|
25
|
-
|
26
|
-
co = co_create(self->cache, ir->cache, self, ir,
|
27
|
-
(free_ft)&bv_destroy, (void *)bv);
|
28
|
-
}
|
29
|
-
return (BitVector *)co->obj;
|
36
|
+
return (BitVector *)co->obj;
|
30
37
|
}
|
31
38
|
|
32
|
-
char *filt_to_s_i(Filter *
|
39
|
+
static char *filt_to_s_i(Filter *filt)
|
33
40
|
{
|
34
|
-
|
41
|
+
return estrdup(filt->name);
|
35
42
|
}
|
36
43
|
|
37
|
-
|
44
|
+
ulong filt_hash_default(Filter *filt)
|
38
45
|
{
|
39
|
-
|
46
|
+
(void)filt;
|
47
|
+
return 0;
|
40
48
|
}
|
41
49
|
|
42
|
-
int
|
50
|
+
int filt_eq_default(Filter *filt, Filter *o)
|
43
51
|
{
|
44
|
-
|
52
|
+
(void)filt; (void)o;
|
53
|
+
return false;
|
45
54
|
}
|
46
55
|
|
47
|
-
Filter *filt_create(char *name)
|
56
|
+
Filter *filt_create(size_t size, const char *name)
|
48
57
|
{
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
58
|
+
Filter *filt = (Filter *)emalloc(size);
|
59
|
+
filt->cache = co_hash_create();
|
60
|
+
filt->name = estrdup(name);
|
61
|
+
filt->to_s = &filt_to_s_i;
|
62
|
+
filt->hash = &filt_hash_default;
|
63
|
+
filt->eq = &filt_eq_default;
|
64
|
+
filt->destroy_i = &filt_destroy_i;
|
65
|
+
filt->ref_cnt = 1;
|
66
|
+
return filt;
|
57
67
|
}
|
58
68
|
|
59
|
-
|
69
|
+
ulong filt_hash(Filter *filt)
|
60
70
|
{
|
61
|
-
|
71
|
+
return str_hash(filt->name) ^ filt->hash(filt);
|
62
72
|
}
|
63
73
|
|
64
|
-
int filt_eq(Filter *
|
74
|
+
int filt_eq(Filter *filt, Filter *o)
|
65
75
|
{
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
|
76
|
+
return ((filt == o)
|
77
|
+
|| ((strcmp(filt->name, o->name) == 0)
|
78
|
+
&& (filt->eq == o->eq)
|
79
|
+
&& (filt->eq(filt, o))));
|
70
80
|
}
|
71
81
|
|
72
82
|
/***************************************************************************
|
@@ -75,67 +85,72 @@ int filt_eq(Filter *self, Filter *o)
|
|
75
85
|
*
|
76
86
|
***************************************************************************/
|
77
87
|
|
78
|
-
|
88
|
+
#define QF(filt) ((QueryFilter *)(filt))
|
89
|
+
typedef struct QueryFilter
|
90
|
+
{
|
91
|
+
Filter super;
|
92
|
+
Query *query;
|
93
|
+
} QueryFilter;
|
94
|
+
|
95
|
+
static char *qfilt_to_s(Filter *filt)
|
79
96
|
{
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
return filter_str;
|
97
|
+
Query *query = QF(filt)->query;
|
98
|
+
char *query_str = query->to_s(query, "");
|
99
|
+
char *filter_str = strfmt("QueryFilter< %s >", query_str);
|
100
|
+
free(query_str);
|
101
|
+
return filter_str;
|
86
102
|
}
|
87
103
|
|
88
|
-
BitVector *
|
104
|
+
static BitVector *qfilt_get_bv_i(Filter *filt, IndexReader *ir)
|
89
105
|
{
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
106
|
+
BitVector *bv = bv_new_capa(ir->max_doc(ir));
|
107
|
+
Searcher *sea = isea_new(ir);
|
108
|
+
Weight *weight = q_weight(QF(filt)->query, sea);
|
109
|
+
Scorer *scorer = weight->scorer(weight, ir);
|
110
|
+
if (scorer) {
|
111
|
+
while (scorer->next(scorer)) {
|
112
|
+
bv_set(bv, scorer->doc);
|
113
|
+
}
|
114
|
+
scorer->destroy(scorer);
|
98
115
|
}
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
free(sea);
|
103
|
-
return bv;
|
116
|
+
weight->destroy(weight);
|
117
|
+
free(sea);
|
118
|
+
return bv;
|
104
119
|
}
|
105
120
|
|
106
|
-
|
121
|
+
static ulong qfilt_hash(Filter *filt)
|
107
122
|
{
|
108
|
-
|
109
|
-
return q_hash(query);
|
123
|
+
return q_hash(QF(filt)->query);
|
110
124
|
}
|
111
125
|
|
112
|
-
int qfilt_eq(Filter *
|
126
|
+
static int qfilt_eq(Filter *filt, Filter *o)
|
113
127
|
{
|
114
|
-
|
115
|
-
Query *query2 = (Query *)((QueryFilter *)o->data)->query;
|
116
|
-
return q_eq(query1, query2);
|
128
|
+
return q_eq(QF(filt)->query, QF(o)->query);
|
117
129
|
}
|
118
130
|
|
119
|
-
void
|
131
|
+
static void qfilt_destroy_i(Filter *filt)
|
120
132
|
{
|
121
|
-
|
122
|
-
|
123
|
-
|
124
|
-
filt_destroy(self);
|
133
|
+
Query *query = QF(filt)->query;
|
134
|
+
q_deref(query);
|
135
|
+
filt_destroy_i(filt);
|
125
136
|
}
|
126
137
|
|
127
|
-
Filter *
|
138
|
+
Filter *qfilt_new_nr(Query *query)
|
128
139
|
{
|
129
|
-
|
130
|
-
QueryFilter *qfilt = ALLOC(QueryFilter);
|
131
|
-
qfilt->query = query;
|
140
|
+
Filter *filt = filt_new(QueryFilter);
|
132
141
|
|
133
|
-
|
134
|
-
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
139
|
-
|
140
|
-
|
142
|
+
QF(filt)->query = query;
|
143
|
+
|
144
|
+
filt->get_bv_i = &qfilt_get_bv_i;
|
145
|
+
filt->hash = &qfilt_hash;
|
146
|
+
filt->eq = &qfilt_eq;
|
147
|
+
filt->to_s = &qfilt_to_s;
|
148
|
+
filt->destroy_i = &qfilt_destroy_i;
|
149
|
+
return filt;
|
150
|
+
}
|
151
|
+
|
152
|
+
Filter *qfilt_new(Query *query)
|
153
|
+
{
|
154
|
+
REF(query);
|
155
|
+
return qfilt_new_nr(query);
|
141
156
|
}
|
data/ext/fs_store.c
CHANGED
@@ -1,478 +1,459 @@
|
|
1
1
|
#include "store.h"
|
2
|
-
#include "frtio.h"
|
3
|
-
#ifdef WIN32
|
4
|
-
# ifndef S_IRUSR
|
5
|
-
# define S_IRUSR _S_IREAD
|
6
|
-
# endif
|
7
|
-
# ifndef S_IWUSR
|
8
|
-
# define S_IWUSR _S_IWRITE
|
9
|
-
# endif
|
10
|
-
# include <io.h>
|
11
|
-
#else
|
12
|
-
#include <unistd.h>
|
13
|
-
# define _close close
|
14
|
-
# define _open open
|
15
|
-
# define _creat creat
|
16
|
-
# define _lseek lseek
|
17
|
-
# define _read read
|
18
|
-
# define _stat stat
|
19
|
-
# define _fstat fstat
|
20
|
-
#endif
|
21
2
|
#include <sys/types.h>
|
22
3
|
#include <fcntl.h>
|
23
4
|
#include <sys/stat.h>
|
24
5
|
#include <errno.h>
|
25
6
|
#include <string.h>
|
26
7
|
#include <stdio.h>
|
8
|
+
#ifdef POSH_OS_WIN32
|
9
|
+
# include <io.h>
|
10
|
+
# include "win32.h"
|
11
|
+
# ifndef sleep
|
12
|
+
# define sleep _sleep
|
13
|
+
# endif
|
14
|
+
# ifndef DIR_SEPARATOR
|
15
|
+
# define DIR_SEPARATOR "\\"
|
16
|
+
# endif
|
17
|
+
# ifndef S_IRUSR
|
18
|
+
# define S_IRUSR _S_IREAD
|
19
|
+
# endif
|
20
|
+
# ifndef S_IWUSR
|
21
|
+
# define S_IWUSR _S_IWRITE
|
22
|
+
# endif
|
23
|
+
#else
|
24
|
+
# define DIR_SEPARATOR "/"
|
25
|
+
# include <unistd.h>
|
26
|
+
# include <dirent.h>
|
27
|
+
#endif
|
28
|
+
#ifndef O_BINARY
|
29
|
+
# define O_BINARY 0
|
30
|
+
#endif
|
27
31
|
|
28
|
-
|
29
|
-
static char * const FILE_OPEN_ERROR_MSG = "Couldn't open the file to read";
|
30
|
-
static char * const SEEK_ERROR_MSG = "Seek error message";
|
31
|
-
static char * const WRITE_ERROR_MSG = "Write error message";
|
32
|
-
|
32
|
+
extern Store *store_new();
|
33
33
|
extern void store_destroy(Store *store);
|
34
|
+
extern OutStream *os_new();
|
35
|
+
extern InStream *is_new();
|
36
|
+
extern int file_is_lock(char *filename);
|
34
37
|
|
35
38
|
/**
|
36
|
-
* Create
|
37
|
-
*
|
38
|
-
* @param store the store to create the file in
|
39
|
-
* @param filename the name of the file to create
|
40
|
-
* @throws IO_ERROR if the file cannot be created
|
39
|
+
* Create a filepath for a file in the store using the operating systems
|
40
|
+
* default file seperator.
|
41
41
|
*/
|
42
|
+
static char *join_path(char *buf, const char *base, const char *filename)
|
43
|
+
{
|
44
|
+
sprintf(buf, "%s"DIR_SEPARATOR"%s", base, filename);
|
45
|
+
return buf;
|
46
|
+
}
|
47
|
+
|
42
48
|
static void fs_touch(Store *store, char *filename)
|
43
49
|
{
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
50
|
+
int f;
|
51
|
+
char path[MAX_FILE_PATH];
|
52
|
+
join_path(path, store->dir.path, filename);
|
53
|
+
if ((f = creat(path, S_IRUSR | S_IWUSR)) == 0) {
|
54
|
+
RAISE(IO_ERROR, "couldn't create file %s: <%s>", path,
|
55
|
+
strerror(errno));
|
56
|
+
}
|
57
|
+
close(f);
|
51
58
|
}
|
52
59
|
|
53
|
-
|
54
|
-
* Return true if a file of name +filename+ exists in +store+.
|
55
|
-
*
|
56
|
-
* @param store the store to check in
|
57
|
-
* @param filename the name of the file to check for
|
58
|
-
* @returns true if the file exists
|
59
|
-
* @throws IO_ERROR if there is an error checking for the files existance
|
60
|
-
*/
|
61
|
-
int fs_exists(Store *store, char *filename)
|
60
|
+
static int fs_exists(Store *store, char *filename)
|
62
61
|
{
|
63
|
-
|
64
|
-
|
62
|
+
int fd;
|
63
|
+
char path[MAX_FILE_PATH];
|
64
|
+
join_path(path, store->dir.path, filename);
|
65
|
+
fd = open(path, 0);
|
66
|
+
if (fd < 0) {
|
67
|
+
if (errno != ENOENT) {
|
68
|
+
RAISE(IO_ERROR, "checking existance of %s: <%s>", path,
|
69
|
+
strerror(errno));
|
70
|
+
}
|
71
|
+
return false;
|
72
|
+
}
|
73
|
+
close(fd);
|
74
|
+
return true;
|
65
75
|
}
|
66
76
|
|
67
|
-
|
68
|
-
* Remove the file +filename+ from the +store+
|
69
|
-
*
|
70
|
-
* @param store the store to remove from
|
71
|
-
* @param filename the name of the file to remove
|
72
|
-
* @returns On success, zero is returned. On error, -1 is returned, and errno
|
73
|
-
* is set appropriately.
|
74
|
-
*/
|
75
|
-
int fs_remove(Store *store, char *filename)
|
77
|
+
static int fs_remove(Store *store, char *filename)
|
76
78
|
{
|
77
|
-
|
78
|
-
|
79
|
+
char path[MAX_FILE_PATH];
|
80
|
+
return remove(join_path(path, store->dir.path, filename));
|
79
81
|
}
|
80
82
|
|
81
|
-
|
82
|
-
* Rename the file in the +store+ from the name +from+ to the name +to+.
|
83
|
-
*
|
84
|
-
* @param store the store to rename the file in
|
85
|
-
* @param from the name of the file to rename
|
86
|
-
* @param to the new name of the file
|
87
|
-
* @returns On success, zero is returned. On error, -1 is returned, and errno
|
88
|
-
* is set appropriately.
|
89
|
-
*/
|
90
|
-
int fs_rename(Store *store, char *from, char *to)
|
83
|
+
static void fs_rename(Store *store, char *from, char *to)
|
91
84
|
{
|
92
|
-
|
85
|
+
char path1[MAX_FILE_PATH], path2[MAX_FILE_PATH];
|
93
86
|
|
94
|
-
#ifdef
|
95
|
-
|
87
|
+
#ifdef POSH_OS_WIN32
|
88
|
+
remove(join_path(path1, store->dir.path, to));
|
96
89
|
#endif
|
97
90
|
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
91
|
+
if (rename(join_path(path1, store->dir.path, from),
|
92
|
+
join_path(path2, store->dir.path, to)) < 0) {
|
93
|
+
RAISE(IO_ERROR, "couldn't rename file \"%s\" to \"%s\": <%s>",
|
94
|
+
path1, path2, strerror(errno));
|
95
|
+
}
|
103
96
|
}
|
104
97
|
|
105
|
-
|
106
|
-
* Returns the number of files in the store.
|
107
|
-
*
|
108
|
-
* @param store the store to count the files in
|
109
|
-
* @return the number of files in the store
|
110
|
-
* @throws IO_ERROR if there is an error opening the directory
|
111
|
-
*/
|
112
|
-
int fs_count(Store *store)
|
98
|
+
static int fs_count(Store *store)
|
113
99
|
{
|
114
|
-
|
100
|
+
int cnt = 0;
|
101
|
+
struct dirent *de;
|
102
|
+
DIR *d = opendir(store->dir.path);
|
103
|
+
|
104
|
+
if (!d) {
|
105
|
+
RAISE(IO_ERROR, "counting files in %s: <%s>",
|
106
|
+
store->dir.path, strerror(errno));
|
107
|
+
}
|
108
|
+
|
109
|
+
while ((de = readdir(d)) != NULL) {
|
110
|
+
if (de->d_name[0] > '/') { /* skip ., .., / and '\0'*/
|
111
|
+
cnt++;
|
112
|
+
}
|
113
|
+
}
|
114
|
+
closedir(d);
|
115
|
+
|
116
|
+
return cnt;
|
115
117
|
}
|
116
118
|
|
117
|
-
|
118
|
-
|
119
|
-
*
|
120
|
-
*
|
121
|
-
|
122
|
-
|
123
|
-
|
124
|
-
|
125
|
-
|
126
|
-
|
127
|
-
|
128
|
-
|
119
|
+
static void fs_each(Store *store, void (*func)(char *fname, void *arg), void *arg)
|
120
|
+
{
|
121
|
+
struct dirent *de;
|
122
|
+
DIR *d = opendir(store->dir.path);
|
123
|
+
|
124
|
+
if (!d) {
|
125
|
+
RAISE(IO_ERROR, "doing 'each' in %s: <%s>",
|
126
|
+
store->dir.path, strerror(errno));
|
127
|
+
}
|
128
|
+
|
129
|
+
while ((de = readdir(d)) != NULL) {
|
130
|
+
if (de->d_name[0] > '/' /* skip ., .., / and '\0'*/
|
131
|
+
&& !file_is_lock(de->d_name)) {
|
132
|
+
func(de->d_name, arg);
|
133
|
+
}
|
134
|
+
}
|
135
|
+
closedir(d);
|
136
|
+
}
|
137
|
+
|
138
|
+
static void fs_clear_locks(Store *store)
|
129
139
|
{
|
130
|
-
|
140
|
+
struct dirent *de;
|
141
|
+
DIR *d = opendir(store->dir.path);
|
142
|
+
|
143
|
+
if (!d) {
|
144
|
+
RAISE(IO_ERROR, "clearing locks in %s: <%s>",
|
145
|
+
store->dir.path, strerror(errno));
|
146
|
+
}
|
147
|
+
|
148
|
+
while ((de = readdir(d)) != NULL) {
|
149
|
+
if (file_is_lock(de->d_name)) {
|
150
|
+
char path[MAX_FILE_PATH];
|
151
|
+
remove(join_path(path, store->dir.path, de->d_name));
|
152
|
+
}
|
153
|
+
}
|
154
|
+
closedir(d);
|
155
|
+
}
|
156
|
+
|
157
|
+
static void fs_clear(Store *store)
|
158
|
+
{
|
159
|
+
struct dirent *de;
|
160
|
+
DIR *d = opendir(store->dir.path);
|
161
|
+
|
162
|
+
if (!d) {
|
163
|
+
RAISE(IO_ERROR, "clearing files in %s: <%s>",
|
164
|
+
store->dir.path, strerror(errno));
|
165
|
+
}
|
166
|
+
|
167
|
+
while ((de = readdir(d)) != NULL) {
|
168
|
+
if (de->d_name[0] > '/' /* skip ., .., / and '\0'*/
|
169
|
+
&& !file_is_lock(de->d_name)) {
|
170
|
+
char path[MAX_FILE_PATH];
|
171
|
+
remove(join_path(path, store->dir.path, de->d_name));
|
172
|
+
}
|
173
|
+
}
|
174
|
+
closedir(d);
|
175
|
+
}
|
176
|
+
|
177
|
+
static void fs_clear_all(Store *store)
|
178
|
+
{
|
179
|
+
struct dirent *de;
|
180
|
+
DIR *d = opendir(store->dir.path);
|
181
|
+
|
182
|
+
if (!d) {
|
183
|
+
RAISE(IO_ERROR, "clearing all files in %s: <%s>",
|
184
|
+
store->dir.path, strerror(errno));
|
185
|
+
}
|
186
|
+
|
187
|
+
while ((de = readdir(d)) != NULL) {
|
188
|
+
if (de->d_name[0] > '/') { /* skip ., .., / and '\0'*/
|
189
|
+
char path[MAX_FILE_PATH];
|
190
|
+
remove(join_path(path, store->dir.path, de->d_name));
|
191
|
+
}
|
192
|
+
}
|
193
|
+
closedir(d);
|
131
194
|
}
|
132
195
|
|
133
196
|
/**
|
134
197
|
* Destroy the store.
|
135
198
|
*
|
136
199
|
* @param p the store to destroy
|
137
|
-
* @
|
200
|
+
* @raise IO_ERROR if there is an error deleting the locks
|
138
201
|
*/
|
139
|
-
void fs_destroy(Store *store)
|
202
|
+
static void fs_destroy(Store *store)
|
140
203
|
{
|
141
|
-
|
142
|
-
|
143
|
-
|
204
|
+
fs_clear_locks(store);
|
205
|
+
free(store->dir.path);
|
206
|
+
store_destroy(store);
|
144
207
|
}
|
145
208
|
|
146
|
-
|
147
|
-
* Return the length of the file +filename+ in +store+
|
148
|
-
*
|
149
|
-
* @param store the store to use
|
150
|
-
* @param the name of the file to check the length of
|
151
|
-
* @return the length of the file in bytes
|
152
|
-
* @throws IO_ERROR if there is an error checking the file length
|
153
|
-
*/
|
154
|
-
int fs_length(Store *store, char *filename)
|
209
|
+
static off_t fs_length(Store *store, char *filename)
|
155
210
|
{
|
156
|
-
|
157
|
-
|
211
|
+
char path[MAX_FILE_PATH];
|
212
|
+
struct stat stt;
|
158
213
|
|
159
|
-
|
160
|
-
|
214
|
+
if (stat(join_path(path, store->dir.path, filename), &stt)) {
|
215
|
+
RAISE(IO_ERROR, "getting lenth of %s: <%s>", path,
|
216
|
+
strerror(errno));
|
217
|
+
}
|
161
218
|
|
162
|
-
|
219
|
+
return stt.st_size;
|
163
220
|
}
|
164
221
|
|
165
|
-
|
166
|
-
* Flush +len+ characters from +src+ to the output stream +os+
|
167
|
-
*
|
168
|
-
* @param os the output stream to write to
|
169
|
-
* @param src the characters to write to the output stream
|
170
|
-
* @param len the number of characters to write
|
171
|
-
* @throws IO_ERROR if there is an error writing the characters
|
172
|
-
*/
|
173
|
-
void fso_flush_internal(OutStream *os, uchar *src, int len)
|
222
|
+
static void fso_flush_i(OutStream *os, uchar *src, int len)
|
174
223
|
{
|
175
|
-
|
176
|
-
|
224
|
+
if (len != write(os->file.fd, src, len)) {
|
225
|
+
RAISE(IO_ERROR, "flushing src of length %d", len);
|
226
|
+
}
|
177
227
|
}
|
178
228
|
|
179
|
-
|
180
|
-
* Seek +pos+ in the output stream
|
181
|
-
*
|
182
|
-
* @param os the output stream to seek in
|
183
|
-
* @param pos the position to seek in the stream
|
184
|
-
* @throws IO_ERROR if there is an error seeking in the output stream
|
185
|
-
*/
|
186
|
-
void fso_seek_internal(OutStream *os, int pos)
|
229
|
+
static void fso_seek_i(OutStream *os, off_t pos)
|
187
230
|
{
|
188
|
-
|
189
|
-
|
231
|
+
if (lseek(os->file.fd, pos, SEEK_SET) < 0) {
|
232
|
+
RAISE(IO_ERROR, "seeking position %"F_OFF_T_PFX"d: <%s>",
|
233
|
+
pos, strerror(errno));
|
234
|
+
}
|
190
235
|
}
|
191
236
|
|
192
|
-
|
193
|
-
* Close any resources used by the output stream +os+
|
194
|
-
*
|
195
|
-
* @param os the output stream to seek in
|
196
|
-
* @throws IO_ERROR if there is an closing the file
|
197
|
-
*/
|
198
|
-
void fso_close_internal(OutStream *os)
|
237
|
+
static void fso_close_i(OutStream *os)
|
199
238
|
{
|
200
|
-
|
201
|
-
|
239
|
+
if (close(os->file.fd)) {
|
240
|
+
RAISE(IO_ERROR, "closing file: <%s>", strerror(errno));
|
241
|
+
}
|
202
242
|
}
|
203
243
|
|
204
|
-
|
205
|
-
|
206
|
-
|
207
|
-
|
208
|
-
|
209
|
-
|
210
|
-
*
|
211
|
-
* @throws IO_ERROR if there is an error opening the output stream resources
|
212
|
-
*/
|
213
|
-
OutStream *fs_create_output(Store *store, const char *filename)
|
244
|
+
const struct OutStreamMethods FS_OUT_STREAM_METHODS = {
|
245
|
+
fso_flush_i,
|
246
|
+
fso_seek_i,
|
247
|
+
fso_close_i
|
248
|
+
};
|
249
|
+
|
250
|
+
static OutStream *fs_new_output(Store *store, const char *filename)
|
214
251
|
{
|
215
|
-
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
|
224
|
-
|
225
|
-
|
252
|
+
char path[MAX_FILE_PATH];
|
253
|
+
int fd = open(join_path(path, store->dir.path, filename),
|
254
|
+
O_WRONLY | O_CREAT | O_BINARY, S_IRUSR | S_IWUSR);
|
255
|
+
OutStream *os;
|
256
|
+
if (fd < 0) {
|
257
|
+
RAISE(IO_ERROR, "couldn't create OutStream %s: <%s>",
|
258
|
+
path, strerror(errno));
|
259
|
+
}
|
260
|
+
|
261
|
+
os = os_new();
|
262
|
+
os->file.fd = fd;
|
263
|
+
os->m = &FS_OUT_STREAM_METHODS;
|
264
|
+
return os;
|
226
265
|
}
|
227
266
|
|
228
|
-
|
229
|
-
* Read +len+ characters from the input stream into the +offset+ position in
|
230
|
-
* +b+, an array of unsigned characters.
|
231
|
-
*
|
232
|
-
* @param is the input stream to read from
|
233
|
-
* @param b an array of characters which must be allocated with at least
|
234
|
-
* +offset+ + +len+ bytes
|
235
|
-
* @param offset the position in +b+ to start writing the bytes read
|
236
|
-
* @param len the number of bytes to read
|
237
|
-
* @throws IO_ERROR if there is an error reading from the input stream
|
238
|
-
*/
|
239
|
-
void fsi_read_internal(InStream *is, uchar *b, int offset, int len)
|
267
|
+
static void fsi_read_i(InStream *is, uchar *path, int len)
|
240
268
|
{
|
241
|
-
|
242
|
-
|
243
|
-
|
244
|
-
|
245
|
-
|
246
|
-
|
247
|
-
|
248
|
-
|
249
|
-
|
269
|
+
int fd = is->file.fd;
|
270
|
+
int pos = is_pos(is);
|
271
|
+
if (pos != lseek(fd, 0, SEEK_CUR)) {
|
272
|
+
lseek(fd, pos, SEEK_SET);
|
273
|
+
}
|
274
|
+
if (read(fd, path, len) != len) {
|
275
|
+
/* win: the wrong value can be returned for some reason so double check */
|
276
|
+
if (lseek(fd, 0, SEEK_CUR) != (pos + len)) {
|
277
|
+
RAISE(IO_ERROR, "couldn't read %d chars from %s: <%s>",
|
278
|
+
len, path, strerror(errno));
|
279
|
+
}
|
250
280
|
}
|
251
|
-
}
|
252
281
|
}
|
253
282
|
|
254
|
-
|
255
|
-
* Seek position +pos+ in input stream +is+
|
256
|
-
*
|
257
|
-
* @param is the input stream to seek in
|
258
|
-
* @param pos the position to seek
|
259
|
-
* @throws IO_ERROR if the seek fails
|
260
|
-
*/
|
261
|
-
void fsi_seek_internal(InStream *is, int pos)
|
283
|
+
static void fsi_seek_i(InStream *is, off_t pos)
|
262
284
|
{
|
263
|
-
|
264
|
-
|
265
|
-
|
285
|
+
if (lseek(is->file.fd, pos, SEEK_SET) < 0) {
|
286
|
+
RAISE(IO_ERROR, "seeking pos %"F_OFF_T_PFX"d: <%s>",
|
287
|
+
pos, strerror(errno));
|
288
|
+
}
|
266
289
|
}
|
267
290
|
|
268
|
-
|
269
|
-
* Close the resources allocated to the inputstream +is+
|
270
|
-
*
|
271
|
-
* @param is the input stream to close
|
272
|
-
* @throws IO_ERROR if the close fails
|
273
|
-
*/
|
274
|
-
void fsi_close_internal(InStream *is)
|
291
|
+
static void fsi_close_i(InStream *is)
|
275
292
|
{
|
276
|
-
|
277
|
-
|
278
|
-
RAISE(IO_ERROR, strerror(errno));
|
293
|
+
if (close(is->file.fd)) {
|
294
|
+
RAISE(IO_ERROR, strerror(errno));
|
279
295
|
}
|
280
296
|
free(is->d.path);
|
281
|
-
}
|
282
297
|
}
|
283
298
|
|
284
|
-
|
285
|
-
* Returns the length of the input stream +is+
|
286
|
-
*
|
287
|
-
* @param is the input stream to get the length of
|
288
|
-
* @throws IO_ERROR if there is an error getting the file length
|
289
|
-
*/
|
290
|
-
int fsi_length(InStream *is)
|
299
|
+
static off_t fsi_length_i(InStream *is)
|
291
300
|
{
|
292
|
-
|
293
|
-
|
294
|
-
|
295
|
-
|
296
|
-
|
301
|
+
struct stat stt;
|
302
|
+
if (fstat(is->file.fd, &stt)) {
|
303
|
+
RAISE(IO_ERROR, "fstat failed: <%s>", strerror(errno));
|
304
|
+
}
|
305
|
+
return stt.st_size;
|
297
306
|
}
|
298
307
|
|
299
|
-
|
300
|
-
|
301
|
-
|
302
|
-
|
303
|
-
|
308
|
+
static const struct InStreamMethods FS_IN_STREAM_METHODS = {
|
309
|
+
fsi_read_i,
|
310
|
+
fsi_seek_i,
|
311
|
+
fsi_length_i,
|
312
|
+
fsi_close_i
|
313
|
+
};
|
304
314
|
|
305
|
-
|
306
|
-
* Open an input stream in the +store+ with the name +filename+
|
307
|
-
*
|
308
|
-
* @param store the store to create the input stream in
|
309
|
-
* @param filename the name of the input stream
|
310
|
-
* @throws IO_ERROR if the input stream cannot be opened
|
311
|
-
*/
|
312
|
-
InStream *fs_open_input(Store *store, const char *filename)
|
315
|
+
static InStream *fs_open_input(Store *store, const char *filename)
|
313
316
|
{
|
314
|
-
|
315
|
-
|
316
|
-
|
317
|
-
|
318
|
-
|
319
|
-
|
320
|
-
|
321
|
-
|
322
|
-
|
323
|
-
|
324
|
-
|
325
|
-
|
326
|
-
is->seek_internal = &fsi_seek_internal;
|
327
|
-
is->close_internal = &fsi_close_internal;
|
328
|
-
is->clone_internal = &fsi_clone_internal;
|
329
|
-
is->length_internal= &fsi_length;
|
330
|
-
return is;
|
317
|
+
InStream *is;
|
318
|
+
char path[MAX_FILE_PATH];
|
319
|
+
int fd = open(join_path(path, store->dir.path, filename), O_RDONLY | O_BINARY);
|
320
|
+
if (fd < 0) {
|
321
|
+
RAISE(IO_ERROR, "couldn't create InStream %s: <%s>",
|
322
|
+
path, strerror(errno));
|
323
|
+
}
|
324
|
+
is = is_new();
|
325
|
+
is->file.fd = fd;
|
326
|
+
is->d.path = estrdup(path);
|
327
|
+
is->m = &FS_IN_STREAM_METHODS;
|
328
|
+
return is;
|
331
329
|
}
|
332
330
|
|
333
331
|
#define LOCK_OBTAIN_TIMEOUT 5
|
334
332
|
|
335
|
-
|
336
|
-
* Obtain a lock on the lock +lock+
|
337
|
-
*
|
338
|
-
* @param lock the lock to obtain
|
339
|
-
*/
|
340
|
-
int fs_lock_obtain(Lock *lock)
|
333
|
+
static int fs_lock_obtain(Lock *lock)
|
341
334
|
{
|
342
|
-
|
343
|
-
|
344
|
-
|
345
|
-
|
346
|
-
|
347
|
-
|
348
|
-
|
349
|
-
|
350
|
-
|
351
|
-
|
352
|
-
|
353
|
-
|
335
|
+
int f;
|
336
|
+
int trys = LOCK_OBTAIN_TIMEOUT;
|
337
|
+
while (((f =
|
338
|
+
open(lock->name, O_CREAT | O_EXCL | O_RDWR,
|
339
|
+
S_IRUSR | S_IWUSR)) < 0) && (trys > 0)) {
|
340
|
+
trys--;
|
341
|
+
/* FIXME:: implement sleep properly to sleep in milliseconds
|
342
|
+
* sleep(1); */
|
343
|
+
}
|
344
|
+
if (f >= 0) {
|
345
|
+
close(f);
|
346
|
+
return true;
|
347
|
+
}
|
348
|
+
else {
|
349
|
+
return false;
|
350
|
+
}
|
354
351
|
}
|
355
352
|
|
356
|
-
|
357
|
-
* Returns true if +lock+ is locked. To test if the file is locked:wq
|
358
|
-
*
|
359
|
-
* @param lock the lock to test
|
360
|
-
* @throws
|
361
|
-
*/
|
362
|
-
int fs_lock_is_locked(Lock *lock)
|
353
|
+
static int fs_lock_is_locked(Lock *lock)
|
363
354
|
{
|
364
|
-
|
365
|
-
|
366
|
-
|
367
|
-
|
355
|
+
int f = open(lock->name, O_CREAT | O_EXCL | O_WRONLY, S_IRUSR | S_IWUSR);
|
356
|
+
if (f >= 0) {
|
357
|
+
if (close(f) || remove(lock->name)) {
|
358
|
+
RAISE(IO_ERROR, "couldn't close lock \"%s\": <%s>", lock->name,
|
359
|
+
strerror(errno));
|
360
|
+
}
|
361
|
+
return false;
|
362
|
+
}
|
363
|
+
else {
|
364
|
+
return true;
|
368
365
|
}
|
369
|
-
return false;
|
370
|
-
} else {
|
371
|
-
return true;
|
372
|
-
}
|
373
366
|
}
|
374
367
|
|
375
368
|
void fs_lock_release(Lock *lock)
|
376
369
|
{
|
377
|
-
|
370
|
+
remove(lock->name);
|
378
371
|
}
|
379
372
|
|
380
|
-
Lock *fs_open_lock(Store *store, char *lockname)
|
373
|
+
static Lock *fs_open_lock(Store *store, char *lockname)
|
381
374
|
{
|
382
|
-
|
383
|
-
|
384
|
-
|
385
|
-
|
386
|
-
|
387
|
-
|
388
|
-
|
389
|
-
|
390
|
-
|
375
|
+
Lock *lock = ALLOC(Lock);
|
376
|
+
char lname[100];
|
377
|
+
char path[MAX_FILE_PATH];
|
378
|
+
sprintf(lname, "%s%s.lck", LOCK_PREFIX, lockname);
|
379
|
+
lock->name = estrdup(join_path(path, store->dir.path, lname));
|
380
|
+
lock->store = store;
|
381
|
+
lock->obtain = &fs_lock_obtain;
|
382
|
+
lock->release = &fs_lock_release;
|
383
|
+
lock->is_locked = &fs_lock_is_locked;
|
384
|
+
return lock;
|
391
385
|
}
|
392
386
|
|
393
|
-
void fs_close_lock(Lock *lock)
|
387
|
+
static void fs_close_lock(Lock *lock)
|
394
388
|
{
|
395
|
-
|
396
|
-
|
397
|
-
|
389
|
+
remove(lock->name);
|
390
|
+
free(lock->name);
|
391
|
+
free(lock);
|
398
392
|
}
|
399
393
|
|
400
|
-
|
401
|
-
|
402
|
-
|
403
|
-
|
404
|
-
|
405
|
-
|
406
|
-
|
407
|
-
|
408
|
-
|
409
|
-
|
410
|
-
|
411
|
-
/* free_value */(free_ft)&fs_destroy
|
412
|
-
};
|
413
|
-
#else
|
414
|
-
static HshTable stores = {
|
415
|
-
fill:0,
|
416
|
-
used:0,
|
417
|
-
mask:Hsh_MINSIZE - 1,
|
418
|
-
table:stores.smalltable,
|
419
|
-
lookup:&h_lookup_str,
|
420
|
-
hash:NULL,
|
421
|
-
eq:NULL,
|
422
|
-
free_key:(free_ft)&dummy_free,
|
423
|
-
free_value:(free_ft)&fs_destroy
|
394
|
+
static HashTable stores = {
|
395
|
+
/* fill */ 0,
|
396
|
+
/* used */ 0,
|
397
|
+
/* mask */ HASH_MINSIZE - 1,
|
398
|
+
/* table */ stores.smalltable,
|
399
|
+
/* smalltable */ {{0, NULL, NULL}},
|
400
|
+
/* lookup */ (h_lookup_ft)&h_lookup_str,
|
401
|
+
/* hash */ NULL,
|
402
|
+
/* eq */ NULL,
|
403
|
+
/* free_key */ (free_ft)&dummy_free,
|
404
|
+
/* free_value */ (free_ft)&fs_destroy
|
424
405
|
};
|
425
|
-
#endif
|
426
406
|
|
427
|
-
#ifndef
|
407
|
+
#ifndef UNTHREADED
|
428
408
|
static mutex_t stores_mutex = MUTEX_INITIALIZER;
|
429
409
|
#endif
|
430
410
|
|
431
|
-
void fs_close_i(Store *store)
|
411
|
+
static void fs_close_i(Store *store)
|
432
412
|
{
|
433
|
-
|
434
|
-
|
435
|
-
|
413
|
+
mutex_lock(&stores_mutex);
|
414
|
+
h_del(&stores, store->dir.path);
|
415
|
+
mutex_unlock(&stores_mutex);
|
436
416
|
}
|
437
417
|
|
438
|
-
Store *
|
418
|
+
static Store *fs_store_new(const char *pathname)
|
439
419
|
{
|
440
|
-
|
441
|
-
|
442
|
-
|
443
|
-
|
444
|
-
|
445
|
-
|
446
|
-
|
447
|
-
|
448
|
-
|
449
|
-
|
450
|
-
|
451
|
-
|
452
|
-
|
453
|
-
|
454
|
-
|
455
|
-
|
456
|
-
|
457
|
-
|
458
|
-
|
420
|
+
Store *new_store = store_new();
|
421
|
+
|
422
|
+
new_store->dir.path = estrdup(pathname);
|
423
|
+
new_store->touch = &fs_touch;
|
424
|
+
new_store->exists = &fs_exists;
|
425
|
+
new_store->remove = &fs_remove;
|
426
|
+
new_store->rename = &fs_rename;
|
427
|
+
new_store->count = &fs_count;
|
428
|
+
new_store->close_i = &fs_close_i;
|
429
|
+
new_store->clear = &fs_clear;
|
430
|
+
new_store->clear_all = &fs_clear_all;
|
431
|
+
new_store->clear_locks = &fs_clear_locks;
|
432
|
+
new_store->length = &fs_length;
|
433
|
+
new_store->each = &fs_each;
|
434
|
+
new_store->new_output = &fs_new_output;
|
435
|
+
new_store->open_input = &fs_open_input;
|
436
|
+
new_store->open_lock = &fs_open_lock;
|
437
|
+
new_store->close_lock = &fs_close_lock;
|
438
|
+
return new_store;
|
459
439
|
}
|
460
440
|
|
461
441
|
Store *open_fs_store(const char *pathname)
|
462
442
|
{
|
463
|
-
|
464
|
-
|
465
|
-
|
466
|
-
|
467
|
-
|
468
|
-
|
469
|
-
|
470
|
-
|
471
|
-
|
472
|
-
|
473
|
-
|
474
|
-
|
475
|
-
|
476
|
-
|
477
|
-
|
443
|
+
Store *store = NULL;
|
444
|
+
|
445
|
+
mutex_lock(&stores_mutex);
|
446
|
+
store = h_get(&stores, pathname);
|
447
|
+
if (store) {
|
448
|
+
mutex_lock(&store->mutex);
|
449
|
+
store->ref_cnt++;
|
450
|
+
mutex_unlock(&store->mutex);
|
451
|
+
}
|
452
|
+
else {
|
453
|
+
store = fs_store_new(pathname);
|
454
|
+
h_set(&stores, store->dir.path, store);
|
455
|
+
}
|
456
|
+
mutex_unlock(&stores_mutex);
|
457
|
+
|
458
|
+
return store;
|
478
459
|
}
|