3 # This file is part of Koha.
5 # Koha is free software; you can redistribute it and/or modify it
6 # under the terms of the GNU General Public License as published by
7 # the Free Software Foundation; either version 3 of the License, or
8 # (at your option) any later version.
10 # Koha is distributed in the hope that it will be useful, but
11 # WITHOUT ANY WARRANTY; without even the implied warranty of
12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 # GNU General Public License for more details.
15 # You should have received a copy of the GNU General Public License
16 # along with Koha; if not, see <http://www.gnu.org/licenses>.
20 use Scalar::Util qw( looks_like_number );
21 use List::Util qw( first );
22 use C4::Output qw( output_html_with_http_headers );
23 use C4::Auth qw( get_template_and_user );
26 use Koha::SearchEngine::Elasticsearch;
27 use Koha::SearchEngine::Elasticsearch::QueryBuilder;
28 use Koha::SearchMarcMaps;
29 use Koha::SearchFields;
32 use Try::Tiny qw( catch try );
33 use Module::Load::Conditional qw( can_load );
37 my ( $template, $borrowernumber, $cookie ) = get_template_and_user(
39 template_name => 'admin/searchengine/elasticsearch/mappings.tt',
42 flagsrequired => { parameters => 'manage_search_engine_config' },
46 unless ( can_load( modules => { 'Koha::SearchEngine::Elasticsearch::Indexer' => undef } ) ) {
47 output_and_exit( $input, $cookie, $template, 'missing_es_modules');
51 my $index = $input->param('index') || 'biblios';
52 my $op = $input->param('op') || 'list';
53 my ( @messages, @errors );
54 push @messages, { type => 'message', code => 'elasticsearch_disabled' }
55 if ( C4::Context->preference('SearchEngine') ne 'Elasticsearch' );
57 my $database = Koha::Database->new();
58 my $schema = $database->schema;
60 my $marc_type = lc C4::Context->preference('marcflavour');
62 my @index_names = ($Koha::SearchEngine::Elasticsearch::BIBLIOS_INDEX, $Koha::SearchEngine::Elasticsearch::AUTHORITIES_INDEX);
64 my $update_mappings = sub {
65 for my $index_name (@index_names) {
66 my $indexer = Koha::SearchEngine::Elasticsearch::Indexer->new({ index => $index_name });
68 $indexer->update_mappings();
70 my $conf = $indexer->get_elasticsearch_params();
73 code => 'error_on_update_es_mappings',
75 index => $conf->{index_name},
81 my $search_fields_aliases = {};
82 while ( my ( $key, $value ) = each(%{Koha::SearchEngine::Elasticsearch::QueryBuilder->get_index_field_convert}) ) {
83 my $field_aliases = $search_fields_aliases->{$value};
84 $field_aliases = [] unless $field_aliases;
85 push @$field_aliases, $key;
86 $search_fields_aliases->{$value} = $field_aliases;
89 if ( $op eq 'edit' ) {
91 $schema->storage->txn_begin;
93 my @field_name = $input->multi_param('search_field_name');
94 my @field_label = $input->multi_param('search_field_label');
95 my @field_type = $input->multi_param('search_field_type');
96 my @field_weight = $input->multi_param('search_field_weight');
97 my @field_staff_client = $input->multi_param('search_field_staff_client');
98 my @field_opac = $input->multi_param('search_field_opac');
100 my @index_name = $input->multi_param('mapping_index_name');
101 my @search_field_name = $input->multi_param('mapping_search_field_name');
102 my @mapping_sort = $input->multi_param('mapping_sort');
103 my @mapping_facet = $input->multi_param('mapping_facet');
104 my @mapping_suggestible = $input->multi_param('mapping_suggestible');
105 my @mapping_search = $input->multi_param('mapping_search');
106 my @mapping_marc_field = $input->multi_param('mapping_marc_field');
107 my @faceted_field_names = $input->multi_param('display_facet');
111 for my $i ( 0 .. scalar(@field_name) - 1 ) {
112 my $field_name = $field_name[$i];
113 my $field_label = $field_label[$i];
114 my $field_type = $field_type[$i];
115 my $field_weight = $field_weight[$i];
116 my $field_staff_client = $field_staff_client[$i];
117 my $field_opac = $field_opac[$i];
119 my $search_field = Koha::SearchFields->find( { name => $field_name }, { key => 'name' } );
120 $search_field->label($field_label);
121 $search_field->type($field_type);
123 if (!length($field_weight)) {
124 $search_field->weight(undef);
126 elsif ($field_weight <= 0 || !looks_like_number($field_weight)) {
127 push @errors, { type => 'error', code => 'invalid_field_weight', 'weight' => $field_weight };
130 $search_field->weight($field_weight);
132 $search_field->staff_client($field_staff_client ? 1 : 0);
133 $search_field->opac($field_opac ? 1 : 0);
135 my $facet_order = first { $faceted_field_names[$_] eq $field_name } 0 .. $#faceted_field_names;
136 $search_field->facet_order(defined $facet_order ? $facet_order + 1 : undef);
137 $search_field->store;
140 Koha::SearchMarcMaps->search( { marc_type => $marc_type, } )->delete;
141 my @facetable_fields = Koha::SearchEngine::Elasticsearch->get_facetable_fields();
142 my @facetable_field_names = map { $_->name } @facetable_fields;
144 my $mandatory_before = Koha::SearchFields->search({mandatory=>1})->count;
145 my $mandatory_after = 0;
147 for my $i ( 0 .. scalar(@index_name) - 1 ) {
148 my $index_name = $index_name[$i];
149 my $search_field_name = $search_field_name[$i];
150 my $mapping_marc_field = $mapping_marc_field[$i];
151 my $mapping_facet = $mapping_facet[$i];
152 $mapping_facet = ( grep { $_ eq $search_field_name } @facetable_field_names ) ? $mapping_facet : 0;
153 my $mapping_suggestible = $mapping_suggestible[$i];
154 my $mapping_sort = $mapping_sort[$i];
155 my $mapping_search = $mapping_search[$i];
157 my $search_field = Koha::SearchFields->find({ name => $search_field_name }, { key => 'name' });
158 $mandatory_after++ if $search_field->mandatory && !defined $seen_fields{$search_field_name};
159 $seen_fields{$search_field_name} = 1;
161 # TODO Check mapping format
162 $mapping_marc_field =~ s/\s//g;
164 my $marc_field = Koha::SearchMarcMaps->find_or_create({
165 index_name => $index_name,
166 marc_type => $marc_type,
167 marc_field => $mapping_marc_field
169 $search_field->add_to_search_marc_maps($marc_field, {
170 facet => $mapping_facet,
171 suggestible => $mapping_suggestible,
172 sort => $mapping_sort,
173 search => $mapping_search
176 push @errors, { type => 'error', code => 'missing_mandatory_fields' } if $mandatory_after < $mandatory_before;
179 push @errors, { type => 'error', code => 'error_on_update', message => $@, }; # FIXME $@ can be empty but @errors
180 $schema->storage->txn_rollback;
182 push @messages, { type => 'message', code => 'success_on_update' };
184 C4::Log::logaction( 'SEARCHENGINE', 'EDIT_MAPPINGS', undef, q{} );
186 $schema->storage->txn_commit;
188 Koha::SearchEngine::Elasticsearch->clear_search_fields_cache();
190 $update_mappings->();
193 elsif( $op eq 'reset_confirmed' ) {
194 Koha::SearchEngine::Elasticsearch->reset_elasticsearch_mappings;
195 push @messages, { type => 'message', code => 'success_on_reset' };
196 C4::Log::logaction( 'SEARCHENGINE', 'RESET_MAPPINGS', undef, q{} );
198 elsif( $op eq 'reset_confirm' ) {
199 $template->param( reset_confirm => 1 );
204 for my $index_name (@index_names) {
205 my $indexer = Koha::SearchEngine::Elasticsearch::Indexer->new({ index => $index_name });
206 if (!$indexer->is_index_status_ok) {
207 my $conf = $indexer->get_elasticsearch_params();
208 if ($indexer->is_index_status_reindex_required) {
211 code => 'reindex_required',
212 index => $conf->{index_name},
215 elsif($indexer->is_index_status_recreate_required) {
218 code => 'recreate_required',
219 index => $conf->{index_name},
225 my @facetable_fields = Koha::SearchEngine::Elasticsearch->get_facetable_fields();
226 for my $index_name (@index_names) {
227 my $search_fields = Koha::SearchFields->search(
229 'search_marc_map.index_name' => $index_name,
230 'search_marc_map.marc_type' => $marc_type,
233 join => { search_marc_to_fields => 'search_marc_map' },
235 'search_marc_to_fields.facet',
236 'search_marc_to_fields.suggestible',
237 'search_marc_to_fields.sort',
238 'search_marc_to_fields.search',
239 'search_marc_map.marc_field'
248 order_by => { -asc => [qw/name marc_field/] }
253 my @facetable_field_names = map { $_->name } @facetable_fields;
255 while ( my $s = $search_fields->next ) {
258 search_field_name => $name,
259 search_field_label => $s->label,
260 search_field_type => $s->type,
261 search_field_mandatory => $s->mandatory,
262 marc_field => $s->get_column('marc_field'),
263 sort => $s->get_column('sort') // 'undef', # To avoid warnings "Use of uninitialized value in lc"
264 suggestible => $s->get_column('suggestible'),
265 search => $s->get_column('search'),
266 facet => $s->get_column('facet'),
267 is_facetable => ( grep { $_ eq $name } @facetable_field_names ) ? 1 : 0,
271 push @indexes, { index_name => $index_name, mappings => \@mappings };
274 my $search_fields = Koha::SearchFields->search( {}, { order_by => ['name'] } );
275 my @all_search_fields;
276 while ( my $search_field = $search_fields->next ) {
277 my $search_field_unblessed = $search_field->unblessed;
278 $search_field_unblessed->{mapped_biblios} = 1 if $search_field->is_mapped_biblios;
279 $search_field_unblessed->{aliases} = $search_fields_aliases->{$search_field_unblessed->{name}};
280 push @all_search_fields, $search_field_unblessed;
283 push @messages, @errors;
285 indexes => \@indexes,
286 all_search_fields => \@all_search_fields,
287 facetable_fields => \@facetable_fields,
288 messages => \@messages,
291 output_html_with_http_headers $input, $cookie, $template->output;