package com.mongodb.internal.connection;
import com.mongodb.AuthenticationMechanism;
import com.mongodb.MongoCommandException;
import com.mongodb.MongoSecurityException;
import com.mongodb.diagnostics.logging.Logger;
import com.mongodb.diagnostics.logging.Loggers;
import com.mongodb.internal.async.SingleResultCallback;
import com.mongodb.connection.ConnectionDescription;
import org.bson.BsonDocument;
import org.bson.BsonInt32;
import org.bson.BsonString;
import static com.mongodb.internal.async.ErrorHandlingResultCallback.errorHandlingCallback;
import static com.mongodb.internal.connection.CommandHelper.executeCommand;
import static com.mongodb.internal.connection.CommandHelper.executeCommandAsync;
import static com.mongodb.internal.operation.ServerVersionHelper.serverIsLessThanVersionThreeDotFour;
class X509Authenticator extends Authenticator implements SpeculativeAuthenticator {
public static final Logger LOGGER = Loggers.getLogger("authenticator");
private BsonDocument speculativeAuthenticateResponse;
X509Authenticator(final MongoCredentialWithCache credential) {
super(credential);
}
@Override
void authenticate(final InternalConnection connection, final ConnectionDescription connectionDescription) {
if (this.speculativeAuthenticateResponse != null) {
return;
}
try {
validateUserName(connectionDescription);
BsonDocument authCommand = getAuthCommand(getMongoCredential().getUserName());
executeCommand(getMongoCredential().getSource(), authCommand, connection);
} catch (MongoCommandException e) {
throw new MongoSecurityException(getMongoCredential(), "Exception authenticating", e);
}
}
@Override
void authenticateAsync(final InternalConnection connection, final ConnectionDescription connectionDescription,
final SingleResultCallback<Void> callback) {
if (speculativeAuthenticateResponse != null) {
callback.onResult(null, null);
} else {
SingleResultCallback<Void> errHandlingCallback = errorHandlingCallback(callback, LOGGER);
try {
validateUserName(connectionDescription);
executeCommandAsync(getMongoCredential().getSource(), getAuthCommand(getMongoCredential().getUserName()), connection,
new SingleResultCallback<BsonDocument>() {
@Override
public void onResult(final BsonDocument nonceResult, final Throwable t) {
if (t != null) {
errHandlingCallback.onResult(null, translateThrowable(t));
} else {
errHandlingCallback.onResult(null, null);
}
}
});
} catch (Throwable t) {
errHandlingCallback.onResult(null, t);
}
}
}
@Override
public BsonDocument createSpeculativeAuthenticateCommand(final InternalConnection connection) {
return getAuthCommand(getMongoCredential().getUserName()).append("db", new BsonString("$external"));
}
@Override
public void setSpeculativeAuthenticateResponse(final BsonDocument response) {
this.speculativeAuthenticateResponse = response;
}
@Override
public BsonDocument getSpeculativeAuthenticateResponse() {
return speculativeAuthenticateResponse;
}
private BsonDocument getAuthCommand(final String userName) {
BsonDocument cmd = new BsonDocument();
cmd.put("authenticate", new BsonInt32(1));
if (userName != null) {
cmd.put("user", new BsonString(userName));
}
cmd.put("mechanism", new BsonString(AuthenticationMechanism.MONGODB_X509.getMechanismName()));
return cmd;
}
private Throwable translateThrowable(final Throwable t) {
if (t instanceof MongoCommandException) {
return new MongoSecurityException(getMongoCredential(), "Exception authenticating", t);
} else {
return t;
}
}
private void validateUserName(final ConnectionDescription connectionDescription) {
if (getMongoCredential().getUserName() == null && serverIsLessThanVersionThreeDotFour(connectionDescription)) {
throw new MongoSecurityException(getMongoCredential(), "User name is required for the MONGODB-X509 authentication mechanism "
+ "on server versions less than 3.4");
}
}
}