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

io.undertow.websockets.jsr.OrderedExecutor Maven / Gradle / Ivy

/*
 * JBoss, Home of Professional Open Source.
 * Copyright 2014 Red Hat, Inc., and individual contributors
 * as indicated by the @author tags.
 *
 * 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 io.undertow.websockets.jsr;

import java.util.Deque;
import java.util.concurrent.ConcurrentLinkedDeque;
import java.util.concurrent.Executor;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;

/**
 * Executor that executes tasks in the order they are submitted, using at most one thread at a time.
 *
 * @author Stuart Douglas
 */
public class OrderedExecutor implements Executor {

    private final Deque tasks = new ConcurrentLinkedDeque<>();
    private final Executor delegate;
    private final ExecutorTask task = new ExecutorTask();
    private volatile int state = 0;

    private static final AtomicIntegerFieldUpdater stateUpdater = AtomicIntegerFieldUpdater.newUpdater(OrderedExecutor.class, "state");

    private static final int STATE_NOT_RUNNING = 0;
    private static final int STATE_RUNNING = 1;

    public OrderedExecutor(Executor delegate) {
        this.delegate = delegate;
    }


    @Override
    public void execute(Runnable command) {
        tasks.add(command);
        if (stateUpdater.get(this) == STATE_NOT_RUNNING) {
            delegate.execute(task);
        }
    }

    private final class ExecutorTask implements Runnable {

        @Override
        public void run() {
            do {
                //if there is no thread active then we run
                if (stateUpdater.compareAndSet(OrderedExecutor.this, STATE_NOT_RUNNING, STATE_RUNNING)) {
                    Runnable task = tasks.poll();
                    //while the queue is not empty we process in order
                    while (task != null) {
                        try {
                            task.run();
                        } catch (Throwable e) {
                            JsrWebSocketLogger.REQUEST_LOGGER.exceptionInWebSocketMethod(e);
                        }
                        task = tasks.poll();
                    }
                    //set state back to not running.
                    stateUpdater.set(OrderedExecutor.this, STATE_NOT_RUNNING);
                } else {
                    return;
                }
                //we loop again based on tasks not being empty. Otherwise there is a window where the state is running,
                //but poll() has returned null, so a submitting thread will believe that it does not need re-execute.
                //this check fixes the issue
            } while (!tasks.isEmpty());
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy