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

net.sf.hajdbc.state.distributed.DistributedStateManager Maven / Gradle / Ivy

/*
 * HA-JDBC: High-Availability JDBC
 * Copyright (C) 2012  Paul Ferraro
 *
 * This program 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.
 *
 * This program 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 this program.  If not, see .
 */
package net.sf.hajdbc.state.distributed;

import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.io.Serializable;
import java.util.*;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;

import java.util.concurrent.CopyOnWriteArrayList;
import net.sf.hajdbc.Database;
import net.sf.hajdbc.DatabaseCluster;
import net.sf.hajdbc.Messages;
import net.sf.hajdbc.distributed.Command;
import net.sf.hajdbc.distributed.CommandDispatcher;
import net.sf.hajdbc.distributed.CommandDispatcherFactory;
import net.sf.hajdbc.distributed.Member;
import net.sf.hajdbc.distributed.MembershipListener;
import net.sf.hajdbc.distributed.Remote;
import net.sf.hajdbc.distributed.Stateful;
import net.sf.hajdbc.distributed.jgroups.AddressMember;
import net.sf.hajdbc.durability.InvocationEvent;
import net.sf.hajdbc.durability.InvokerEvent;
import net.sf.hajdbc.logging.Level;
import net.sf.hajdbc.logging.Logger;
import net.sf.hajdbc.logging.LoggerFactory;
import net.sf.hajdbc.state.*;
import net.sf.hajdbc.state.health.ClusterHealth;
import net.sf.hajdbc.state.health.ClusterHealthImpl;
import net.sf.hajdbc.state.sync.SyncMgr;
import net.sf.hajdbc.state.sync.SyncMgrImpl;

/**
 * @author Paul Ferraro
 */
public class DistributedStateManager> implements StateManager, DistributedManager, StateCommandContext, MembershipListener, Stateful
{
	private final Logger logger = LoggerFactory.getLogger(this.getClass());
	private final DatabaseCluster cluster;
	private final StateManager stateManager;


  public CommandDispatcher> getDispatcher() {
    return dispatcher;
  }

  private final CommandDispatcher> dispatcher;
	private final ConcurrentMap>> remoteInvokerMap = new ConcurrentHashMap>>();
	private final Set members = Collections.newSetFromMap(new ConcurrentHashMap());
  private final List membershipListeners = new CopyOnWriteArrayList<>();
  private final Map extContexts = new HashMap<>();
	private final ClusterHealth health;
	private final SyncMgr syncMgr;


	public DistributedStateManager(DatabaseCluster cluster, CommandDispatcherFactory dispatcherFactory) throws Exception
	{
		this.cluster = cluster;
		this.stateManager = cluster.getStateManager();
		StateCommandContext context = this;
		this.dispatcher = dispatcherFactory.createCommandDispatcher(cluster.getId() + ".state", context, this, this);
    this.health = new ClusterHealthImpl(this);
		this.syncMgr = new SyncMgrImpl(this);
	}

  public ClusterHealth getHealth() {
    return health;
  }

