All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.elasticsearch.index.rankeval.RankEvalSpec Maven / Gradle / Ivy

There is a newer version: 6.2.3.31
Show newest version
/*
 * Licensed to Elasticsearch under one or more contributor
 * license agreements. See the NOTICE file distributed with
 * this work for additional information regarding copyright
 * ownership. Elasticsearch licenses this file to you under
 * the Apache License, Version 2.0 (the "License"); you may
 * not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */

package org.elasticsearch.index.rankeval;

import org.elasticsearch.common.ParseField;
import org.elasticsearch.common.ParsingException;
import org.elasticsearch.common.Strings;
import org.elasticsearch.common.io.stream.StreamInput;
import org.elasticsearch.common.io.stream.StreamOutput;
import org.elasticsearch.common.io.stream.Writeable;
import org.elasticsearch.common.xcontent.ConstructingObjectParser;
import org.elasticsearch.common.xcontent.ToXContentObject;
import org.elasticsearch.common.xcontent.XContentBuilder;
import org.elasticsearch.common.xcontent.XContentParser;
import org.elasticsearch.common.xcontent.XContentParserUtils;
import org.elasticsearch.script.Script;

import java.io.IOException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Objects;

/**
 * Specification of the ranking evaluation request.
* This class groups the queries to evaluate, including their document ratings, * and the evaluation metric including its parameters. */ public class RankEvalSpec implements Writeable, ToXContentObject { /** List of search request to use for the evaluation */ private final List ratedRequests; /** Definition of the quality metric, e.g. precision at N */ private final EvaluationMetric metric; /** Maximum number of requests to execute in parallel. */ private int maxConcurrentSearches = MAX_CONCURRENT_SEARCHES; /** Default max number of requests. */ private static final int MAX_CONCURRENT_SEARCHES = 10; /** optional: Templates to base test requests on */ private Map templates = new HashMap<>(); /** the indices this ranking evaluation targets */ private final List indices; public RankEvalSpec(List ratedRequests, EvaluationMetric metric, Collection templates) { this.metric = Objects.requireNonNull(metric, "Cannot evaluate ranking if no evaluation metric is provided."); if (ratedRequests == null || ratedRequests.isEmpty()) { throw new IllegalArgumentException( "Cannot evaluate ranking if no search requests with rated results are provided. Seen: " + ratedRequests); } this.ratedRequests = ratedRequests; if (templates == null || templates.isEmpty()) { for (RatedRequest request : ratedRequests) { if (request.getTestRequest() == null) { throw new IllegalStateException("Cannot evaluate ranking if neither template nor test request is " + "provided. Seen for request id: " + request.getId()); } } } if (templates != null) { for (ScriptWithId idScript : templates) { this.templates.put(idScript.id, idScript.script); } } this.indices = new ArrayList<>(); } public RankEvalSpec(List ratedRequests, EvaluationMetric metric) { this(ratedRequests, metric, null); } public RankEvalSpec(StreamInput in) throws IOException { int specSize = in.readVInt(); ratedRequests = new ArrayList<>(specSize); for (int i = 0; i < specSize; i++) { ratedRequests.add(new RatedRequest(in)); } metric = in.readNamedWriteable(EvaluationMetric.class); int size = in.readVInt(); for (int i = 0; i < size; i++) { String key = in.readString(); Script value = new Script(in); this.templates.put(key, value); } maxConcurrentSearches = in.readVInt(); int indicesSize = in.readInt(); indices = new ArrayList<>(indicesSize); for (int i = 0; i < indicesSize; i++) { this.indices.add(in.readString()); } } @Override public void writeTo(StreamOutput out) throws IOException { out.writeVInt(ratedRequests.size()); for (RatedRequest spec : ratedRequests) { spec.writeTo(out); } out.writeNamedWriteable(metric); out.writeVInt(templates.size()); for (Entry entry : templates.entrySet()) { out.writeString(entry.getKey()); entry.getValue().writeTo(out); } out.writeVInt(maxConcurrentSearches); out.writeInt(indices.size()); for (String index : indices) { out.writeString(index); } } /** Returns the metric to use for quality evaluation.*/ public EvaluationMetric getMetric() { return metric; } /** Returns a list of intent to query translation specifications to evaluate. */ public List getRatedRequests() { return Collections.unmodifiableList(ratedRequests); } /** Returns the template to base test requests on. */ public Map getTemplates() { return this.templates; } /** Returns the max concurrent searches allowed. */ public int getMaxConcurrentSearches() { return this.maxConcurrentSearches; } /** Set the max concurrent searches allowed. */ public void setMaxConcurrentSearches(int maxConcurrentSearches) { this.maxConcurrentSearches = maxConcurrentSearches; } public void addIndices(List indices) { this.indices.addAll(indices); } public List getIndices() { return Collections.unmodifiableList(indices); } private static final ParseField TEMPLATES_FIELD = new ParseField("templates"); private static final ParseField METRIC_FIELD = new ParseField("metric"); private static final ParseField REQUESTS_FIELD = new ParseField("requests"); private static final ParseField MAX_CONCURRENT_SEARCHES_FIELD = new ParseField("max_concurrent_searches"); @SuppressWarnings("unchecked") private static final ConstructingObjectParser PARSER = new ConstructingObjectParser<>("rank_eval", a -> new RankEvalSpec((List) a[0], (EvaluationMetric) a[1], (Collection) a[2])); static { PARSER.declareObjectArray(ConstructingObjectParser.constructorArg(), (p, c) -> RatedRequest.fromXContent(p), REQUESTS_FIELD); PARSER.declareObject(ConstructingObjectParser.constructorArg(), (p, c) -> parseMetric(p), METRIC_FIELD); PARSER.declareObjectArray(ConstructingObjectParser.optionalConstructorArg(), (p, c) -> ScriptWithId.fromXContent(p), TEMPLATES_FIELD); PARSER.declareInt(RankEvalSpec::setMaxConcurrentSearches, MAX_CONCURRENT_SEARCHES_FIELD); } private static EvaluationMetric parseMetric(XContentParser parser) throws IOException { XContentParserUtils.ensureExpectedToken(XContentParser.Token.START_OBJECT, parser.currentToken(), parser::getTokenLocation); XContentParserUtils.ensureExpectedToken(XContentParser.Token.FIELD_NAME, parser.nextToken(), parser::getTokenLocation); EvaluationMetric metric = parser.namedObject(EvaluationMetric.class, parser.currentName(), null); XContentParserUtils.ensureExpectedToken(XContentParser.Token.END_OBJECT, parser.nextToken(), parser::getTokenLocation); return metric; } public static RankEvalSpec parse(XContentParser parser) { return PARSER.apply(parser, null); } static class ScriptWithId { private Script script; private String id; private static final ParseField TEMPLATE_FIELD = new ParseField("template"); private static final ParseField TEMPLATE_ID_FIELD = new ParseField("id"); ScriptWithId(String id, Script script) { this.id = id; this.script = script; } private static final ConstructingObjectParser PARSER = new ConstructingObjectParser<>("script_with_id", a -> new ScriptWithId((String) a[0], (Script) a[1])); public static ScriptWithId fromXContent(XContentParser parser) { return PARSER.apply(parser, null); } static { PARSER.declareString(ConstructingObjectParser.constructorArg(), TEMPLATE_ID_FIELD); PARSER.declareObject(ConstructingObjectParser.constructorArg(), (p, c) -> { try { return Script.parse(p, "mustache"); } catch (IOException ex) { throw new ParsingException(p.getTokenLocation(), "error parsing rank request", ex); } }, TEMPLATE_FIELD); } } @Override public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException { builder.startObject(); builder.startArray(TEMPLATES_FIELD.getPreferredName()); for (Entry entry : templates.entrySet()) { builder.startObject(); builder.field(ScriptWithId.TEMPLATE_ID_FIELD.getPreferredName(), entry.getKey()); builder.field(ScriptWithId.TEMPLATE_FIELD.getPreferredName(), entry.getValue()); builder.endObject(); } builder.endArray(); builder.startArray(REQUESTS_FIELD.getPreferredName()); for (RatedRequest spec : this.ratedRequests) { spec.toXContent(builder, params); } builder.endArray(); builder.field(METRIC_FIELD.getPreferredName(), this.metric); builder.field(MAX_CONCURRENT_SEARCHES_FIELD.getPreferredName(), maxConcurrentSearches); builder.endObject(); return builder; } @Override public String toString() { return Strings.toString(this); } @Override public final boolean equals(Object obj) { if (this == obj) { return true; } if (obj == null || getClass() != obj.getClass()) { return false; } RankEvalSpec other = (RankEvalSpec) obj; return Objects.equals(ratedRequests, other.ratedRequests) && Objects.equals(metric, other.metric) && Objects.equals(maxConcurrentSearches, other.maxConcurrentSearches) && Objects.equals(templates, other.templates) && Objects.equals(indices, other.indices); } @Override public final int hashCode() { return Objects.hash(ratedRequests, metric, templates, maxConcurrentSearches, indices); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy