org.apache.cassandra.utils.Clock Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of cassandra-all Show documentation
Show all versions of cassandra-all Show documentation
The Apache Cassandra Project develops a highly scalable second-generation distributed database, bringing together Dynamo's fully distributed design and Bigtable's ColumnFamily-based data model.
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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
*
* http://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.apache.cassandra.utils;
import java.util.concurrent.TimeUnit;
import org.slf4j.Logger;
import static org.apache.cassandra.config.CassandraRelevantProperties.CLOCK_GLOBAL;
import static org.apache.cassandra.utils.Shared.Scope.SIMULATION;
/**
* Wrapper around time related functions that are either implemented by using the default JVM calls
* or by using a custom implementation for testing purposes.
*
* See {@link Global#instance} for how to use a custom implementation.
*
* Please note that {@link java.time.Clock} wasn't used, as it would not be possible to provide an
* implementation for {@link #nanoTime()} with the exact same properties of {@link System#nanoTime()}.
*/
@Shared(scope = SIMULATION)
public interface Clock
{
public static class Global
{
// something weird happens with class loading Logger that can cause a deadlock
private static Throwable FAILED_TO_INITIALISE;
private static String INITIALIZE_MESSAGE;
/**
* Static singleton object that will be instantiated by default with a system clock
* implementation. Set cassandra.clock
system property to a FQCN to use a
* different implementation instead.
*/
private static final Clock instance;
static
{
String classname = CLOCK_GLOBAL.getString();
Clock clock = new Default();
Throwable errorOutcome = null;
String outcome = null;
if (classname != null)
{
try
{
outcome = "Using custom clock implementation: " + classname;
clock = (Clock) Class.forName(classname).newInstance();
}
catch (Throwable t)
{
outcome = "Failed to load clock implementation " + classname;
errorOutcome = t;
}
}
instance = clock;
FAILED_TO_INITIALISE = errorOutcome;
INITIALIZE_MESSAGE = outcome;
}
public static void logInitializationOutcome(Logger logger)
{
if (FAILED_TO_INITIALISE != null)
{
logger.error(INITIALIZE_MESSAGE, FAILED_TO_INITIALISE);
}
else if (INITIALIZE_MESSAGE != null)
{
logger.debug(INITIALIZE_MESSAGE);
}
FAILED_TO_INITIALISE = null;
INITIALIZE_MESSAGE = null;
}
/**
* Semantically equivalent to {@link System#nanoTime()}
*/
public static long nanoTime()
{
return instance.nanoTime();
}
/**
* Semantically equivalent to {@link System#currentTimeMillis()}
*/
public static long currentTimeMillis()
{
return instance.currentTimeMillis();
}
}
public static class Default implements Clock
{
/**
* {@link System#nanoTime()}
*/
public long nanoTime()
{
return System.nanoTime(); // checkstyle: permit system clock
}
/**
* {@link System#currentTimeMillis()}
*/
public long currentTimeMillis()
{
return System.currentTimeMillis(); // checkstyle: permit system clock
}
}
/**
* Semantically equivalent to {@link System#nanoTime()}
*/
public long nanoTime();
/**
* Semantically equivalent to {@link System#currentTimeMillis()}
*/
public long currentTimeMillis();
@Intercept
public static void waitUntil(long deadlineNanos) throws InterruptedException
{
long waitNanos = deadlineNanos - Clock.Global.nanoTime();
if (waitNanos > 0)
TimeUnit.NANOSECONDS.sleep(waitNanos);
}
}