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

org.opensearch.search.aggregations.support.BaseMultiValuesSourceFieldConfig Maven / Gradle / Ivy

There is a newer version: 2.18.0
Show newest version
/*
 * SPDX-License-Identifier: Apache-2.0
 *
 * The OpenSearch Contributors require contributions made to
 * this file be licensed under the Apache-2.0 license or a
 * compatible open source license.
 */

package org.opensearch.search.aggregations.support;

import org.opensearch.LegacyESVersion;
import org.opensearch.common.TriConsumer;
import org.opensearch.common.joda.Joda;
import org.opensearch.common.time.DateUtils;
import org.opensearch.core.ParseField;
import org.opensearch.core.common.Strings;
import org.opensearch.core.common.io.stream.StreamInput;
import org.opensearch.core.common.io.stream.StreamOutput;
import org.opensearch.core.common.io.stream.Writeable;
import org.opensearch.core.xcontent.MediaTypeRegistry;
import org.opensearch.core.xcontent.ObjectParser;
import org.opensearch.core.xcontent.ToXContentObject;
import org.opensearch.core.xcontent.XContentBuilder;
import org.opensearch.core.xcontent.XContentParser;
import org.opensearch.script.Script;

import java.io.IOException;
import java.time.ZoneId;
import java.time.ZoneOffset;
import java.util.Objects;

/**
 * A configuration that tells aggregation how to retrieve data from index
 * in order to run a specific aggregation.
 *
 * @opensearch.internal
 */
public abstract class BaseMultiValuesSourceFieldConfig implements Writeable, ToXContentObject {
    private final String fieldName;
    private final Object missing;
    private final Script script;
    private final ZoneId timeZone;

    static TriConsumer<
        ObjectParser>, Void>,
        Boolean,
        Boolean> PARSER = (parser, scriptable, timezoneAware) -> {
            parser.declareString(Builder::setFieldName, ParseField.CommonFields.FIELD);
            parser.declareField(
                Builder::setMissing,
                XContentParser::objectText,
                ParseField.CommonFields.MISSING,
                ObjectParser.ValueType.VALUE
            );

            if (scriptable) {
                parser.declareField(
                    Builder::setScript,
                    (p, context) -> Script.parse(p),
                    Script.SCRIPT_PARSE_FIELD,
                    ObjectParser.ValueType.OBJECT_OR_STRING
                );
            }

            if (timezoneAware) {
                parser.declareField(Builder::setTimeZone, p -> {
                    if (p.currentToken() == XContentParser.Token.VALUE_STRING) {
                        return ZoneId.of(p.text());
                    } else {
                        return ZoneOffset.ofHours(p.intValue());
                    }
                }, ParseField.CommonFields.TIME_ZONE, ObjectParser.ValueType.LONG);
            }
        };

    public BaseMultiValuesSourceFieldConfig(String fieldName, Object missing, Script script, ZoneId timeZone) {
        this.fieldName = fieldName;
        this.missing = missing;
        this.script = script;
        this.timeZone = timeZone;
    }

    public BaseMultiValuesSourceFieldConfig(StreamInput in) throws IOException {
        if (in.getVersion().onOrAfter(LegacyESVersion.V_7_6_0)) {
            this.fieldName = in.readOptionalString();
        } else {
            this.fieldName = in.readString();
        }
        this.missing = in.readGenericValue();
        this.script = in.readOptionalWriteable(Script::new);
        if (in.getVersion().before(LegacyESVersion.V_7_0_0)) {
            this.timeZone = DateUtils.dateTimeZoneToZoneId(Joda.readOptionalTimeZone(in));
        } else {
            this.timeZone = in.readOptionalZoneId();
        }
    }

    @Override
    public void writeTo(StreamOutput out) throws IOException {
        if (out.getVersion().onOrAfter(LegacyESVersion.V_7_6_0)) {
            out.writeOptionalString(fieldName);
        } else {
            out.writeString(fieldName);
        }
        out.writeGenericValue(missing);
        out.writeOptionalWriteable(script);
        if (out.getVersion().before(LegacyESVersion.V_7_0_0)) {
            Joda.writeOptionalTimeZone(out, DateUtils.zoneIdToDateTimeZone(timeZone));
        } else {
            out.writeOptionalZoneId(timeZone);
        }
        doWriteTo(out);
    }

    @Override
    public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException {
        builder.startObject();
        if (missing != null) {
            builder.field(ParseField.CommonFields.MISSING.getPreferredName(), missing);
        }
        if (script != null) {
            builder.field(Script.SCRIPT_PARSE_FIELD.getPreferredName(), script);
        }
        if (fieldName != null) {
            builder.field(ParseField.CommonFields.FIELD.getPreferredName(), fieldName);
        }
        if (timeZone != null) {
            builder.field(ParseField.CommonFields.TIME_ZONE.getPreferredName(), timeZone.getId());
        }
        doXContentBody(builder, params);
        builder.endObject();
        return builder;
    }

    public Object getMissing() {
        return missing;
    }

    public Script getScript() {
        return script;
    }

    public ZoneId getTimeZone() {
        return timeZone;
    }

    public String getFieldName() {
        return fieldName;
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        BaseMultiValuesSourceFieldConfig that = (BaseMultiValuesSourceFieldConfig) o;
        return Objects.equals(fieldName, that.fieldName)
            && Objects.equals(missing, that.missing)
            && Objects.equals(script, that.script)
            && Objects.equals(timeZone, that.timeZone);
    }

    @Override
    public int hashCode() {
        return Objects.hash(fieldName, missing, script, timeZone);
    }

    @Override
    public String toString() {
        return Strings.toString(MediaTypeRegistry.JSON, this);
    }

    abstract void doXContentBody(XContentBuilder builder, Params params) throws IOException;

    abstract void doWriteTo(StreamOutput out) throws IOException;

    /**
     * Base builder for the multi values source field configuration
     *
     * @opensearch.internal
     */
    public abstract static class Builder> {
        String fieldName;
        Object missing = null;
        Script script = null;
        ZoneId timeZone = null;

        public String getFieldName() {
            return fieldName;
        }

        public B setFieldName(String fieldName) {
            this.fieldName = fieldName;
            return (B) this;
        }

        public Object getMissing() {
            return missing;
        }

        public B setMissing(Object missing) {
            this.missing = missing;
            return (B) this;
        }

        public Script getScript() {
            return script;
        }

        public B setScript(Script script) {
            this.script = script;
            return (B) this;
        }

        public ZoneId getTimeZone() {
            return timeZone;
        }

        public B setTimeZone(ZoneId timeZone) {
            this.timeZone = timeZone;
            return (B) this;
        }

        abstract public C build();
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy