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

org.jboss.resteasy.reactive.server.jaxrs.AsyncResponseImpl Maven / Gradle / Ivy

There is a newer version: 3.17.5
Show newest version
package org.jboss.resteasy.reactive.server.jaxrs;

import java.lang.reflect.InvocationTargetException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.TimeUnit;

import jakarta.ws.rs.ServiceUnavailableException;
import jakarta.ws.rs.container.AsyncResponse;
import jakarta.ws.rs.container.CompletionCallback;
import jakarta.ws.rs.container.ConnectionCallback;
import jakarta.ws.rs.container.TimeoutHandler;
import jakarta.ws.rs.core.HttpHeaders;
import jakarta.ws.rs.core.Response;
import jakarta.ws.rs.core.Response.ResponseBuilder;

import org.jboss.resteasy.reactive.server.core.ResteasyReactiveRequestContext;

public class AsyncResponseImpl implements AsyncResponse, Runnable {

    private final ResteasyReactiveRequestContext context;
    private volatile boolean suspended;
    private volatile boolean cancelled;
    private volatile TimeoutHandler timeoutHandler;
    // only used with lock, no need for volatile
    private Runnable timerCancelTask = null;

    public AsyncResponseImpl(ResteasyReactiveRequestContext context) {
        this.context = context;
        suspended = true;
    }

    @Override
    public synchronized boolean resume(Object response) {
        if (!suspended) {
            return false;
        } else {
            suspended = false;
        }
        cancelTimer();
        context.setResult(response);
        context.resume();
        return true;
    }

    @Override
    public synchronized boolean resume(Throwable response) {
        if (!suspended) {
            return false;
        } else {
            suspended = false;
        }
        cancelTimer();
        context.handleException(response);
        context.resume();
        return true;
    }

    @Override
    public boolean cancel() {
        return internalCancel(null);
    }

    @Override
    public boolean cancel(int retryAfter) {
        return internalCancel(retryAfter);
    }

    private synchronized boolean internalCancel(Object retryAfter) {
        if (cancelled) {
            return true;
        }
        if (!suspended) {
            return false;
        }
        cancelTimer();
        suspended = false;
        cancelled = true;
        ResponseBuilder response = Response.status(503);
        if (retryAfter != null)
            response.header(HttpHeaders.RETRY_AFTER, retryAfter);
        // It's not clear if we should go via the exception handlers here, but our TCK setup makes us
        // go through it, while RESTEasy doesn't because it does resume like this, so we do too
        context.setResult(response.build());
        context.resume();
        return true;
    }

    @Override
    public boolean cancel(Date retryAfter) {
        return internalCancel(retryAfter);
    }

    // CALL WITH LOCK
    private void cancelTimer() {
        if (timerCancelTask != null) {
            timerCancelTask.run();
            timerCancelTask = null;
        }
    }

    @Override
    public boolean isSuspended() {
        return suspended;
    }

    @Override
    public boolean isCancelled() {
        return cancelled;
    }

    @Override
    public boolean isDone() {
        // we start suspended and stop being suspended on resume/cancel/timeout(which resumes) so
        // this flag is enough to know if we're done
        return !suspended;
    }

    @Override
    public synchronized boolean setTimeout(long time, TimeUnit unit) {
        if (!suspended)
            return false;
        if (timerCancelTask != null) {
            timerCancelTask.run();
        }
        timerCancelTask = context.registerTimer(TimeUnit.MILLISECONDS.convert(time, unit), this);
        return true;
    }

    @Override
    public void setTimeoutHandler(TimeoutHandler handler) {
        timeoutHandler = handler;
    }

    @Override
    public Collection> register(Class callback) {
        Objects.requireNonNull(callback);
        // FIXME: does this mean we should use CDI to look it up?
        try {
            return register(callback.getDeclaredConstructor().newInstance());
        } catch (InstantiationException | IllegalAccessException | NoSuchMethodException | InvocationTargetException e) {
            throw new RuntimeException(e);
        }
    }

    @Override
    public Map, Collection>> register(Class callback, Class... callbacks) {
        Objects.requireNonNull(callback);
        Objects.requireNonNull(callbacks);
        Map, Collection>> ret = new HashMap<>();
        ret.put(callback, register(callback));
        for (Class cb : callbacks) {
            ret.put(cb, register(cb));
        }
        return ret;
    }

    @Override
    public Collection> register(Object callback) {
        Objects.requireNonNull(callback);
        List> ret = new ArrayList<>(2);
        if (callback instanceof ConnectionCallback) {
            context.registerConnectionCallback((ConnectionCallback) callback);
            ret.add(ConnectionCallback.class);
        }
        if (callback instanceof CompletionCallback) {
            context.registerCompletionCallback((CompletionCallback) callback);
            ret.add(CompletionCallback.class);
        }
        return ret;
    }

    @Override
    public Map, Collection>> register(Object callback, Object... callbacks) {
        Objects.requireNonNull(callback);
        Objects.requireNonNull(callbacks);
        Map, Collection>> ret = new HashMap<>();
        ret.put(callback.getClass(), register(callback));
        for (Object cb : callbacks) {
            ret.put(cb.getClass(), register(cb));
        }
        return ret;
    }

    @Override
    public synchronized void run() {
        // make sure we're not marked as waiting for a timer anymore
        timerCancelTask = null;
        // if we're not suspended anymore, or we were cancelled, drop it
        if (!suspended || cancelled)
            return;
        if (timeoutHandler != null) {
            timeoutHandler.handleTimeout(this);
            // Spec says:
            // In case the time-out handler does not take any of the actions mentioned above [resume/new timeout],
            // a default time-out strategy is executed by the runtime.
            // Stef picked to do this if the handler did not resume or set a new timeout:
            if (suspended && timerCancelTask == null)
                resume(new ServiceUnavailableException());
        } else {
            resume(new ServiceUnavailableException());
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy