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

org.springframework.statemachine.zookeeper.ZookeeperStateMachineEnsemble Maven / Gradle / Ivy

There is a newer version: 4.0.0
Show newest version
/*
 * Copyright 2015-2019 the original author or authors.
 *
 * 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
 *
 * https://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.springframework.statemachine.zookeeper;

import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.Queue;
import java.util.UUID;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicReference;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.curator.framework.CuratorFramework;
import org.apache.curator.framework.api.CuratorWatcher;
import org.apache.curator.framework.api.transaction.CuratorTransaction;
import org.apache.curator.framework.api.transaction.CuratorTransactionFinal;
import org.apache.curator.framework.imps.CuratorFrameworkState;
import org.apache.curator.framework.recipes.locks.InterProcessSemaphoreMutex;
import org.apache.curator.framework.recipes.nodes.PersistentNode;
import org.apache.curator.framework.state.ConnectionState;
import org.apache.curator.framework.state.ConnectionStateListener;
import org.apache.zookeeper.CreateMode;
import org.apache.zookeeper.WatchedEvent;
import org.apache.zookeeper.data.Stat;
import org.springframework.statemachine.StateMachine;
import org.springframework.statemachine.StateMachineContext;
import org.springframework.statemachine.StateMachineException;
import org.springframework.statemachine.StateMachinePersist;
import org.springframework.statemachine.ensemble.StateMachineEnsemble;
import org.springframework.statemachine.ensemble.StateMachineEnsembleException;
import org.springframework.statemachine.ensemble.StateMachineEnsembleObjectSupport;

import reactor.core.publisher.Mono;

/**
 * {@link StateMachineEnsemble} backed by a zookeeper.
 *
 * @author Janne Valkealahti
 *
 * @param  the type of state
 * @param  the type of event
 */
public class ZookeeperStateMachineEnsemble extends StateMachineEnsembleObjectSupport {

	private final static Log log = LogFactory.getLog(ZookeeperStateMachineEnsemble.class);
	private final String uuid = UUID.randomUUID().toString();
	private final static int DEFAULT_LOGSIZE = 32;
	private final static String PATH_CURRENT = "current";
	private final static String PATH_LOG = "log";
	private final static String PATH_MEMBERS = "members";
	private final static String PATH_MUTEX = "mutex";
	private final CuratorFramework curatorClient;
	private final String baseDataPath;
	private final String statePath;
	private final String logPath;
	private final int logSize;
	private final String memberPath;
	private final String mutexPath;
	private final boolean cleanState;
	private final StateMachinePersist persist;
	private final AtomicReference stateRef = new AtomicReference();
	private final AtomicReference notifyRef = new AtomicReference();
	private final CuratorWatcher watcher = new StateWatcher();
	private PersistentNode node;
	private final Queue> joinQueue = new ConcurrentLinkedQueue>();
	private final List> joined = new ArrayList>();
	private final Object joinLock = new Object();
	private final ConnectionStateListener connectionListener = new LocalConnectionStateListener();

	/**
	 * Instantiates a new zookeeper state machine ensemble.
	 *
	 * @param curatorClient the curator client
	 * @param basePath the base zookeeper path
	 */
	public ZookeeperStateMachineEnsemble(CuratorFramework curatorClient, String basePath) {
		this(curatorClient, basePath, true, DEFAULT_LOGSIZE);
	}

	/**
	 * Instantiates a new zookeeper state machine ensemble.
	 *
	 * @param curatorClient the curator client
	 * @param basePath the base zookeeper path
	 * @param cleanState if true clean existing state
	 * @param logSize the log size
	 */
	public ZookeeperStateMachineEnsemble(CuratorFramework curatorClient, String basePath, boolean cleanState, int logSize) {
		this.curatorClient = curatorClient;
		this.cleanState = cleanState;
		this.logSize = logSize;
		this.baseDataPath = basePath + "/data";
		this.statePath = baseDataPath + "/" + PATH_CURRENT;
		this.logPath = baseDataPath + "/" + PATH_LOG;
		this.memberPath = basePath + "/" + PATH_MEMBERS;
		this.mutexPath = basePath + "/" + PATH_MUTEX;
		this.persist = new ZookeeperStateMachinePersist(curatorClient, statePath, logPath, logSize);
		setAutoStartup(true);
	}

