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

org.apache.statemachine.StateMachine Maven / Gradle / Ivy

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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 org.apache.statemachine;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.List;
import java.util.Queue;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.Future;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

public class StateMachine {

    private static final Logger LOG = LoggerFactory.getLogger(StateMachine.class);

    private static final String HANDLER_METHOD_NAME = "handleEvent";

    private static ConcurrentHashMap, ConcurrentHashMap, Method>> stateCaches;

    static {
        stateCaches = new ConcurrentHashMap<>();
    }

    public static abstract class State {

        protected final Fsm fsm;
        private final ConcurrentHashMap, Method> handlerCache;

        public State(Fsm fsm) {
            this.fsm = fsm;

            ConcurrentHashMap, Method> handlerCache = stateCaches.get(getClass());
            if (handlerCache == null) {
                handlerCache = new ConcurrentHashMap<>();
                ConcurrentHashMap, Method> old = stateCaches.putIfAbsent(getClass(), handlerCache);
                if (old != null) {
                    handlerCache = old;
                }
            }
            this.handlerCache = handlerCache;
        }

        private Method findHandlerInternal(Class state, Class e) throws NoSuchMethodException {
            Method[] methods = state.getMethods();
            List candidates = new ArrayList<>();
            for (Method m : methods) {
                if (m.getName().equals(HANDLER_METHOD_NAME)
                        && State.class.isAssignableFrom(m.getReturnType())
                        && m.getGenericParameterTypes().length == 1) {
                    candidates.add(m);
                }
            }

            Method best = null;
            for (Method m : candidates) {
                if (m.getParameterTypes()[0].isAssignableFrom(e)) {
                    if (best == null) {
                        best = m;
                    } else if (best.getParameterTypes()[0]
                            .isAssignableFrom(m.getParameterTypes()[0])) {
                        best = m;
                    }
                }
            }
            if (best != null) {
                best.setAccessible(true);
                return best;
            }
            throw new NoSuchMethodException("Handler doesn't exist");
        }

        private Method findHandler(Class event) throws NoSuchMethodException {
            Method m = handlerCache.get(event);
            if (m == null) {
                m = findHandlerInternal(getClass(), event);
                Method m2 = handlerCache.putIfAbsent(event, m);
                if (m2 != null) {
                    m = m2;
                }
            }
            return m;
        }

        State dispatch(Event e) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException {
            return (State) findHandler(e.getClass()).invoke(this, e);
        }

    }

    public interface Event {
    }

    public interface Fsm {
        Fsm newChildFsm();

        void setInitState(State initState);

        void sendEvent(Event e);

        Future sendEvent(Event e, long delay, TimeUnit unit);

        void deferEvent(DeferrableEvent e);
    }

    public static class FsmImpl implements Fsm {
        ScheduledExecutorService executor;
        private State state;
        private Queue deferred;

        public FsmImpl(ScheduledExecutorService executor) {
            this.executor = executor;
            state = null;
            deferred = new ArrayDeque<>();
        }

        private void errorDeferredEvents(Throwable t) {
            Queue oldDeferred = deferred;
            deferred = new ArrayDeque<>();

            for (DeferrableEvent e : oldDeferred) {
                e.error(new IllegalStateException(t));
            }
        }

        @Override
        public Fsm newChildFsm() {
            return new FsmImpl(executor);
        }

        @Override
        public void setInitState(State initState) {
            assert (state == null);
            state = initState;
        }

        public State getState() {
            return state;
        }

        void setState(final State curState, final State newState) {
            if (curState != state) {
                LOG.error("FSM-{}: Tried to transition from {} to {}, but current state is {}",
                          getFsmId(), state, newState, curState);
                throw new IllegalArgumentException();
            }
            state = newState;

            if (LOG.isDebugEnabled()) {
                LOG.debug("FSM-{}: State transition {} -> {}", getFsmId(), curState, newState);
            }
        }

        boolean processEvent(Event e) {
            if (LOG.isDebugEnabled()) {
                LOG.debug("FSM-{}: Received event {}@{} in state {}@{}",
                          getFsmId(), e.getClass().getSimpleName(),
                          System.identityHashCode(e),
                          state.getClass().getSimpleName(),
                          System.identityHashCode(state));
            }
            try {
                State newState = state.dispatch(e);

                if (newState != state) {
                    setState(state, newState);
                    return true;
                }
            } catch (Throwable t) {
                LOG.error("Caught throwable while handling event", t);
                errorDeferredEvents(t);
            }
            return false;
        }

        class FSMRunnable implements Runnable {
            final Event e;

            FSMRunnable(Event e) {
                this.e = e;
            }

            @Override
            public void run() {
                boolean stateChanged = processEvent(e);
                while (stateChanged) {
                    stateChanged = false;
                    Queue prevDeferred = deferred;
                    deferred = new ArrayDeque<>();
                    for (DeferrableEvent d : prevDeferred) {
                        if (stateChanged) {
                            deferred.add(d);
                        } else if (processEvent(d)) {
                            stateChanged = true;
                        }
                    }
                }
            }
        }

        @Override
        public void sendEvent(final Event e) {
            executor.submit(new FSMRunnable(e));
        }

        @Override
        public Future sendEvent(final Event e, final long delay, final TimeUnit unit) {
            return executor.schedule(new FSMRunnable(e), delay, unit);
        }

        @Override
        public void deferEvent(DeferrableEvent e) {
            if (LOG.isDebugEnabled()) {
                LOG.debug("FSM-{}: deferred {}@{}",
                          getFsmId(), e.getClass().getSimpleName(), System.identityHashCode(e));
            }
            deferred.add(e);
        }

        int getFsmId() {
            return System.identityHashCode(this);
        }

        @Override
        protected void finalize() throws Throwable {
            super.finalize();
            LOG.debug("FSM-{}: Finalizing", getFsmId());
        }
    }

    public interface DeferrableEvent extends Event {
        void error(Throwable exception);
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy