public class ClientState
extends java.lang.Object
Modifier and Type | Field and Description |
---|---|
boolean |
isInternal |
Modifier | Constructor and Description |
---|---|
protected |
ClientState(ClientState source) |
protected |
ClientState(java.net.InetSocketAddress remoteAddress) |
Modifier and Type | Method and Description |
---|---|
ClientState |
cloneWithKeyspaceIfSet(java.lang.String keyspace)
Clone this ClientState object, but use the provided keyspace instead of the
keyspace in this ClientState object.
|
void |
ensureAllKeyspacesPermission(Permission perm) |
void |
ensureIsSuperuser(java.lang.String message) |
void |
ensureKeyspacePermission(java.lang.String keyspace,
Permission perm) |
void |
ensureNotAnonymous() |
void |
ensurePermission(Permission permission,
Function function) |
void |
ensurePermission(Permission perm,
IResource resource) |
void |
ensureTablePermission(java.lang.String keyspace,
java.lang.String table,
Permission perm) |
void |
ensureTablePermission(TableMetadata table,
Permission perm) |
void |
ensureTablePermission(TableMetadataRef tableRef,
Permission perm) |
static ClientState |
forExternalCalls(java.net.SocketAddress remoteAddress) |
static ClientState |
forInternalCalls() |
static ClientState |
forInternalCalls(java.lang.String keyspace) |
static QueryHandler |
getCQLQueryHandler() |
java.util.Optional<java.lang.String> |
getDriverName() |
java.util.Optional<java.lang.String> |
getDriverVersion() |
java.lang.String |
getKeyspace() |
java.lang.String |
getRawKeyspace() |
java.net.InetSocketAddress |
getRemoteAddress() |
static long |
getTimestamp()
This clock guarantees that updates for the same ClientState will be ordered
in the sequence seen, even if multiple updates happen in the same millisecond.
|
long |
getTimestampForPaxos(long minTimestampToUse)
Returns a timestamp suitable for paxos given the timestamp of the last known commit (or in progress update).
|
AuthenticatedUser |
getUser() |
void |
login(AuthenticatedUser user)
Attempts to login the given user.
|
void |
setDriverName(java.lang.String driverName) |
void |
setDriverVersion(java.lang.String driverVersion) |
void |
setKeyspace(java.lang.String ks) |
void |
validateLogin() |
protected ClientState(java.net.InetSocketAddress remoteAddress)
protected ClientState(ClientState source)
public static ClientState forInternalCalls()
public static ClientState forInternalCalls(java.lang.String keyspace)
public static ClientState forExternalCalls(java.net.SocketAddress remoteAddress)
public ClientState cloneWithKeyspaceIfSet(java.lang.String keyspace)
public static long getTimestamp()
public long getTimestampForPaxos(long minTimestampToUse)
Paxos ensures that the timestamp it uses for commits respects the serial order of those commits. It does so by having each replica reject any proposal whose timestamp is not strictly greater than the last proposal it accepted. So in practice, which timestamp we use for a given proposal doesn't affect correctness but it does affect the chance of making progress (if we pick a timestamp lower than what has been proposed before, our new proposal will just get rejected).
As during the prepared phase replica send us the last propose they accepted, a first option would be to take the maximum of those last accepted proposal timestamp plus 1 (and use a default value, say 0, if it's the first known proposal for the partition). This would most work (giving commits the timestamp 0, 1, 2, ... in the order they are commited) up to 2 important caveats: 1) it would give a very poor experience when Paxos and non-Paxos updates are mixed in the same partition, since paxos operations wouldn't be using microseconds timestamps. And while you shouldn't theoretically mix the 2 kind of operations, this would still be pretty unintuitive. And what if you started writing normal updates and realize later you should switch to Paxos to enforce a property you want? 2) this wouldn't actually be safe due to the expiration set on the Paxos state table.
So instead, we initially chose to use the current time in microseconds as for normal update. Which works in general but mean that clock skew creates unavailability periods for Paxos updates (either a node has his clock in the past and he may no be able to get commit accepted until its clock catch up, or a node has his clock in the future and then once one of its commit his accepted, other nodes ones won't be until they catch up). This is ok for small clock skew (few ms) but can be pretty bad for large one.
Hence our current solution: we mix both approaches. That is, we compare the timestamp of the last known accepted proposal and the local time. If the local time is greater, we use it, thus keeping paxos timestamps locked to the current time in general (making mixing Paxos and non-Paxos more friendly, and behaving correctly when the paxos state expire (as long as your maximum clock skew is lower than the Paxos state expiration time)). Otherwise (the local time is lower than the last proposal, meaning that this last proposal was done with a clock in the future compared to the local one), we use the last proposal timestamp plus 1, ensuring progress.
minTimestampToUse
- the max timestamp of the last proposal accepted by replica having responded
to the prepare phase of the paxos round this is for. In practice, that's the minimum timestamp this method
may return.getTimestamp()
method, the return value is not guaranteed to be unique (nor
monotonic) across calls since it can return it's argument (so if the same argument is passed multiple times,
it may be returned multiple times). Note that we still ensure Paxos "ballot" are unique (for different
proposal) by (securely) randomizing the non-timestamp part of the UUID.public java.util.Optional<java.lang.String> getDriverName()
public java.util.Optional<java.lang.String> getDriverVersion()
public void setDriverName(java.lang.String driverName)
public void setDriverVersion(java.lang.String driverVersion)
public static QueryHandler getCQLQueryHandler()
public java.net.InetSocketAddress getRemoteAddress()
public java.lang.String getRawKeyspace()
public java.lang.String getKeyspace() throws InvalidRequestException
InvalidRequestException
public void setKeyspace(java.lang.String ks)
public void login(AuthenticatedUser user)
public void ensureAllKeyspacesPermission(Permission perm)
public void ensureKeyspacePermission(java.lang.String keyspace, Permission perm)
public void ensureTablePermission(java.lang.String keyspace, java.lang.String table, Permission perm)
public void ensureTablePermission(TableMetadataRef tableRef, Permission perm)
public void ensureTablePermission(TableMetadata table, Permission perm)
public void ensurePermission(Permission perm, IResource resource)
public void ensurePermission(Permission permission, Function function)
public void validateLogin()
public void ensureNotAnonymous()
public void ensureIsSuperuser(java.lang.String message)
public AuthenticatedUser getUser()
Copyright © 2009-2021 The Apache Software Foundation