	@Override
	protected void onInit() throws Exception {
		initPaths();
	}

	@Override
	protected Mono doPreStartReactively() {
		return Mono.fromRunnable(() -> doStart());
	}

	protected void doStart() {
		// initially setting a watcher here, further watchers
		// will be set when events are received.
		registerWatcherForStatePath();

		StateWrapper stateWrapper = stateRef.get();
		if (stateWrapper == null) {
			try {
				StateWrapper currentStateWrapper = readCurrentContext();
				stateRef.set(currentStateWrapper);
				notifyRef.set(currentStateWrapper);
			} catch (Exception e) {
				log.error("Error reading current state during start", e);
			}
		}
		curatorClient.getConnectionStateListenable().addListener(connectionListener);
		if (curatorClient.getState() == CuratorFrameworkState.STARTED) {
			handleZkConnect();
		} else {
			curatorClient.start();
		}
	}

	@Override
	protected Mono doPreStopReactively() {
		return Mono.fromRunnable(() -> doStop());
	}

	protected void doStop() {
		if (node != null && curatorClient.getState() != CuratorFrameworkState.STOPPED) {
			try {
				node.close();
			} catch (IOException e) {
			} finally {
				node = null;
			}
		}
		curatorClient.getConnectionStateListenable().removeListener(connectionListener);
	}

	@Override
	public void join(StateMachine stateMachine) {
		if (!isRunning()) {
			joinQueue.add(stateMachine);
		} else {
			StateWrapper stateWrapper = stateRef.get();
			synchronized (joinLock) {
				joined.add(stateMachine);
			}
			notifyJoined(stateMachine, stateWrapper != null ? stateWrapper.context : null);
		}
	}

	@Override
	public StateMachine getLeader() {
		return null;
	}

	private void joinQueued() {
		StateMachine stateMachine = null;
		synchronized (joinLock) {
			while ((stateMachine = joinQueue.poll()) != null) {
				joined.add(stateMachine);
			}
		}
	}

	private void notifyJoined() {
		StateWrapper stateWrapper = stateRef.get();
		synchronized (joinLock) {
			for (StateMachine stateMachine : joined) {
				notifyJoined(stateMachine, stateWrapper != null ? stateWrapper.context : null);
			}
		}
	}

	private void notifyLeft() {
		StateWrapper stateWrapper = stateRef.get();
		synchronized (joinLock) {
			for (StateMachine stateMachine : joined) {
				notifyLeft(stateMachine, stateWrapper != null ? stateWrapper.context : null);
			}
		}
	}

	@Override
	public void leave(StateMachine stateMachine) {
		// TODO: think when to close
		if (node != null) {
			try {
				node.close();
			} catch (IOException e) {
			}
		}
		boolean removed = false;
		synchronized (joinLock) {
			removed = joined.remove(stateMachine);
		}
		if (removed) {
			StateWrapper stateWrapper = stateRef.get();
			notifyLeft(stateMachine, stateWrapper != null ? stateWrapper.context : null);
		}
	}

	@Override
	public synchronized void setState(StateMachineContext context) {
		if (log.isDebugEnabled()) {
			log.debug("Setting state context=" + context);
		}
		try {
			Stat stat = new Stat();
			StateWrapper stateWrapper = stateRef.get();
			if (stateWrapper != null) {
				stat.setVersion(stateWrapper.version);
			}
			if (log.isDebugEnabled()) {
				log.debug("Requesting persist write " + context + " with version " + stat.getVersion() + " for ensemble " + uuid);
			}
			persist.write(context, stat);
			if (log.isDebugEnabled()) {
				log.debug("Request persist write ok " + context + " new version " + stat.getVersion() + " for ensemble " + uuid);
			}
			stateRef.set(new StateWrapper(context, stat.getVersion()));
		} catch (Exception e) {
			throw new StateMachineException("Error persisting data", e);
		}
	}

