
org.neo4j.cluster.NetworkedServerFactory Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of neo4j-cluster Show documentation
Show all versions of neo4j-cluster Show documentation
Library implementing Paxos and Heartbeat components required for High Availability Neo4j
/*
* Copyright (c) 2002-2016 "Neo Technology,"
* Network Engine for Objects in Lund AB [http://neotechnology.com]
*
* This file is part of Neo4j.
*
* Neo4j is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero 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 Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see .
*/
package org.neo4j.cluster;
import java.net.URI;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import org.neo4j.cluster.com.NetworkReceiver;
import org.neo4j.cluster.com.NetworkSender;
import org.neo4j.cluster.protocol.atomicbroadcast.AtomicBroadcastSerializer;
import org.neo4j.cluster.protocol.atomicbroadcast.ObjectInputStreamFactory;
import org.neo4j.cluster.protocol.atomicbroadcast.ObjectOutputStreamFactory;
import org.neo4j.cluster.protocol.atomicbroadcast.multipaxos.AcceptorInstanceStore;
import org.neo4j.cluster.protocol.election.ElectionCredentialsProvider;
import org.neo4j.cluster.statemachine.StateTransitionLogger;
import org.neo4j.cluster.timeout.TimeoutStrategy;
import org.neo4j.helpers.HostnamePort;
import org.neo4j.helpers.NamedThreadFactory;
import org.neo4j.kernel.configuration.Config;
import org.neo4j.kernel.lifecycle.LifeSupport;
import org.neo4j.kernel.lifecycle.Lifecycle;
import org.neo4j.logging.LogProvider;
/**
* TODO
*/
public class NetworkedServerFactory
{
private LifeSupport life;
private ProtocolServerFactory protocolServerFactory;
private TimeoutStrategy timeoutStrategy;
private final NetworkReceiver.Monitor networkReceiverMonitor;
private final NetworkSender.Monitor networkSenderMonitor;
private LogProvider logProvider;
private ObjectInputStreamFactory objectInputStreamFactory;
private ObjectOutputStreamFactory objectOutputStreamFactory;
private final NamedThreadFactory.Monitor namedThreadFactoryMonitor;
public NetworkedServerFactory( LifeSupport life, ProtocolServerFactory protocolServerFactory,
TimeoutStrategy timeoutStrategy,
LogProvider logProvider,
ObjectInputStreamFactory objectInputStreamFactory,
ObjectOutputStreamFactory objectOutputStreamFactory,
NetworkReceiver.Monitor networkReceiverMonitor,
NetworkSender.Monitor networkSenderMonitor,
NamedThreadFactory.Monitor namedThreadFactoryMonitor)
{
this.life = life;
this.protocolServerFactory = protocolServerFactory;
this.timeoutStrategy = timeoutStrategy;
this.networkReceiverMonitor = networkReceiverMonitor;
this.networkSenderMonitor = networkSenderMonitor;
this.logProvider = logProvider;
this.objectInputStreamFactory = objectInputStreamFactory;
this.objectOutputStreamFactory = objectOutputStreamFactory;
this.namedThreadFactoryMonitor = namedThreadFactoryMonitor;
}
public ProtocolServer newNetworkedServer( final Config config, AcceptorInstanceStore acceptorInstanceStore,
ElectionCredentialsProvider electionCredentialsProvider )
{
final NetworkReceiver receiver = new NetworkReceiver( networkReceiverMonitor,
new NetworkReceiver.Configuration()
{
@Override
public HostnamePort clusterServer()
{
return config.get( ClusterSettings.cluster_server );
}
@Override
public int defaultPort()
{
return 5001;
}
@Override
public String name()
{
return null;
}
}, logProvider );
final NetworkSender sender = new NetworkSender(networkSenderMonitor,
new NetworkSender.Configuration()
{
@Override
public int defaultPort()
{
return 5001;
}
@Override
public int port()
{
return config.get( ClusterSettings.cluster_server ).getPort();
}
}, receiver, logProvider );
ExecutorLifecycleAdapter stateMachineExecutor = new ExecutorLifecycleAdapter( () ->
Executors.newSingleThreadExecutor(
new NamedThreadFactory( "State machine", namedThreadFactoryMonitor ) ) );
final ProtocolServer protocolServer = protocolServerFactory.newProtocolServer(
config.get( ClusterSettings.server_id ), config.get( ClusterSettings.max_acceptors ),
timeoutStrategy, receiver, sender,
acceptorInstanceStore, electionCredentialsProvider, stateMachineExecutor, objectInputStreamFactory,
objectOutputStreamFactory );
receiver.addNetworkChannelsListener( new NetworkReceiver.NetworkChannelsListener()
{
private StateTransitionLogger logger;
@Override
public void listeningAt( URI me )
{
protocolServer.listeningAt( me );
if ( logger == null )
{
logger = new StateTransitionLogger( logProvider,
new AtomicBroadcastSerializer(objectInputStreamFactory, objectOutputStreamFactory));
protocolServer.addStateTransitionListener( logger );
}
}
@Override
public void channelOpened( URI to )
{
}
@Override
public void channelClosed( URI to )
{
}
} );
life.add( stateMachineExecutor );
// Timeout timer - triggers every 10 ms
life.add( new Lifecycle()
{
private ScheduledExecutorService scheduler;
@Override
public void init()
throws Throwable
{
protocolServer.getTimeouts().tick( System.currentTimeMillis() );
}
@Override
public void start()
throws Throwable
{
scheduler = Executors.newSingleThreadScheduledExecutor( new NamedThreadFactory( "timeout" ) );
scheduler.scheduleWithFixedDelay( new Runnable()
{
@Override
public void run()
{
long now = System.currentTimeMillis();
protocolServer.getTimeouts().tick( now );
}
}, 0, 10, TimeUnit.MILLISECONDS );
}
@Override
public void stop()
throws Throwable
{
scheduler.shutdownNow();
}
@Override
public void shutdown()
throws Throwable
{
}
} );
// Add this last to ensure that timeout service is setup first
life.add( sender );
life.add( receiver );
return protocolServer;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy