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

net.openhft.chronicle.wire.domestic.extractor.DocumentExtractor Maven / Gradle / Ivy

There is a newer version: 2.27ea1
Show newest version
/*
 * Copyright 2016-2022 chronicle.software
 *
 *       https://chronicle.software
 *
 * Licensed 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 net.openhft.chronicle.wire.domestic.extractor;

import net.openhft.chronicle.core.annotation.NonNegative;
import net.openhft.chronicle.core.io.InvalidMarshallableException;
import net.openhft.chronicle.wire.Wire;
import net.openhft.chronicle.wire.internal.extractor.DocumentExtractorBuilder;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

import java.util.function.*;

import static net.openhft.chronicle.core.util.ObjectUtils.requireNonNull;

/**
 * The DocumentExtractor functional interface defines methods to extract data from documents, with
 * the ability to transform the extracted data using provided mappers. Implementers of this
 * interface should provide the logic for extracting data from documents based on the provided
 * wire and index.
 *
 * @param  The type of data to be extracted from the document.
 */
@FunctionalInterface
public interface DocumentExtractor {

    /**
     * Extracts a value of type T from the provided {@code wire} and {@code index} or else {@code null}
     * if no value can be extracted.
     * 

* {@code null} may be returned if the queue was written with a method writer and there are messages in the * queue but of another type. * * @param wire to use * @param index to use * @return extracted value or {@code null} */ @Nullable T extract(@NotNull Wire wire, @NonNegative long index) throws InvalidMarshallableException; /** * Creates and returns a new DocumentExtractor consisting of the results (of type R) of applying the provided * {@code mapper } to the elements of this DocumentExtractor. *

* Values mapped to {@code null} are removed. * * @param mapper to apply * @param type to map to * @return a new mapped DocumentExtractor * @throws NullPointerException if the provided {@code mapper} is {@code null} */ default DocumentExtractor map(@NotNull final Function mapper) { requireNonNull(mapper); return (wire, index) -> { final T value = extract(wire, index); if (value == null) { return null; } return mapper.apply(value); }; } /** * Creates and returns a new ToLongDocumentExtractor consisting of applying the provided * {@code mapper } to the elements of this DocumentExtractor. *

* Values mapped to {@link Long#MIN_VALUE } are removed. * * @param mapper to apply * @return a new mapped DocumentExtractor * @throws NullPointerException if the provided {@code mapper} is {@code null} */ default ToLongDocumentExtractor mapToLong(@NotNull final ToLongFunction mapper) { requireNonNull(mapper); return (wire, index) -> { final T value = extract(wire, index); if (value == null) { return Long.MIN_VALUE; } return mapper.applyAsLong(value); }; } /** * Creates and returns a new DocumentExtractor consisting of the elements of this DocumentExtractor that match * the provided {@code predicate}. * * @param predicate to apply to each element to determine if it * should be included * @return a DocumentExtractor consisting of the elements of this DocumentExtractor that match * @throws NullPointerException if the provided {@code predicate} is {@code null} */ default DocumentExtractor filter(@NotNull final Predicate predicate) { requireNonNull(predicate); return (wire, index) -> { final T value = extract(wire, index); if (value == null) { // The value is already filtered so just propagate the lack of a value return null; } return predicate.test(value) ? value : null; }; } // skip // peek /** * Generic builder that can be used to build DocumentExtractor objects of common types. * * @param element type to extract */ interface Builder extends net.openhft.chronicle.core.util.Builder> { /** * Specifies a {@code supplier} of element that shall be reused when extracting elements from excerpts. *

* By default, thread local reuse objects are created on demand but this can be changed by means of the * {@link #withThreadConfinedReuse()} method. *

* The provided supplier must provide distinct objects on each invocation. This can be accomplished * , for example, by referencing {@code Foo:new}. * * @param supplier to call when reuse object are needed (non-null) * @return this Builder */ @NotNull Builder withReusing(@NotNull Supplier supplier); /** * Specifies that only one reuse object, confined to the first using thread, shall be reused. *

* The DocumentExtractor is guaranteed to prevent accidental concurrent thread access by throwing * an {@link IllegalStateException} if accessed by a foreign thread. * * @return this Builder */ @NotNull Builder withThreadConfinedReuse(); /** * Specifies an {@code interfaceType } that was previously used to write messages of type E using * a method writer via invocations of the provided {@code methodReference}. *

* The provided {@code methodReference} must be a true method reference (e.g. {@code Greeting:message}) * or a corresponding lambda expression * (e.g. {@code (Greeting greeting, String msg) -> greeting.message(m))} ) or else the * result is undefined. * * @param interfaceType interface that has at least one method that takes a single * argument parameter of type E * @param methodReference connecting the interface type to a method that takes a single * argument parameter of type E * @param interface type * @return this Builder */ @NotNull Builder withMethod(@NotNull final Class interfaceType, @NotNull final BiConsumer methodReference); } /** * Creates and returns a new Builder that can be used to create DocumentExtractor objects * of the provided {@code elementType}. * * @param elementType type of element to extract * @param element type * @return a new Builder */ static Builder builder(@NotNull final Class elementType) { requireNonNull(elementType); return new DocumentExtractorBuilder<>(elementType); } }