	@Override
	public StateMachineContext getState() {
		return readCurrentContext().context;
	}

	private void handleZkConnect() {
		log.info("Handling Zookeeper connect");
		joinQueued();
		notifyJoined();
		registerWatcherForStatePath();
	}

	private void handleZkDisconnect() {
		log.info("Handling Zookeeper disconnect");
		notifyError(new StateMachineEnsembleException("Lost connection to zookeeper"));
		notifyLeft();
	}

	private StateWrapper readCurrentContext() {
		try {
			Stat stat = new Stat();
			registerWatcherForStatePath();
			StateMachineContext context = persist.read(stat);
			return new StateWrapper(context, stat.getVersion());
		} catch (Exception e) {
			throw new StateMachineException("Error reading data", e);
		}
	}

	/**
	 * Create all needed paths including what ZookeeperStateMachinePersist
	 * is going to need because it doesn't handle any path creation. We also
	 * use a mutex lock to make a decision if cleanState is enabled to wipe
	 * out existing data.
	 */
	private void initPaths() {
		InterProcessSemaphoreMutex mutex = new InterProcessSemaphoreMutex(curatorClient, mutexPath);
		try {
			if (log.isTraceEnabled()) {
				log.trace("About to acquire mutex");
			}
			mutex.acquire();
			if (log.isTraceEnabled()) {
				log.trace("Mutex acquired");
			}

			if (cleanState) {
				if (curatorClient.checkExists().forPath(memberPath) != null) {
					if (curatorClient.getChildren().forPath(memberPath).size() == 0) {
						log.info("Deleting from " + baseDataPath);
						curatorClient.delete().deletingChildrenIfNeeded().forPath(baseDataPath);
					}
				}
			}

			node = new PersistentNode(curatorClient, CreateMode.EPHEMERAL, true, memberPath + "/" + uuid, new byte[0]);
			node.start();
			node.waitForInitialCreate(60, TimeUnit.SECONDS);

			if (curatorClient.checkExists().forPath(baseDataPath) == null) {
				CuratorTransaction tx = curatorClient.inTransaction();
				CuratorTransactionFinal tt = tx.create().forPath(baseDataPath).and();
				tt = tt.create().forPath(statePath).and();
				tt = tt.create().forPath(logPath).and();
				for (int i = 0; i notifyWrapper.version) {
			if (log.isDebugEnabled()) {
				log.debug("Wrapper version higher that notifyWrapper version, notifyWrapper=[" + notifyWrapper
						+ "], wrapper=[" + wrapper + "] for " + this);
			}
			notifyRef.set(wrapper);
			notifyStateChanged(wrapper.context);
		}
	}

	private void traceLogWrappers(StateWrapper currentWrapper, StateWrapper notifyWrapper, StateWrapper newWrapper) {
		if (log.isTraceEnabled()) {
			log.trace("Wrappers id=" + uuid + "\ncurrentWrapper=[" + currentWrapper + "] \nnotifyWrapper=["
					+ notifyWrapper + "] \nnewWrapper=[" + newWrapper + "]");
		}
	}

	@Override
	public String toString() {
		return "ZookeeperStateMachineEnsemble [uuid=" + uuid + "]";
	}

	private class StateWatcher implements CuratorWatcher {
		// zk is not really reliable for watching events because
		// you need to re-register watcher when it fires. most likely
		// we will miss events so need to do little tricks here via
		// event logs.

		// NOTE: because paths are pre-created, version always start
		//       from 1 when real data is set. initial path contains
		//       empty data with version 0.

		@Override
		public void process(WatchedEvent event) throws Exception {
			if (log.isTraceEnabled()) {
				log.trace("Process WatchedEvent: id=" + uuid + " " + event);
			}
			switch (event.getType()) {
			case NodeDataChanged:
				try {
					// re-read once if we did read log history
					// there might be unread change
					if (handleDataChange()) {
						handleDataChange();
					}
				} catch (Exception e) {
					log.error("Error handling event", e);
				}
				registerWatcherForStatePath();
				break;
			default:
				registerWatcherForStatePath();
				break;
			}
		}

	}

	/**
	 * Handles internal logic of reading and comparing current
	 * wrapper references and re-plays logs if needed.
	 *
	 * @return true if log is replayed
	 * @throws Exception if error occurred
	 */
	private boolean handleDataChange() throws Exception {
		StateWrapper currentWrapper = stateRef.get();
		StateWrapper notifyWrapper = notifyRef.get();
		StateWrapper newWrapper = readCurrentContext();
		traceLogWrappers(currentWrapper, notifyWrapper, newWrapper);

		if (currentWrapper.version + 1 == newWrapper.version
				&& notifyWrapper.version >= currentWrapper.version
				&& stateRef.compareAndSet(currentWrapper, newWrapper)) {
			// simply used to check if we don't need to replay, if so
			// we can just try to notify
			mayNotifyStateChanged(newWrapper);
		} else {
			final int start = (notifyWrapper != null ? (notifyWrapper.version) : 0) % logSize;
			int count = newWrapper.version - (notifyWrapper != null ? (notifyWrapper.version) : 0);
			if (log.isDebugEnabled()) {
				log.debug("Events missed, trying to replay start " + start + " count " + count);
			}
			for (int i = start; i < (start + count); i++) {
				Stat stat = new Stat();
				StateMachineContext context = ((ZookeeperStateMachinePersist) persist).readLog(i, stat);
				int ver = (stat.getVersion() - 1) * logSize + (i + 1);

				// check if we're behind more than a log size meaning we can't
				// replay full history, notify and break out from a loop
				if (i + logSize < ver) {
					notifyError(new StateMachineEnsembleException("Current version behind more than log size"));
					break;
				}
				if (log.isDebugEnabled()) {
					log.debug("Replay position " + i + " with version " + ver);
					log.debug("Context in position " + i + " " + context);
				}

				StateWrapper wrapper = new StateWrapper(context, ver);

				// need to set stateRef when replaying if its
				// context is not set or otherwise just set
				// if stateRef still is currentWrapper
				StateWrapper currentWrapperx = stateRef.get();
				if (currentWrapperx.context == null) {
					stateRef.set(wrapper);
				} else if (wrapper.version == currentWrapperx.version + 1) {
					stateRef.set(wrapper);
				}
				mayNotifyStateChanged(wrapper);
			}
			// did we replay
			return count > 0;
		}
		return false;
	}

	private class LocalConnectionStateListener implements ConnectionStateListener {

		@Override
		public void stateChanged(CuratorFramework client, ConnectionState newState) {
			if (curatorClient == client) {
				switch (newState) {
				case CONNECTED:
				case RECONNECTED:
					handleZkConnect();
					break;
				case READ_ONLY:
					break;
				case LOST:
				case SUSPENDED:
					handleZkDisconnect();
					break;
				default:
					break;
				}
			}
		}

	};

	/**
	 * Wrapper object for a {@link StateMachineContext} and its
	 * current version.
	 */
	private class StateWrapper {
		private final StateMachineContext context;
		private final int version;

		public StateWrapper(StateMachineContext context, int version) {
			this.context = context;
			this.version = version;
		}

		@Override
		public String toString() {
			return "StateWrapper [context=" + context + ", version=" + version + "]";
		}
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy