package com.mongodb.internal.connection;
import com.mongodb.AuthenticationMechanism;
import com.mongodb.MongoCommandException;
import com.mongodb.MongoSecurityException;
import com.mongodb.async.SingleResultCallback;
import com.mongodb.connection.ConnectionDescription;
import org.bson.BsonDocument;
import org.bson.BsonInt32;
import org.bson.BsonString;
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 {
X509Authenticator(final MongoCredentialWithCache credential) {
super(credential);
}
@Override
void authenticate(final InternalConnection connection, final ConnectionDescription connectionDescription) {
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) {
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) {
callback.onResult(null, translateThrowable(t));
} else {
callback.onResult(null, null);
}
}
});
} catch (Throwable t) {
callback.onResult(null, t);
}
}
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");
}
}
}