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

discord4j.common.sinks.TimeoutEmissionStrategy Maven / Gradle / Ivy

There is a newer version: 3.3.0-RC1
Show newest version
/*
 * This file is part of Discord4J.
 *
 * Discord4J is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * Discord4J is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with Discord4J. If not, see .
 */

package discord4j.common.sinks;

import reactor.core.publisher.Sinks;
import reactor.util.Logger;
import reactor.util.Loggers;

import java.util.concurrent.locks.LockSupport;
import java.util.function.Supplier;

class TimeoutEmissionStrategy implements EmissionStrategy {

    private static final Logger log = Loggers.getLogger(TimeoutEmissionStrategy.class);

    private final long parkNanos;
    private final long timeoutNanos;
    private final boolean errorOnTimeout;

    TimeoutEmissionStrategy(long parkNanos, long timeoutNanos, boolean errorOnTimeout) {
        this.parkNanos = parkNanos;
        this.timeoutNanos = timeoutNanos;
        this.errorOnTimeout = errorOnTimeout;
    }

    @Override
    public  boolean emitNext(Sinks.Many sink, T element) {
        long remaining = 0;
        if (timeoutNanos > 0) {
            remaining = timeoutNanos;
        }
        for (;;) {
            Sinks.EmitResult emission = sink.tryEmitNext(element);
            if (emission.isSuccess()) {
                return true;
            }
            remaining -= parkNanos;
            if (timeoutNanos >= 0 && remaining <= 0) {
                log.debug("Emission timed out at {}: {}", sink.name(), element.toString());
                if (errorOnTimeout) {
                    throw new Sinks.EmissionException(emission, "Emission timed out");
                }
                return false;
            }
            switch (emission) {
                case FAIL_ZERO_SUBSCRIBER:
                case FAIL_CANCELLED:
                case FAIL_TERMINATED:
                    return false;
                case FAIL_NON_SERIALIZED:
                    LockSupport.parkNanos(parkNanos);
                    continue;
                case FAIL_OVERFLOW:
                    log.trace("Emission overflowing at {}: {}", sink.name(), element.toString());
                    LockSupport.parkNanos(parkNanos);
                    continue;
                default:
                    throw new Sinks.EmissionException(emission, "Unknown emitResult value");
            }
        }
    }

    @Override
    public  boolean emitComplete(Sinks.Many sink) {
        return emitTerminal(sink::tryEmitComplete);
    }

    @Override
    public  boolean emitError(Sinks.Many sink, Throwable error) {
        return emitTerminal(() -> sink.tryEmitError(error));
    }

    private  boolean emitTerminal(Supplier resultSupplier) {
        long remaining = 0;
        if (timeoutNanos > 0) {
            remaining = timeoutNanos;
        }
        for (;;) {
            Sinks.EmitResult emission = resultSupplier.get();
            if (emission.isSuccess()) {
                return true;
            }
            remaining -= parkNanos;
            if (timeoutNanos >= 0 && remaining <= 0) {
                if (errorOnTimeout) {
                    throw new Sinks.EmissionException(emission, "Emission timed out");
                }
                return false;
            }
            switch (emission) {
                case FAIL_ZERO_SUBSCRIBER:
                case FAIL_CANCELLED:
                case FAIL_TERMINATED:
                case FAIL_OVERFLOW:
                    return false;
                case FAIL_NON_SERIALIZED:
                    LockSupport.parkNanos(parkNanos);
                    continue;
                default:
                    throw new Sinks.EmissionException(emission, "Unknown emitResult value");
            }
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy