org.neo4j.kernel.lifecycle.SafeLifecycle Maven / Gradle / Ivy
Show all versions of ongdb-common Show documentation
/*
* Copyright (c) 2018-2020 "Graph Foundation,"
* Graph Foundation, Inc. [https://graphfoundation.org]
*
* This file is part of ONgDB.
*
* ONgDB is free software: you can redistribute it and/or modify
* it under the terms of the GNU 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
/*
* Copyright (c) 2002-2020 "Neo4j,"
* Neo4j Sweden AB [http://neo4j.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 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
package org.neo4j.kernel.lifecycle;
import static org.neo4j.kernel.lifecycle.SafeLifecycle.State.HALT;
import static org.neo4j.kernel.lifecycle.SafeLifecycle.State.IDLE;
import static org.neo4j.kernel.lifecycle.SafeLifecycle.State.PRE;
import static org.neo4j.kernel.lifecycle.SafeLifecycle.State.RUN;
/**
* A safer lifecycle adapter with strict semantics and as
* a result simpler error handling and reasoning. If the
* semantics of this class is not consistent with what you
* are after, then it is probably better to create a new
* adapter with the specific semantics required.
*
* Guide
*
* *: No-op (will not invoke operation)
* -: Error (will throw IllegalStateException)
*
* P: PRE ("pre init")
* I: IDLE ("initialized" or "stopped")
* R: RUN ("started")
* H: HALT ("shutdown")
*
* A successful operation is an operation not throwing an exception.
*
* End states on a successful operation.
*
* ---------------------------------------------------
* | FROM \ op | init() start() stop() shutdown() |
* ---------------------------------------------------
* | PRE | IDLE - - HALT(*) |
* | IDLE | - RUN IDLE(*) HALT |
* | RUN | - - IDLE - |
* | HALT | - - - - |
* ---------------------------------------------------
*
* End states on a failed operation.
*
* ---------------------------------------------------
* | FROM \ op | init() start() stop() shutdown() |
* ---------------------------------------------------
* | PRE | PRE - - HALT(*) |
* | IDLE | - IDLE IDLE(*) HALT |
* | RUN | - - IDLE - |
* | HALT | - - - - |
* ---------------------------------------------------
*
* A few notes:
*
* - will not invoke stop0() if start0() wasn't successful
* - will not invoke shutdown0() if init0() wasn't successful
*
* The expectation with regards to error handling and cleanup is that
* an unclean start() is cleaned up by the start0() method and thus
* the component is left in IDLE. The same goes for issues happening
* during init0(), which leaves the component in PRE.
*
* Because of the way that {@link LifeSupport} operates today, this
* class will ignore stop() calls made while in IDLE. Similarly, calls
* to shutdown() will be ignored while in PRE. This allows this class
* to be managed by a {@link LifeSupport} without throwing
* {@link IllegalStateException} on those state transitions, which
* otherwise would have been disallowed and handled in the same way
* as other illegal state transitions.
*
* This adapter will not allow a shutdown lifecycle to be reinitialized
* and started again.
*/
public abstract class SafeLifecycle implements Lifecycle
{
private State state;
protected SafeLifecycle()
{
this( PRE );
}
SafeLifecycle( State state )
{
this.state = state;
}
/**
* @param expected The expected from state.
* @param to The to state.
* @param op The state transition operation.
* @param force Causes the state to be updated regardless of if the operation throws.
* @throws Throwable Issues.
*/
private void transition( State expected, State to, Operation op, boolean force ) throws Throwable
{
if ( state != expected )
{
throw new IllegalStateException( String.format( "Expected %s but was %s", expected, state ) );
}
if ( force )
{
state = to;
op.run();
}
else
{
op.run();
state = to;
}
}
@Override
public final synchronized void init() throws Throwable
{
transition( PRE, IDLE, this::init0, false );
}
@Override
public final synchronized void start() throws Throwable
{
transition( IDLE, RUN, this::start0, false );
}
@Override
public final synchronized void stop() throws Throwable
{
if ( state == IDLE )
{
return;
}
transition( RUN, IDLE, this::stop0, true );
}
@Override
public final synchronized void shutdown() throws Throwable
{
if ( state == PRE )
{
state = HALT;
return;
}
transition( IDLE, HALT, this::shutdown0, true );
}
public void init0() throws Throwable
{
}
public void start0() throws Throwable
{
}
public void stop0() throws Throwable
{
}
public void shutdown0() throws Throwable
{
}
public State state()
{
return state;
}
enum State
{
PRE,
IDLE,
RUN,
HALT
}
interface Operation
{
void run() throws Throwable;
}
}