The Grouper Failover Client API is included in the Grouper client v2.1+

This is an API where you can specify a list of endpoints and some logic and it will run against them until it finds one that returns a successful response before a timeout has occurred.

This can be used for any type of endpoint: web service, database, ldap, etc.

The failover client will remember which connnections have how many errors to prefer not to use connections with more errors than others.

If the failover is used in command-line mode, the state can be saved periodically to disk so each invocation remembers the state of the previous.

This API is used in the Grouper client optionally for the discovery service and always available web services and ldap.

Usage

Include the grouperClient jar, and call the failover client:

String result = FailoverClient.failoverLogic("myEndPointType", new FailoverLogic<String>() {

        /**
         * @see FailoverLogic#logic
         */
        @Override
        public String logic(FailoverLogicBean failoverLogicBean) {
          ... do some logic based on this connection: failoverLogicBean.getConnectionName()
          return theResult which is a String since it was declared this type above: FailoverLogic<String>;
        }
      });

Configuration

There is a configuration bean (which you would likely read from a config file, but could be programmatic), and you register the configuration for this type before using it (e.g. on startup)

FailoverConfig failoverConfig = new FailoverConfig();

//if there are no errors in connections, then use the same connection for 30 minutes
failoverConfig.setAffinitySeconds(2400);

//if there are no errors in connections, it will use the 1st tier connection names before the 2nd tier
//note that this configuration example will only work for readonly queries, if it is a readwrite query,
//then there would not be any readonlyConnections available
failoverConfig.setConnectionNames(GrouperClientUtils.toSet("readWriteConnection1", "readWriteConnection2");
failoverConfig.setConnectionNamesSecondTier(GrouperClientUtils.toSet("readonlyConnection1", "readonlyConnnection2", readonlyConnection3");

//this is a label to identify this "pool" of connections
failoverConfig.setConnectionType("myEndPointType");

//if there are no errors in connections, and if there is no affinity, this is the strategy to pick which connection
//active/active will pick one at random from the 1st tier connections, active/standby will use the connections in order
failoverConfig.setFailoverStrategy(FailoverStrategy.activeActive);

//this is how long it will try on connection and wait for a response until giving up and trying another connection
failoverConfig.setTimeoutSeconds(120);

//after you have cycled through all the connections you can wait a little longer for one of the connections to finish
failoverConfig.setExtraTimeoutSeconds(50);

//this is how much time to remember that a connection had errors.  If an error hasnt occurred in a certain amount of time,
//it will be forgotten
failoverConfig.setMinutesToKeepErrors(5);

//if you have a lot of hibernate mapped classes or something, it will give a buffer for the first X seconds for the JVM
//to load and initialize
failoverConfig.setSecondsForClassesToLoad(20);

//register this configuration
FailoverClient.initFailoverClient(failoverConfig);

Here are some settings in grouper.client.properties.  Note saving to disk takes a few milliseconds

########################################
## Misc settings
########################################

# path of a writable directory where files can be created or stored
# for example, cache of discovery configuration, or failover state
# dot is the current directory...  note, this directory must exist
# or it will be created (attempted)
# if this is blank, none of these features will be used, and
# no files will be saved
grouperClient.cacheDirectory = .

# this will save the failover state to a file so if the JVM is stopped, it
# will be there when it starts again.  
# Set to 0 to store on every use (recommended if used command line)
# or set to -1 to not store or read ever
# grouperClient.cacheDirectory must be set
grouperClient.saveFailoverStateEverySeconds = 60
# if the failover client should use threads.  If it doesnt then you cant detect timeouts
grouperClient.failoverClientUseThreads = true

To do

  • Could provide retry after sleeping if not success...
  • Could have error count where below error count does not change connection affinity

See Also

Grouper Client

Discovery Client