	public SyncMgr getSyncMgr() {
		return syncMgr;
	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.state.StateManager#getActiveDatabases()
	 */
	@Override
	public Set getActiveDatabases()
	{
		return this.stateManager.getActiveDatabases();
	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.state.StateManager#setActiveDatabases(java.util.Set)
	 */
	@Override
	public void setActiveDatabases(Set databases)
	{
		this.stateManager.setActiveDatabases(databases);
	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.DatabaseClusterListener#activated(net.sf.hajdbc.state.DatabaseEvent)
	 */
	@Override
	public void activated(DatabaseEvent event)
	{
		this.stateManager.activated(event);
		this.dispatcher.executeAll(new ActivationCommand(event), this.dispatcher.getLocal());
	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.DatabaseClusterListener#deactivated(net.sf.hajdbc.state.DatabaseEvent)
	 */
	@Override
	public void deactivated(DatabaseEvent event)
	{
		this.stateManager.deactivated(event);
		this.dispatcher.executeAll(new DeactivationCommand(event), this.dispatcher.getLocal());

	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.durability.DurabilityListener#afterInvocation(net.sf.hajdbc.durability.InvocationEvent)
	 */
	@Override
	public void afterInvocation(InvocationEvent event)
	{
		this.dispatcher.executeAll(new PostInvocationCommand(this.getRemoteDescriptor(event)));
	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.durability.DurabilityListener#afterInvoker(net.sf.hajdbc.durability.InvokerEvent)
	 */
	@Override
	public void afterInvoker(InvokerEvent event)
	{
		this.dispatcher.executeAll(new InvokerCommand(this.getRemoteDescriptor(event)));
	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.durability.DurabilityListener#beforeInvocation(net.sf.hajdbc.durability.InvocationEvent)
	 */
	@Override
	public void beforeInvocation(InvocationEvent event)
	{
		this.dispatcher.executeAll(new PreInvocationCommand(this.getRemoteDescriptor(event)));
	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.durability.DurabilityListener#beforeInvoker(net.sf.hajdbc.durability.InvokerEvent)
	 */
	@Override
	public void beforeInvoker(InvokerEvent event)
	{
		this.dispatcher.executeAll(new InvokerCommand(this.getRemoteDescriptor(event)));
	}

	private RemoteInvocationDescriptor getRemoteDescriptor(InvocationEvent event)
	{
		return new RemoteInvocationDescriptorImpl(event, this.dispatcher.getLocal());
	}
	
	private RemoteInvokerDescriptor getRemoteDescriptor(InvokerEvent event)
	{
		return new RemoteInvokerDescriptorImpl(event, this.dispatcher.getLocal());
	}
	
	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.Lifecycle#start()
	 */
	@Override
	public void start() throws Exception
	{
		this.stateManager.start();
		this.dispatcher.start();
		this.health.start();

	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.Lifecycle#stop()
	 */
	@Override
	public void stop()
	{
    this.health.stop();
		this.dispatcher.stop();
		this.stateManager.stop();
	}


	@Override
	public boolean isEnabled()
	{
		return this.stateManager.isEnabled() && dispatcher.getLocal().equals(dispatcher.getCoordinator());
	}


  /**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.state.distributed.StateCommandContext#getDatabaseCluster()
	 */
	@Override
	public DatabaseCluster getDatabaseCluster()
	{
		return this.cluster;
	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.state.distributed.StateCommandContext#getLocalStateManager()
	 */
	@Override
	public StateManager getLocalStateManager()
	{
		return this.stateManager;
	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.state.distributed.StateCommandContext#getRemoteInvokers(net.sf.hajdbc.distributed.Remote)
	 */
	@Override
	public Map> getRemoteInvokers(Remote remote)
	{
		return this.remoteInvokerMap.get(remote.getMember());
	}

	@Override
	public  Map executeAll(Command> command,
			Member... excludedMembers) {
		return dispatcher.executeAll(command, excludedMembers);
	}


	@Override
	public  R execute(Command> command, Member member) {
		return dispatcher.execute(command, member);
	}

	@Override
	public  C getExtContext(String key) {
		return (C)extContexts.get(key);
	}

	@Override
	public  C removeExtContext(String key) {
		return (C)extContexts.remove(key);
	}

	@Override
	public  void setExtContext(String key,C context) {
		if(context!=null){
			extContexts.put(key,context);
		}
	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.distributed.Stateful#readState(java.io.ObjectInput)
	 */
	@Override
	public void readState(ObjectInput input) throws IOException
	{
		if (input.available() > 0)
		{
			Set databases = new TreeSet();
			
			int size = input.readInt();
			
			for (int i = 0; i < size; ++i)
			{
				databases.add(input.readUTF());
			}
			
			this.logger.log(Level.INFO, Messages.INITIAL_CLUSTER_STATE_REMOTE.getMessage(databases, this.dispatcher.getCoordinator()));
			
			this.stateManager.setActiveDatabases(databases);


    }
	}

	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.distributed.Stateful#writeState(java.io.ObjectOutput)
	 */
	@Override
	public void writeState(ObjectOutput output) throws IOException
	{
		Set databases = this.cluster.getBalancer();
		output.writeInt(databases.size());
		
		for (D database: databases)
		{
			output.writeUTF(database.getId());
		}
	}



	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.distributed.MembershipListener#added(net.sf.hajdbc.distributed.Member)
	 */
	@Override
	public void added(Member member)
	{
		this.remoteInvokerMap.putIfAbsent(member, new HashMap>());
		members.add(member);

		checkMemberDatabaseConfig(member);

		Iterator iterator = membershipListeners.iterator();
    while(iterator.hasNext()){
      try {
        iterator.next().added(member);
      }catch (Exception e){
        logger.log(Level.WARN,e);
      }
    }
	}

	/*
	public void checkDatabaseConfig() {
		Iterator iterator = members.iterator();
		while(iterator.hasNext()){
			Member member = iterator.next();
			checkMemberDatabaseConfig(member);
		}
	}//*/
	GetDatabaseCommand getDatabaseCommand = new GetDatabaseCommand();
	private void checkMemberDatabaseConfig(Member member) {
		if(!member.equals(getLocal())){
			String ip = getIp(member);
			D database = cluster.getDatabaseByIp(ip);
			if(database==null){
				D db = (D)execute(getDatabaseCommand, member);
				if(db!=null){
					db.setLocal(false);
					cluster.addDatabase(db);
				}
			}
		}
	}


	/**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.distributed.MembershipListener#removed(net.sf.hajdbc.distributed.Member)
	 */
	@Override
	public void removed(Member member)
	{
		if (this.dispatcher.getLocal().equals(this.dispatcher.getCoordinator()))
		{
			Map> invokers = this.remoteInvokerMap.remove(member);
			
			if (invokers != null)
			{
				this.cluster.getDurability().recover(invokers);
			}

		}
		members.remove(member);
    Iterator iterator = membershipListeners.iterator();
    while(iterator.hasNext()){
      try {
        iterator.next().removed(member);
      }catch (Exception e){
        logger.log(Level.WARN,e);
      }

    }
  }


  /**
	 * {@inheritDoc}
	 * @see net.sf.hajdbc.state.StateManager#recover()
	 */
	@Override
	public Map> recover()
	{
		return this.stateManager.recover();
	}

	@Override
	public boolean isValid(Database database) {
		Set ips = getIps();
		if(ips.contains(database.getIp())){
     return true;
		}
		return false;
	}

  public Member getMember(String ip) {
    Member find = null;
    Iterator iterator = members.iterator();
    while(iterator.hasNext()){
      Member next = iterator.next();
      if(getIp(next).equals(ip)){
        find = next;
        break;
      }
    }
    return find;
  }

	private Set getIps() {
		Set ips = new HashSet<>();
		for(Member m:this.members){
      ips.add(getIp(m));

		}
		return ips;
	}


	private String getIp(Member local) {
		return org.jgroups.util.UUID.get(((AddressMember)local).getAddress());
	}

  public Member getCoordinator() {
    return this.dispatcher.getCoordinator();
  }

	@Override
	public List getMembers() {
		return new ArrayList<>(members);
	}

  @Override
  public void addMembershipListener(MembershipListener listener) {
    membershipListeners.add(listener);
  }

  @Override
  public void removeMembershipListener(MembershipListener listener) {
    membershipListeners.remove(listener);
  }

  public Member getLocal() {
    return this.dispatcher.getLocal();
  }

  public String getLocalIp() {
    return getIp(getLocal());
  }


	private static class RemoteDescriptor implements Remote, Serializable
	{
		private static final long serialVersionUID = 3717630867671175936L;
		
		private final Member member;
		
		RemoteDescriptor(Member member)
		{
			this.member = member;
		}

		@Override
		public Member getMember()
		{
			return this.member;
		}
	}
	
	private static class RemoteInvocationDescriptorImpl extends RemoteDescriptor implements RemoteInvocationDescriptor
	{
		private static final long serialVersionUID = 7782082258670023082L;
		
		private final InvocationEvent event;
		
		RemoteInvocationDescriptorImpl(InvocationEvent event, Member member)
		{
			super(member);
			
			this.event = event;
		}
		
		@Override
		public InvocationEvent getEvent()
		{
			return this.event;
		}

		/**
		 * {@inheritDoc}
		 * @see java.lang.Object#toString()
		 */
		@Override
		public String toString()
		{
			return this.event.toString();
		}
	}
	
	private static class RemoteInvokerDescriptorImpl extends RemoteDescriptor implements RemoteInvokerDescriptor
	{
		private static final long serialVersionUID = 6991831573393882786L;
		
		private final InvokerEvent event;
		
		RemoteInvokerDescriptorImpl(InvokerEvent event, Member member)
		{
			super(member);
			
			this.event = event;
		}
		
		@Override
		public InvokerEvent getEvent()
		{
			return this.event;
		}

		/**
		 * {@inheritDoc}
		 * @see java.lang.Object#toString()
		 */
		@Override
		public String toString()
		{
			return this.event.toString();
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy