Iterate over realms asynchonously

This commit moves the iteration of realms for authentication and user lookup to
be done in an asynchronous fashion. The existing blocking methods have been deprecated
to allow custom realm implementors time to switch. All internal realms implement the
asynchronous methods.

This PR is another step toward the full migration to async authentication, but does not
complete the work. Additional work is needed for the LDAP realms, which make blocking
network calls. These blocking calls will be handled in a follow-up PR.

See elastic/elasticsearch#3790

Original commit: elastic/x-pack-elasticsearch@a65a9b2bb4
This commit is contained in:
Jay Modi 2016-11-28 09:28:51 -05:00 committed by GitHub
parent f265ab7cae
commit 637154cc6e
22 changed files with 884 additions and 498 deletions

View File

@ -0,0 +1,67 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License;
* you may not use this file except in compliance with the Elastic License.
*/
package org.elasticsearch.xpack.common;
import org.elasticsearch.action.ActionListener;
import java.util.Collections;
import java.util.List;
import java.util.function.BiConsumer;
/**
* This action listener wraps another listener and provides a framework for iteration over a List while calling an asynchronous function
* for each. The listener calls the {@link BiConsumer} with the current element in the list and a {@link ActionListener}. This function
* is expected to call the listener in case of success or failure due to an exception. If there is a failure due to an exception the wrapped
* listener's {@link ActionListener#onFailure(Exception)} method is called. If the consumer calls {@link #onResponse(Object)} with a
* non-null object, iteration will cease and the wrapped listener will be called with the response. In the case of a null value being passed
* to {@link #onResponse(Object)} then iteration will continue by applying the {@link BiConsumer} to the next item in the list; if the list
* has no more elements then the wrapped listener will be called with a null value.
*
* After creation, iteration is started by calling {@link #run()}
*/
public final class IteratingActionListener<T, U> implements ActionListener<T>, Runnable {
private final List<U> consumables;
private final ActionListener<T> delegate;
private final BiConsumer<U, ActionListener<T>> consumer;
private int position = 0;
public IteratingActionListener(ActionListener<T> delegate, BiConsumer<U, ActionListener<T>> consumer, List<U> consumables) {
this.delegate = delegate;
this.consumer = consumer;
this.consumables = Collections.unmodifiableList(consumables);
}
@Override
public void run() {
if (consumables.isEmpty()) {
onResponse(null);
} else if (position < 0 || position >= consumables.size()) {
onFailure(new IllegalStateException("invalid position [" + position + "]. List size [" + consumables.size() + "]"));
} else {
consumer.accept(consumables.get(position++), this);
}
}
@Override
public void onResponse(T response) {
if (response == null) {
if (position == consumables.size()) {
delegate.onResponse(null);
} else {
consumer.accept(consumables.get(position++), this);
}
} else {
delegate.onResponse(response);
}
}
@Override
public void onFailure(Exception e) {
delegate.onFailure(e);
}
}

View File

@ -92,6 +92,7 @@ public class TransportGetUsersAction extends HandledTransportAction<GetUsersRequ
reservedRealm.lookupUser(user, realmGroupListener); reservedRealm.lookupUser(user, realmGroupListener);
} }
} }
// user store lookups // user store lookups
if (specificUsersRequested && usersToSearchFor.isEmpty()) { if (specificUsersRequested && usersToSearchFor.isEmpty()) {
groupListener.onResponse(Collections.emptyList()); // no users requested notify groupListener.onResponse(Collections.emptyList()); // no users requested notify

View File

@ -20,6 +20,7 @@ import org.elasticsearch.node.Node;
import org.elasticsearch.rest.RestRequest; import org.elasticsearch.rest.RestRequest;
import org.elasticsearch.threadpool.ThreadPool; import org.elasticsearch.threadpool.ThreadPool;
import org.elasticsearch.transport.TransportMessage; import org.elasticsearch.transport.TransportMessage;
import org.elasticsearch.xpack.common.IteratingActionListener;
import org.elasticsearch.xpack.security.audit.AuditTrail; import org.elasticsearch.xpack.security.audit.AuditTrail;
import org.elasticsearch.xpack.security.audit.AuditTrailService; import org.elasticsearch.xpack.security.audit.AuditTrailService;
import org.elasticsearch.xpack.security.authc.Authentication.RealmRef; import org.elasticsearch.xpack.security.authc.Authentication.RealmRef;
@ -29,6 +30,7 @@ import org.elasticsearch.xpack.security.user.User;
import java.io.IOException; import java.io.IOException;
import java.util.List; import java.util.List;
import java.util.function.BiConsumer;
import java.util.function.Consumer; import java.util.function.Consumer;
import static org.elasticsearch.xpack.security.Security.setting; import static org.elasticsearch.xpack.security.Security.setting;
@ -233,19 +235,32 @@ public class AuthenticationService extends AbstractComponent {
handleNullToken(); handleNullToken();
} else { } else {
authenticationToken = token; authenticationToken = token;
Runnable action = () -> consumeUser(null); final List<Realm> realmsList = realms.asList();
final BiConsumer<Realm, ActionListener<User>> realmAuthenticatingConsumer = (realm, userListener) -> {
if (realm.supports(authenticationToken)) {
realm.authenticate(authenticationToken, ActionListener.wrap((user) -> {
if (user == null) {
// the user was not authenticated, call this so we can audit the correct event
request.realmAuthenticationFailed(authenticationToken, realm.name());
} else {
// user was authenticated, populate the authenticated by information
authenticatedBy = new RealmRef(realm.name(), realm.type(), nodeName);
}
userListener.onResponse(user);
}, userListener::onFailure));
} else {
userListener.onResponse(null);
}
};
final IteratingActionListener<User, Realm> authenticatingListener =
new IteratingActionListener<>(ActionListener.wrap(this::consumeUser,
(e) -> listener.onFailure(request.exceptionProcessingRequest(e, token))),
realmAuthenticatingConsumer, realmsList);
try { try {
for (Realm realm : realms) { authenticatingListener.run();
User user = authenticateToken(realm);
if (user != null) {
action = () -> consumeUser(user);
break;
}
}
} catch (Exception e) { } catch (Exception e) {
action = () -> listener.onFailure(request.exceptionProcessingRequest(e, token)); listener.onFailure(request.exceptionProcessingRequest(e, token));
} }
action.run();
} }
} }
@ -293,24 +308,6 @@ public class AuthenticationService extends AbstractComponent {
action.run(); action.run();
} }
/**
* Encapsulates the interaction with the realm and audit trail when attempting to authenticate a token. If the realm supports the
* token, authentication will be attempted. A successful authentication results in returning a non-null user in addition to setting
* the authenticatedBy value. A failed authentication will result in returning {@code null}
*/
private User authenticateToken(Realm realm) {
User user = null;
if (realm.supports(authenticationToken)) {
user = realm.authenticate(authenticationToken);
if (user == null) {
request.realmAuthenticationFailed(authenticationToken, realm.name());
} else {
authenticatedBy = new RealmRef(realm.name(), realm.type(), nodeName);
}
}
return user;
}
/** /**
* Consumes the {@link User} that resulted from attempting to authenticate a token against the {@link Realms}. When the user is * Consumes the {@link User} that resulted from attempting to authenticate a token against the {@link Realms}. When the user is
* {@code null}, authentication fails and does not proceed. When there is a user, the request is inspected to see if the run as * {@code null}, authentication fails and does not proceed. When there is a user, the request is inspected to see if the run as
@ -345,51 +342,31 @@ public class AuthenticationService extends AbstractComponent {
* names of users that exist using a timing attack * names of users that exist using a timing attack
*/ */
private void lookupRunAsUser(final User user, String runAsUsername, Consumer<User> userConsumer) { private void lookupRunAsUser(final User user, String runAsUsername, Consumer<User> userConsumer) {
// FIXME there are certain actions that could be allowed now with the default role that we should probably bail on! final List<Realm> realmsList = realms.asList();
Runnable action = () -> { final BiConsumer<Realm, ActionListener<User>> realmLookupConsumer = (realm, lookupUserListener) -> {
if (lookedupBy != null) { if (realm.userLookupSupported()) {
// the requested run as user does not exist, but we don't throw an error here otherwise this could let realm.lookupUser(runAsUsername, ActionListener.wrap((lookedupUser) -> {
// information leak about users in the system... instead we'll just let the authz service fail throw an if (lookedupUser != null) {
// authorization error lookedupBy = new RealmRef(realm.name(), realm.type(), nodeName);
if (lookedupBy != null) { lookupUserListener.onResponse(lookedupUser);
throw new IllegalStateException("we could not lookup the user but created a realm reference");
}
} else { } else {
userConsumer.accept(new User(user, new User(runAsUsername, Strings.EMPTY_ARRAY))); lookupUserListener.onResponse(null);
}
}, lookupUserListener::onFailure));
} else {
lookupUserListener.onResponse(null);
} }
}; };
final IteratingActionListener<User, Realm> userLookupListener =
new IteratingActionListener<>(ActionListener.wrap((lookupUser) -> userConsumer.accept(new User(user, lookupUser)),
(e) -> listener.onFailure(request.exceptionProcessingRequest(e, authenticationToken))),
realmLookupConsumer, realmsList);
try { try {
for (Realm realm : realms) { userLookupListener.run();
User runAsUser = lookupUser(realm, runAsUsername);
if (runAsUser != null) {
lookedupBy = new RealmRef(realm.name(), realm.type(), nodeName);
action = () -> userConsumer.accept(new User(user, runAsUser));
break;
}
}
} catch (Exception e) { } catch (Exception e) {
action = () -> listener.onFailure(request.exceptionProcessingRequest(e, authenticationToken)); listener.onFailure(request.exceptionProcessingRequest(e, authenticationToken));
} }
// we assign the listener call to an action to avoid calling the listener within a try block and auditing the wrong thing when
// an exception bubbles up even after successful authentication
action.run();
}
/**
* Handles the interaction with the realm and trying to lookup a user. If a user is found, this method also creates the
* {@link RealmRef} that identifies the realm that looked up the user
*/
private User lookupUser(Realm realm, String runAsUsername) {
User lookedUp = null;
if (realm.userLookupSupported()) {
lookedUp = realm.lookupUser(runAsUsername);
if (lookedUp != null) {
lookedupBy = new RealmRef(realm.name(), realm.type(), nodeName);
}
}
return lookedUp;
} }
/** /**

View File

@ -71,25 +71,43 @@ public abstract class Realm implements Comparable<Realm> {
public abstract AuthenticationToken token(ThreadContext context); public abstract AuthenticationToken token(ThreadContext context);
/** /**
* Authenticates the given token. A successful authentication will return the User associated * Authenticates the given token in a blocking fashion. A successful authentication will return the User associated
* with the given token. An unsuccessful authentication returns {@code null}. * with the given token. An unsuccessful authentication returns {@code null}. This method is deprecated in favor of
* {@link #authenticate(AuthenticationToken, ActionListener)}.
* *
* @param token The authentication token * @param token The authentication token
* @return The authenticated user or {@code null} if authentication failed. * @return The authenticated user or {@code null} if authentication failed.
*
*/ */
@Deprecated
public abstract User authenticate(AuthenticationToken token); public abstract User authenticate(AuthenticationToken token);
public void authenticate(AuthenticationToken token, ActionListener<User> listener) {
try {
listener.onResponse(authenticate(token));
} catch (Exception e) {
listener.onFailure(e);
}
}
/** /**
* Looks up the user identified the String identifier. A successful lookup will return the {@link User} identified * Looks up the user identified the String identifier. A successful lookup will return the {@link User} identified
* by the username. An unsuccessful lookup returns {@code null}. * by the username. An unsuccessful lookup returns {@code null}. This method is deprecated in favor of
* {@link #lookupUser(String, ActionListener)}
* *
* @param username the String identifier for the user * @param username the String identifier for the user
* @return the {@link User} or {@code null} if lookup failed * @return the {@link User} or {@code null} if lookup failed
*/ */
@Deprecated
public abstract User lookupUser(String username); public abstract User lookupUser(String username);
public void lookupUser(String username, ActionListener<User> listener) { public void lookupUser(String username, ActionListener<User> listener) {
listener.onResponse(lookupUser(username)); try {
User user = lookupUser(username);
listener.onResponse(user);
} catch (Exception e) {
listener.onFailure(e);
}
} }
public Map<String, Object> usageStats() { public Map<String, Object> usageStats() {
@ -100,9 +118,11 @@ public abstract class Realm implements Comparable<Realm> {
} }
/** /**
* Indicates whether this realm supports user lookup. * Indicates whether this realm supports user lookup. This method is deprecated. In the future if lookup is not supported, simply
* return null when called.
* @return true if the realm supports user lookup * @return true if the realm supports user lookup
*/ */
@Deprecated
public abstract boolean userLookupSupported(); public abstract boolean userLookupSupported();
@Override @Override

View File

@ -124,6 +124,24 @@ public class Realms extends AbstractLifecycleComponent implements Iterable<Realm
} }
} }
public List<Realm> asList() {
if (licenseState.isAuthAllowed() == false) {
return Collections.emptyList();
}
AllowedRealmType allowedRealmType = licenseState.allowedRealmType();
switch (allowedRealmType) {
case ALL:
return Collections.unmodifiableList(realms);
case DEFAULT:
return Collections.unmodifiableList(internalRealmsOnly);
case NATIVE:
return Collections.unmodifiableList(nativeRealmsOnly);
default:
throw new IllegalStateException("authentication should not be enabled");
}
}
public Realm realm(String name) { public Realm realm(String name) {
for (Realm realm : realms) { for (Realm realm : realms) {
if (name.equals(realm.config.name)) { if (name.equals(realm.config.name)) {

View File

@ -30,19 +30,13 @@ public class NativeRealm extends CachingUsernamePasswordRealm {
return true; return true;
} }
@Override
protected User doLookupUser(String username) {
return userStore.getUser(username);
}
@Override @Override
protected void doLookupUser(String username, ActionListener<User> listener) { protected void doLookupUser(String username, ActionListener<User> listener) {
userStore.getUsers(new String[] {username}, ActionListener.wrap(c -> listener.onResponse(c.stream().findAny().orElse(null)), userStore.getUser(username, listener);
listener::onFailure));
} }
@Override @Override
protected User doAuthenticate(UsernamePasswordToken token) { protected void doAuthenticate(UsernamePasswordToken token, ActionListener<User> listener) {
return userStore.verifyPassword(token.principal(), token.credentials()); userStore.verifyPassword(token.principal(), token.credentials(), listener);
} }
} }

View File

@ -63,7 +63,6 @@ import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.concurrent.CountDownLatch; import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicReference; import java.util.concurrent.atomic.AtomicReference;
import java.util.function.Consumer; import java.util.function.Consumer;
@ -108,13 +107,15 @@ public class NativeUsersStore extends AbstractComponent implements ClusterStateL
/** /**
* Blocking version of {@code getUser} that blocks until the User is returned * Blocking version of {@code getUser} that blocks until the User is returned
*/ */
public User getUser(String username) { public void getUser(String username, ActionListener<User> listener) {
if (state() != State.STARTED) { if (state() != State.STARTED) {
logger.trace("attempted to get user [{}] before service was started", username); logger.trace("attempted to get user [{}] before service was started", username);
return null; listener.onResponse(null);
} else {
getUserAndPassword(username, ActionListener.wrap((uap) -> {
listener.onResponse(uap == null ? null : uap.user());
}, listener::onFailure));
} }
UserAndPassword uap = getUserAndPassword(username);
return uap == null ? null : uap.user();
} }
/** /**
@ -586,22 +587,22 @@ public class NativeUsersStore extends AbstractComponent implements ClusterStateL
* *
* @param username username to lookup the user by * @param username username to lookup the user by
* @param password the plaintext password to verify * @param password the plaintext password to verify
* @return {@link} User object if successful or {@code null} if verification fails
*/ */
User verifyPassword(String username, final SecuredString password) { void verifyPassword(String username, final SecuredString password, ActionListener<User> listener) {
if (state() != State.STARTED) { if (state() != State.STARTED) {
logger.trace("attempted to verify user credentials for [{}] but service was not started", username); logger.trace("attempted to verify user credentials for [{}] but service was not started", username);
return null; listener.onResponse(null);
} else {
getUserAndPassword(username, ActionListener.wrap((userAndPassword) -> {
if (userAndPassword == null || userAndPassword.passwordHash() == null) {
listener.onResponse(null);
} else if (hasher.verify(password, userAndPassword.passwordHash())) {
listener.onResponse(userAndPassword.user());
} else {
listener.onResponse(null);
} }
}, listener::onFailure));
UserAndPassword user = getUserAndPassword(username);
if (user == null || user.passwordHash() == null) {
return null;
} }
if (hasher.verify(password, user.passwordHash())) {
return user.user();
}
return null;
} }
public boolean started() { public boolean started() {
@ -612,14 +613,11 @@ public class NativeUsersStore extends AbstractComponent implements ClusterStateL
return securityIndexExists; return securityIndexExists;
} }
ReservedUserInfo getReservedUserInfo(String username) throws Exception { void getReservedUserInfo(String username, ActionListener<ReservedUserInfo> listener) {
assert started(); assert started();
final AtomicReference<ReservedUserInfo> userInfoRef = new AtomicReference<>();
final AtomicReference<Exception> failure = new AtomicReference<>();
final CountDownLatch latch = new CountDownLatch(1);
client.prepareGet(SecurityTemplateService.SECURITY_INDEX_NAME, RESERVED_USER_DOC_TYPE, username) client.prepareGet(SecurityTemplateService.SECURITY_INDEX_NAME, RESERVED_USER_DOC_TYPE, username)
.execute(new LatchedActionListener<>(new ActionListener<GetResponse>() { .execute(new ActionListener<GetResponse>() {
@Override @Override
public void onResponse(GetResponse getResponse) { public void onResponse(GetResponse getResponse) {
if (getResponse.isExists()) { if (getResponse.isExists()) {
@ -627,12 +625,14 @@ public class NativeUsersStore extends AbstractComponent implements ClusterStateL
String password = (String) sourceMap.get(User.Fields.PASSWORD.getPreferredName()); String password = (String) sourceMap.get(User.Fields.PASSWORD.getPreferredName());
Boolean enabled = (Boolean) sourceMap.get(Fields.ENABLED.getPreferredName()); Boolean enabled = (Boolean) sourceMap.get(Fields.ENABLED.getPreferredName());
if (password == null || password.isEmpty()) { if (password == null || password.isEmpty()) {
failure.set(new IllegalStateException("password hash must not be empty!")); listener.onFailure(new IllegalStateException("password hash must not be empty!"));
} else if (enabled == null) { } else if (enabled == null) {
failure.set(new IllegalStateException("enabled must not be null!")); listener.onFailure(new IllegalStateException("enabled must not be null!"));
} else { } else {
userInfoRef.set(new ReservedUserInfo(password.toCharArray(), enabled)); listener.onResponse(new ReservedUserInfo(password.toCharArray(), enabled));
} }
} else {
listener.onResponse(null);
} }
} }
@ -641,30 +641,15 @@ public class NativeUsersStore extends AbstractComponent implements ClusterStateL
if (e instanceof IndexNotFoundException) { if (e instanceof IndexNotFoundException) {
logger.trace((Supplier<?>) () -> new ParameterizedMessage( logger.trace((Supplier<?>) () -> new ParameterizedMessage(
"could not retrieve built in user [{}] info since security index does not exist", username), e); "could not retrieve built in user [{}] info since security index does not exist", username), e);
listener.onResponse(null);
} else { } else {
logger.error( logger.error(
(Supplier<?>) () -> new ParameterizedMessage( (Supplier<?>) () -> new ParameterizedMessage(
"failed to retrieve built in user [{}] info", username), e); "failed to retrieve built in user [{}] info", username), e);
failure.set(e); listener.onFailure(null);
} }
} }
}, latch)); });
try {
final boolean responseReceived = latch.await(30, TimeUnit.SECONDS);
if (responseReceived == false) {
failure.set(new TimeoutException("timed out trying to get built in user [" + username + "]"));
}
} catch (InterruptedException e) {
failure.set(e);
}
Exception failureCause = failure.get();
if (failureCause != null) {
// if there is any sort of failure we need to throw an exception to prevent the fallback to the default password...
throw failureCause;
}
return userInfoRef.get();
} }
void getAllReservedUserInfo(ActionListener<Map<String, ReservedUserInfo>> listener) { void getAllReservedUserInfo(ActionListener<Map<String, ReservedUserInfo>> listener) {

View File

@ -28,7 +28,6 @@ import java.util.Arrays;
import java.util.Collection; import java.util.Collection;
import java.util.Collections; import java.util.Collections;
import java.util.List; import java.util.List;
import java.util.Map;
/** /**
* A realm for predefined users. These users can only be modified in terms of changing their passwords; no other modifications are allowed. * A realm for predefined users. These users can only be modified in terms of changing their passwords; no other modifications are allowed.
@ -54,53 +53,59 @@ public class ReservedRealm extends CachingUsernamePasswordRealm {
} }
@Override @Override
protected User doAuthenticate(UsernamePasswordToken token) { protected void doAuthenticate(UsernamePasswordToken token, ActionListener<User> listener) {
if (enabled == false) { if (enabled == false) {
return null; listener.onResponse(null);
} } else if (isReserved(token.principal(), config.globalSettings()) == false) {
if (isReserved(token.principal(), config.globalSettings()) == false) { listener.onResponse(null);
return null; } else {
} getUserInfo(token.principal(), ActionListener.wrap((userInfo) -> {
Runnable action;
final ReservedUserInfo userInfo = getUserInfo(token.principal());
if (userInfo != null) { if (userInfo != null) {
try { try {
if (Hasher.BCRYPT.verify(token.credentials(), userInfo.passwordHash)) { if (Hasher.BCRYPT.verify(token.credentials(), userInfo.passwordHash)) {
return getUser(token.principal(), userInfo); final User user = getUser(token.principal(), userInfo);
action = () -> listener.onResponse(user);
} else {
action = () -> listener.onFailure(Exceptions.authenticationError("failed to authenticate user [{}]",
token.principal()));
} }
} finally { } finally {
if (userInfo.passwordHash != DEFAULT_PASSWORD_HASH) { if (userInfo.passwordHash != DEFAULT_PASSWORD_HASH) {
Arrays.fill(userInfo.passwordHash, (char) 0); Arrays.fill(userInfo.passwordHash, (char) 0);
} }
} }
} else {
action = () -> listener.onFailure(Exceptions.authenticationError("failed to authenticate user [{}]",
token.principal()));
}
// we want the finally block to clear out the chars before we proceed further so we execute the action here
action.run();
}, listener::onFailure));
} }
// this was a reserved username - don't allow this to go to another realm...
throw Exceptions.authenticationError("failed to authenticate user [{}]", token.principal());
} }
@Override @Override
protected User doLookupUser(String username) { protected void doLookupUser(String username, ActionListener<User> listener) {
if (enabled == false) { if (enabled == false) {
if (anonymousEnabled && AnonymousUser.isAnonymousUsername(username, config.globalSettings())) { if (anonymousEnabled && AnonymousUser.isAnonymousUsername(username, config.globalSettings())) {
return anonymousUser; listener.onResponse(anonymousUser);
} }
return null; listener.onResponse(null);
} } else if (isReserved(username, config.globalSettings()) == false) {
listener.onResponse(null);
if (isReserved(username, config.globalSettings()) == false) { } else if (AnonymousUser.isAnonymousUsername(username, config.globalSettings())) {
return null; listener.onResponse(anonymousEnabled ? anonymousUser : null);
} } else {
getUserInfo(username, ActionListener.wrap((userInfo) -> {
if (AnonymousUser.isAnonymousUsername(username, config.globalSettings())) {
return anonymousEnabled ? anonymousUser : null;
}
final ReservedUserInfo userInfo = getUserInfo(username);
if (userInfo != null) { if (userInfo != null) {
return getUser(username, userInfo); listener.onResponse(getUser(username, userInfo));
} } else {
// this was a reserved username - don't allow this to go to another realm... // this was a reserved username - don't allow this to go to another realm...
throw Exceptions.authenticationError("failed to lookup user [{}]", username); listener.onFailure(Exceptions.authenticationError("failed to lookup user [{}]", username));
}
}, listener::onFailure));
}
} }
@Override @Override
@ -156,26 +161,24 @@ public class ReservedRealm extends CachingUsernamePasswordRealm {
} }
} }
private ReservedUserInfo getUserInfo(final String username) { private void getUserInfo(final String username, ActionListener<ReservedUserInfo> listener) {
if (nativeUsersStore.started() == false) { if (nativeUsersStore.started() == false) {
// we need to be able to check for the user store being started... // we need to be able to check for the user store being started...
return null; listener.onResponse(null);
} } else if (nativeUsersStore.securityIndexExists() == false) {
listener.onResponse(DEFAULT_USER_INFO);
if (nativeUsersStore.securityIndexExists() == false) { } else {
return DEFAULT_USER_INFO; nativeUsersStore.getReservedUserInfo(username, ActionListener.wrap((userInfo) -> {
}
try {
ReservedUserInfo userInfo = nativeUsersStore.getReservedUserInfo(username);
if (userInfo == null) { if (userInfo == null) {
return DEFAULT_USER_INFO; listener.onResponse(DEFAULT_USER_INFO);
} else {
listener.onResponse(userInfo);
} }
return userInfo; }, (e) -> {
} catch (Exception e) { logger.error((Supplier<?>) () ->
logger.error( new ParameterizedMessage("failed to retrieve password hash for reserved user [{}]", username), e);
(Supplier<?>) () -> new ParameterizedMessage("failed to retrieve password hash for reserved user [{}]", username), e); listener.onResponse(null);
return null; }));
} }
} }
} }

View File

@ -7,6 +7,7 @@ package org.elasticsearch.xpack.security.authc.file;
import java.util.Map; import java.util.Map;
import org.elasticsearch.action.ActionListener;
import org.elasticsearch.watcher.ResourceWatcherService; import org.elasticsearch.watcher.ResourceWatcherService;
import org.elasticsearch.xpack.security.authc.RealmConfig; import org.elasticsearch.xpack.security.authc.RealmConfig;
import org.elasticsearch.xpack.security.authc.support.CachingUsernamePasswordRealm; import org.elasticsearch.xpack.security.authc.support.CachingUsernamePasswordRealm;
@ -36,22 +37,23 @@ public class FileRealm extends CachingUsernamePasswordRealm {
} }
@Override @Override
protected User doAuthenticate(UsernamePasswordToken token) { protected void doAuthenticate(UsernamePasswordToken token, ActionListener<User> listener) {
if (!userPasswdStore.verifyPassword(token.principal(), token.credentials())) { if (userPasswdStore.verifyPassword(token.principal(), token.credentials())) {
return null;
}
String[] roles = userRolesStore.roles(token.principal()); String[] roles = userRolesStore.roles(token.principal());
listener.onResponse(new User(token.principal(), roles));
return new User(token.principal(), roles); } else {
listener.onResponse(null);
}
} }
@Override @Override
public User doLookupUser(String username) { protected void doLookupUser(String username, ActionListener<User> listener) {
if (userPasswdStore.userExists(username)) { if (userPasswdStore.userExists(username)) {
String[] roles = userRolesStore.roles(username); String[] roles = userRolesStore.roles(username);
return new User(username, roles); listener.onResponse(new User(username, roles));
} else {
listener.onResponse(null);
} }
return null;
} }
@Override @Override

View File

@ -8,6 +8,7 @@ package org.elasticsearch.xpack.security.authc.ldap.support;
import com.unboundid.ldap.sdk.LDAPException; import com.unboundid.ldap.sdk.LDAPException;
import org.apache.logging.log4j.message.ParameterizedMessage; import org.apache.logging.log4j.message.ParameterizedMessage;
import org.apache.logging.log4j.util.Supplier; import org.apache.logging.log4j.util.Supplier;
import org.elasticsearch.action.ActionListener;
import org.elasticsearch.xpack.security.authc.RealmConfig; import org.elasticsearch.xpack.security.authc.RealmConfig;
import org.elasticsearch.xpack.security.authc.support.CachingUsernamePasswordRealm; import org.elasticsearch.xpack.security.authc.support.CachingUsernamePasswordRealm;
import org.elasticsearch.xpack.security.authc.support.DnRoleMapper; import org.elasticsearch.xpack.security.authc.support.DnRoleMapper;
@ -36,30 +37,43 @@ public abstract class AbstractLdapRealm extends CachingUsernamePasswordRealm {
} }
/** /**
* Given a username and password, open to ldap, retrieve groups, map to roles and build the user. * Given a username and password, open a connection to ldap, bind to authenticate, retrieve groups, map to roles and build the user.
* * This user will then be passed to the listener
* @return User with elasticsearch roles
*/ */
@Override @Override
protected User doAuthenticate(UsernamePasswordToken token) { protected final void doAuthenticate(UsernamePasswordToken token, ActionListener<User> listener) {
// we use a runnable so that we call the listener outside of the try catch block. If we call within the try catch block and the
// listener throws an exception then we mistakenly could continue realm authentication when we already authenticated the user and
// there was some other issue
Runnable action;
try (LdapSession session = sessionFactory.session(token.principal(), token.credentials())) { try (LdapSession session = sessionFactory.session(token.principal(), token.credentials())) {
return createUser(token.principal(), session); final User user = createUser(token.principal(), session);
action = () -> listener.onResponse(user);
} catch (Exception e) { } catch (Exception e) {
logException("authentication", e, token.principal()); logException("authentication", e, token.principal());
return null; action = () -> listener.onResponse(null);
} }
action.run();
} }
@Override @Override
public User doLookupUser(String username) { protected final void doLookupUser(String username, ActionListener<User> listener) {
// we use a runnable so that we call the listener outside of the try catch block. If we call within the try catch block and the
// listener throws an exception then we mistakenly could continue realm lookup when we already found a matching user and
// there was some other issue
Runnable action;
if (sessionFactory.supportsUnauthenticatedSession()) { if (sessionFactory.supportsUnauthenticatedSession()) {
try (LdapSession session = sessionFactory.unauthenticatedSession(username)) { try (LdapSession session = sessionFactory.unauthenticatedSession(username)) {
return createUser(username, session); final User user = createUser(username, session);
action = () -> listener.onResponse(user);
} catch (Exception e) { } catch (Exception e) {
logException("lookup", e, username); logException("lookup", e, username);
action = () -> listener.onResponse(null);
} }
} else {
action = () -> listener.onResponse(null);
} }
return null; action.run();
} }
@Override @Override

View File

@ -9,6 +9,7 @@ import org.apache.logging.log4j.Logger;
import org.apache.logging.log4j.message.ParameterizedMessage; import org.apache.logging.log4j.message.ParameterizedMessage;
import org.apache.logging.log4j.util.Supplier; import org.apache.logging.log4j.util.Supplier;
import org.elasticsearch.ElasticsearchException; import org.elasticsearch.ElasticsearchException;
import org.elasticsearch.action.ActionListener;
import org.elasticsearch.common.Strings; import org.elasticsearch.common.Strings;
import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.common.util.concurrent.ThreadContext; import org.elasticsearch.common.util.concurrent.ThreadContext;
@ -81,13 +82,18 @@ public class PkiRealm extends Realm {
@Override @Override
public User authenticate(AuthenticationToken authToken) { public User authenticate(AuthenticationToken authToken) {
X509AuthenticationToken token = (X509AuthenticationToken)authToken; throw new UnsupportedOperationException("internal realms do not support blocking calls");
if (isCertificateChainTrusted(trustManager, token, logger) == false) {
return null;
} }
@Override
public void authenticate(AuthenticationToken authToken, ActionListener<User> listener) {
X509AuthenticationToken token = (X509AuthenticationToken)authToken;
if (isCertificateChainTrusted(trustManager, token, logger) == false) {
listener.onResponse(null);
} else {
Set<String> roles = roleMapper.resolveRoles(token.dn(), Collections.<String>emptyList()); Set<String> roles = roleMapper.resolveRoles(token.dn(), Collections.<String>emptyList());
return new User(token.principal(), roles.toArray(new String[roles.size()])); listener.onResponse(new User(token.principal(), roles.toArray(new String[roles.size()])));
}
} }
@Override @Override

View File

@ -5,13 +5,9 @@
*/ */
package org.elasticsearch.xpack.security.authc.support; package org.elasticsearch.xpack.security.authc.support;
import org.apache.logging.log4j.message.ParameterizedMessage;
import org.apache.logging.log4j.util.Supplier;
import org.elasticsearch.ElasticsearchSecurityException;
import org.elasticsearch.action.ActionListener; import org.elasticsearch.action.ActionListener;
import org.elasticsearch.common.cache.Cache; import org.elasticsearch.common.cache.Cache;
import org.elasticsearch.common.cache.CacheBuilder; import org.elasticsearch.common.cache.CacheBuilder;
import org.elasticsearch.common.cache.CacheLoader;
import org.elasticsearch.common.unit.TimeValue; import org.elasticsearch.common.unit.TimeValue;
import org.elasticsearch.xpack.security.authc.AuthenticationToken; import org.elasticsearch.xpack.security.authc.AuthenticationToken;
import org.elasticsearch.xpack.security.authc.RealmConfig; import org.elasticsearch.xpack.security.authc.RealmConfig;
@ -19,7 +15,6 @@ import org.elasticsearch.xpack.security.user.User;
import java.util.Map; import java.util.Map;
import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;
public abstract class CachingUsernamePasswordRealm extends UsernamePasswordRealm implements CachingRealm { public abstract class CachingUsernamePasswordRealm extends UsernamePasswordRealm implements CachingRealm {
@ -67,116 +62,71 @@ public abstract class CachingUsernamePasswordRealm extends UsernamePasswordRealm
* doAuthenticate * doAuthenticate
* *
* @param authToken The authentication token * @param authToken The authentication token
* @return an authenticated user with roles
*/ */
@Override @Override
public final User authenticate(AuthenticationToken authToken) { public final void authenticate(AuthenticationToken authToken, ActionListener<User> listener) {
UsernamePasswordToken token = (UsernamePasswordToken)authToken; UsernamePasswordToken token = (UsernamePasswordToken)authToken;
try {
if (cache == null) { if (cache == null) {
return doAuthenticate(token); doAuthenticate(token, listener);
} else {
authenticateWithCache(token, listener);
}
} catch (Exception e) {
// each realm should handle exceptions, if we get one here it should be considered fatal
listener.onFailure(e);
}
} }
try { private void authenticateWithCache(UsernamePasswordToken token, ActionListener<User> listener) {
UserWithHash userWithHash = cache.get(token.principal()); UserWithHash userWithHash = cache.get(token.principal());
if (userWithHash == null) { if (userWithHash == null) {
if (logger.isDebugEnabled()) { if (logger.isDebugEnabled()) {
logger.debug("user not found in cache, proceeding with normal authentication"); logger.debug("user not found in cache, proceeding with normal authentication");
} }
User user = doAuthenticate(token); doAuthenticateAndCache(token, ActionListener.wrap((user) -> {
if (user == null) { if (user != null) {
return null;
}
userWithHash = new UserWithHash(user, token.credentials(), hasher);
// it doesn't matter if we already computed it elsewhere
cache.put(token.principal(), userWithHash);
if (logger.isDebugEnabled()) {
logger.debug("authenticated user [{}], with roles [{}]", token.principal(), user.roles()); logger.debug("authenticated user [{}], with roles [{}]", token.principal(), user.roles());
} }
return user; listener.onResponse(user);
}, listener::onFailure));
} else if (userWithHash.hasHash()) {
if (userWithHash.verify(token.credentials())) {
logger.debug("authenticated user [{}], with roles [{}]", token.principal(), userWithHash.user.roles());
listener.onResponse(userWithHash.user);
} else {
cache.invalidate(token.principal());
doAuthenticateAndCache(token, ActionListener.wrap((user) -> {
if (user != null) {
logger.debug("cached user's password changed. authenticated user [{}], with roles [{}]", token.principal(),
user.roles());
}
listener.onResponse(user);
}, listener::onFailure));
}
} else {
cache.invalidate(token.principal());
doAuthenticateAndCache(token, ActionListener.wrap((user) -> {
if (user != null) {
logger.debug("cached user came from a lookup and could not be used for authentication. authenticated user [{}]" +
" with roles [{}]", token.principal(), user.roles());
}
listener.onResponse(user);
}, listener::onFailure));
}
} }
final boolean hadHash = userWithHash.hasHash(); private void doAuthenticateAndCache(UsernamePasswordToken token, ActionListener<User> listener) {
if (hadHash) { doAuthenticate(token, ActionListener.wrap((user) -> {
if (userWithHash.verify(token.credentials())) {
if (logger.isDebugEnabled()) {
logger.debug("authenticated user [{}], with roles [{}]", token.principal(), userWithHash.user.roles());
}
return userWithHash.user;
}
}
//this handles when a user's password has changed or the user was looked up for run as and not authenticated
cache.invalidate(token.principal());
User user = doAuthenticate(token);
if (user == null) { if (user == null) {
return null; listener.onResponse(null);
} } else {
userWithHash = new UserWithHash(user, token.credentials(), hasher); UserWithHash userWithHash = new UserWithHash(user, token.credentials(), hasher);
// it doesn't matter if we already computed it elsewhere // it doesn't matter if we already computed it elsewhere
cache.put(token.principal(), userWithHash); cache.put(token.principal(), userWithHash);
if (logger.isDebugEnabled()) { listener.onResponse(user);
if (hadHash) {
logger.debug("cached user's password changed. authenticated user [{}], with roles [{}]", token.principal(),
userWithHash.user.roles());
} else {
logger.debug("cached user came from a lookup and could not be used for authentication. authenticated user [{}]" +
" with roles [{}]", token.principal(), userWithHash.user.roles());
}
}
return userWithHash.user;
} catch (Exception ee) {
if (ee instanceof ElasticsearchSecurityException) {
// this should bubble out
throw ee;
}
if (logger.isTraceEnabled()) {
logger.trace(
(Supplier<?>) () -> new ParameterizedMessage(
"realm [{}] could not authenticate [{}]", type(), token.principal()), ee);
} else if (logger.isDebugEnabled()) {
logger.debug("realm [{}] could not authenticate [{}]", type(), token.principal());
}
return null;
}
}
@Override
public final User lookupUser(final String username) {
if (!userLookupSupported()) {
return null;
}
CacheLoader<String, UserWithHash> callback = key -> {
if (logger.isDebugEnabled()) {
logger.debug("user [{}] not found in cache, proceeding with normal lookup", username);
}
User user = doLookupUser(username);
if (user == null) {
return null;
}
return new UserWithHash(user, null, null);
};
try {
UserWithHash userWithHash = cache.computeIfAbsent(username, callback);
assert userWithHash != null : "the cache contract requires that a value returned from computeIfAbsent be non-null or an " +
"ExecutionException should be thrown";
return userWithHash.user;
} catch (ExecutionException ee) {
if (ee.getCause() instanceof ElasticsearchSecurityException) {
// this should bubble out
throw (ElasticsearchSecurityException) ee.getCause();
}
if (logger.isTraceEnabled()) {
logger.trace((Supplier<?>) () -> new ParameterizedMessage("realm [{}] could not lookup [{}]", name(), username), ee);
} else if (logger.isDebugEnabled()) {
logger.debug("realm [{}] could not lookup [{}]", name(), username);
}
return null;
} }
}, listener::onFailure));
} }
@Override @Override
@ -186,54 +136,60 @@ public abstract class CachingUsernamePasswordRealm extends UsernamePasswordRealm
return stats; return stats;
} }
protected abstract User doAuthenticate(UsernamePasswordToken token); protected abstract void doAuthenticate(UsernamePasswordToken token, ActionListener<User> listener);
@Override @Override
public void lookupUser(String username, ActionListener<User> listener) { public final void lookupUser(String username, ActionListener<User> listener) {
if (!userLookupSupported()) { if (!userLookupSupported()) {
listener.onResponse(null); listener.onResponse(null);
} else { } else if (cache != null) {
UserWithHash withHash = cache.get(username); UserWithHash withHash = cache.get(username);
if (withHash == null) { if (withHash == null) {
doLookupUser(username, ActionListener.wrap((user) -> {
try { try {
doLookupUser(username, ActionListener.wrap((user) -> {
Runnable action = () -> listener.onResponse(null);
if (user != null) { if (user != null) {
UserWithHash userWithHash = new UserWithHash(user, null, null); UserWithHash userWithHash = new UserWithHash(user, null, null);
try {
// computeIfAbsent is used here to avoid overwriting a value from a concurrent authenticate call as it
// contains the password hash, which provides a performance boost and we shouldn't just erase that
cache.computeIfAbsent(username, (n) -> userWithHash); cache.computeIfAbsent(username, (n) -> userWithHash);
} action = () -> listener.onResponse(userWithHash.user);
listener.onResponse(user);
} catch (ExecutionException e) { } catch (ExecutionException e) {
action = () -> listener.onFailure(e);
}
}
action.run();
}, listener::onFailure));
} catch (Exception e) {
listener.onFailure(e); listener.onFailure(e);
} }
}, listener::onFailure));
} else { } else {
listener.onResponse(withHash.user); listener.onResponse(withHash.user);
} }
} else {
doLookupUser(username, listener);
} }
} }
protected abstract User doLookupUser(String username); protected abstract void doLookupUser(String username, ActionListener<User> listener);
protected void doLookupUser(String username, ActionListener<User> listener) {
listener.onResponse(doLookupUser(username));
}
private static class UserWithHash { private static class UserWithHash {
User user; User user;
char[] hash; char[] hash;
Hasher hasher; Hasher hasher;
public UserWithHash(User user, SecuredString password, Hasher hasher) { UserWithHash(User user, SecuredString password, Hasher hasher) {
this.user = user; this.user = user;
this.hash = password == null ? null : hasher.hash(password); this.hash = password == null ? null : hasher.hash(password);
this.hasher = hasher; this.hasher = hasher;
} }
public boolean verify(SecuredString password) { boolean verify(SecuredString password) {
return hash != null && hasher.verify(password, hash); return hash != null && hasher.verify(password, hash);
} }
public boolean hasHash() { boolean hasHash() {
return hash != null; return hash != null;
} }
} }

View File

@ -9,10 +9,11 @@ import org.elasticsearch.common.util.concurrent.ThreadContext;
import org.elasticsearch.xpack.security.authc.AuthenticationToken; import org.elasticsearch.xpack.security.authc.AuthenticationToken;
import org.elasticsearch.xpack.security.authc.Realm; import org.elasticsearch.xpack.security.authc.Realm;
import org.elasticsearch.xpack.security.authc.RealmConfig; import org.elasticsearch.xpack.security.authc.RealmConfig;
import org.elasticsearch.xpack.security.user.User;
public abstract class UsernamePasswordRealm extends Realm { abstract class UsernamePasswordRealm extends Realm {
public UsernamePasswordRealm(String type, RealmConfig config) { UsernamePasswordRealm(String type, RealmConfig config) {
super(type, config); super(type, config);
} }
@ -25,4 +26,11 @@ public abstract class UsernamePasswordRealm extends Realm {
return token instanceof UsernamePasswordToken; return token instanceof UsernamePasswordToken;
} }
public final User authenticate(AuthenticationToken token) {
throw new UnsupportedOperationException("internal realms should not support blocking calls!!");
}
public final User lookupUser(String username) {
throw new UnsupportedOperationException("internal realms should not support blocking calls!");
}
} }

View File

@ -8,6 +8,7 @@ package org.elasticsearch.integration;
import org.apache.http.message.BasicHeader; import org.apache.http.message.BasicHeader;
import org.apache.http.util.EntityUtils; import org.apache.http.util.EntityUtils;
import org.elasticsearch.action.ActionListener; import org.elasticsearch.action.ActionListener;
import org.elasticsearch.action.support.PlainActionFuture;
import org.elasticsearch.client.Response; import org.elasticsearch.client.Response;
import org.elasticsearch.common.Strings; import org.elasticsearch.common.Strings;
import org.elasticsearch.common.network.NetworkModule; import org.elasticsearch.common.network.NetworkModule;
@ -236,7 +237,9 @@ public class ClearRealmsCacheTests extends SecurityIntegTestCase {
Map<String, Map<Realm, User>> users = new HashMap<>(); Map<String, Map<Realm, User>> users = new HashMap<>();
for (Realm realm : realms) { for (Realm realm : realms) {
for (String username : usernames) { for (String username : usernames) {
User user = realm.authenticate(tokens.get(username)); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(tokens.get(username), future);
User user = future.actionGet();
assertThat(user, notNullValue()); assertThat(user, notNullValue());
Map<Realm, User> realmToUser = users.get(username); Map<Realm, User> realmToUser = users.get(username);
if (realmToUser == null) { if (realmToUser == null) {
@ -251,7 +254,10 @@ public class ClearRealmsCacheTests extends SecurityIntegTestCase {
for (String username : usernames) { for (String username : usernames) {
for (Realm realm : realms) { for (Realm realm : realms) {
assertThat(realm.authenticate(tokens.get(username)), sameInstance(users.get(username).get(realm))); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(tokens.get(username), future);
User user = future.actionGet();
assertThat(user, sameInstance(users.get(username).get(realm)));
} }
} }
@ -261,7 +267,9 @@ public class ClearRealmsCacheTests extends SecurityIntegTestCase {
// now, user_a should have been evicted, but user_b should still be cached // now, user_a should have been evicted, but user_b should still be cached
for (String username : usernames) { for (String username : usernames) {
for (Realm realm : realms) { for (Realm realm : realms) {
User user = realm.authenticate(tokens.get(username)); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(tokens.get(username), future);
User user = future.actionGet();
assertThat(user, notNullValue()); assertThat(user, notNullValue());
scenario.assertEviction(users.get(username).get(realm), user); scenario.assertEviction(users.get(username).get(realm), user);
} }

View File

@ -0,0 +1,98 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License;
* you may not use this file except in compliance with the Elastic License.
*/
package org.elasticsearch.xpack.common;
import org.elasticsearch.ElasticsearchException;
import org.elasticsearch.action.ActionListener;
import org.elasticsearch.common.collect.HppcMaps.Object;
import org.elasticsearch.test.ESTestCase;
import org.junit.Assert;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.BiConsumer;
import static org.hamcrest.Matchers.sameInstance;
public class IteratingActionListenerTests extends ESTestCase {
public void testIteration() {
final int numberOfItems = scaledRandomIntBetween(1, 32);
final int numberOfIterations = scaledRandomIntBetween(1, numberOfItems);
List<Object> items = new ArrayList<>(numberOfItems);
for (int i = 0; i < numberOfItems; i++) {
items.add(new Object());
}
final AtomicInteger iterations = new AtomicInteger(0);
final BiConsumer<Object, ActionListener<Object>> consumer = (listValue, listener) -> {
final int current = iterations.incrementAndGet();
if (current == numberOfIterations) {
listener.onResponse(items.get(current - 1));
} else {
listener.onResponse(null);
}
};
IteratingActionListener<Object, Object> iteratingListener = new IteratingActionListener<>(ActionListener.wrap((object) -> {
assertNotNull(object);
assertThat(object, sameInstance(items.get(numberOfIterations - 1)));
}, (e) -> {
logger.error("unexpected exception", e);
fail("exception should not have been thrown");
}), consumer, items);
iteratingListener.run();
// we never really went async, its all chained together so verify this for sanity
assertEquals(numberOfIterations, iterations.get());
}
public void testIterationEmptyList() {
IteratingActionListener<Object, Object> listener = new IteratingActionListener<>(ActionListener.wrap(Assert::assertNull,
(e) -> {
logger.error("unexpected exception", e);
fail("exception should not have been thrown");
}), (listValue, iteratingListener) -> {
fail("consumer should not have been called!!!");
}, Collections.emptyList());
listener.run();
}
public void testFailure() {
final int numberOfItems = scaledRandomIntBetween(1, 32);
final int numberOfIterations = scaledRandomIntBetween(1, numberOfItems);
List<Object> items = new ArrayList<>(numberOfItems);
for (int i = 0; i < numberOfItems; i++) {
items.add(new Object());
}
final AtomicInteger iterations = new AtomicInteger(0);
final BiConsumer<Object, ActionListener<Object>> consumer = (listValue, listener) -> {
final int current = iterations.incrementAndGet();
if (current == numberOfIterations) {
listener.onFailure(new ElasticsearchException("expected exception"));
} else {
listener.onResponse(null);
}
};
final AtomicBoolean onFailureCalled = new AtomicBoolean(false);
IteratingActionListener<Object, Object> iteratingListener = new IteratingActionListener<>(ActionListener.wrap((object) -> {
fail("onResponse should not have been called, but was called with: " + object);
}, (e) -> {
assertEquals("expected exception", e.getMessage());
assertTrue(onFailureCalled.compareAndSet(false, true));
}), consumer, items);
iteratingListener.run();
// we never really went async, its all chained together so verify this for sanity
assertEquals(numberOfIterations, iterations.get());
assertTrue(onFailureCalled.get());
}
}

View File

@ -47,6 +47,8 @@ import static org.hamcrest.Matchers.nullValue;
import static org.hamcrest.Matchers.sameInstance; import static org.hamcrest.Matchers.sameInstance;
import static org.mockito.Matchers.any; import static org.mockito.Matchers.any;
import static org.mockito.Matchers.eq; import static org.mockito.Matchers.eq;
import static org.mockito.Mockito.doAnswer;
import static org.mockito.Mockito.doThrow;
import static org.mockito.Mockito.mock; import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.never; import static org.mockito.Mockito.never;
import static org.mockito.Mockito.reset; import static org.mockito.Mockito.reset;
@ -145,9 +147,9 @@ public class AuthenticationServiceTests extends ESTestCase {
public void testAuthenticateBothSupportSecondSucceeds() throws Exception { public void testAuthenticateBothSupportSecondSucceeds() throws Exception {
User user = new User("_username", "r1"); User user = new User("_username", "r1");
when(firstRealm.supports(token)).thenReturn(true); when(firstRealm.supports(token)).thenReturn(true);
when(firstRealm.authenticate(token)).thenReturn(null); // first fails mockAuthenticate(firstRealm, token, null);
when(secondRealm.supports(token)).thenReturn(true); when(secondRealm.supports(token)).thenReturn(true);
when(secondRealm.authenticate(token)).thenReturn(user); mockAuthenticate(secondRealm, token, user);
if (randomBoolean()) { if (randomBoolean()) {
when(firstRealm.token(threadContext)).thenReturn(token); when(firstRealm.token(threadContext)).thenReturn(token);
} else { } else {
@ -168,7 +170,7 @@ public class AuthenticationServiceTests extends ESTestCase {
User user = new User("_username", "r1"); User user = new User("_username", "r1");
when(firstRealm.supports(token)).thenReturn(false); when(firstRealm.supports(token)).thenReturn(false);
when(secondRealm.supports(token)).thenReturn(true); when(secondRealm.supports(token)).thenReturn(true);
when(secondRealm.authenticate(token)).thenReturn(user); mockAuthenticate(secondRealm, token, user);
when(secondRealm.token(threadContext)).thenReturn(token); when(secondRealm.token(threadContext)).thenReturn(token);
Authentication result = authenticateBlocking("_action", message, null); Authentication result = authenticateBlocking("_action", message, null);
@ -219,7 +221,7 @@ public class AuthenticationServiceTests extends ESTestCase {
User user = new User("_username", "r1"); User user = new User("_username", "r1");
when(firstRealm.token(threadContext)).thenReturn(token); when(firstRealm.token(threadContext)).thenReturn(token);
when(firstRealm.supports(token)).thenReturn(true); when(firstRealm.supports(token)).thenReturn(true);
when(firstRealm.authenticate(token)).thenReturn(user); mockAuthenticate(firstRealm, token, user);
Authentication result = authenticateBlocking("_action", message, null); Authentication result = authenticateBlocking("_action", message, null);
@ -276,7 +278,7 @@ public class AuthenticationServiceTests extends ESTestCase {
User fallback = randomBoolean() ? SystemUser.INSTANCE : null; User fallback = randomBoolean() ? SystemUser.INSTANCE : null;
when(firstRealm.token(threadContext)).thenReturn(token); when(firstRealm.token(threadContext)).thenReturn(token);
when(firstRealm.supports(token)).thenReturn(true); when(firstRealm.supports(token)).thenReturn(true);
when(firstRealm.authenticate(token)).thenReturn(user); mockAuthenticate(firstRealm, token, user);
ElasticsearchSecurityException e = ElasticsearchSecurityException e =
expectThrows(ElasticsearchSecurityException.class, () -> authenticateBlocking("_action", message, fallback)); expectThrows(ElasticsearchSecurityException.class, () -> authenticateBlocking("_action", message, fallback));
@ -289,7 +291,7 @@ public class AuthenticationServiceTests extends ESTestCase {
User user = new User("username", new String[] { "r1", "r2" }, null, null, null, false); User user = new User("username", new String[] { "r1", "r2" }, null, null, null, false);
when(firstRealm.token(threadContext)).thenReturn(token); when(firstRealm.token(threadContext)).thenReturn(token);
when(firstRealm.supports(token)).thenReturn(true); when(firstRealm.supports(token)).thenReturn(true);
when(firstRealm.authenticate(token)).thenReturn(user); mockAuthenticate(firstRealm, token, user);
ElasticsearchSecurityException e = ElasticsearchSecurityException e =
expectThrows(ElasticsearchSecurityException.class, () -> authenticateBlocking(restRequest)); expectThrows(ElasticsearchSecurityException.class, () -> authenticateBlocking(restRequest));
@ -303,7 +305,7 @@ public class AuthenticationServiceTests extends ESTestCase {
User fallback = randomBoolean() ? SystemUser.INSTANCE : null; User fallback = randomBoolean() ? SystemUser.INSTANCE : null;
when(firstRealm.token(threadContext)).thenReturn(token); when(firstRealm.token(threadContext)).thenReturn(token);
when(firstRealm.supports(token)).thenReturn(true); when(firstRealm.supports(token)).thenReturn(true);
when(firstRealm.authenticate(token)).thenReturn(user); mockAuthenticate(firstRealm, token, user);
Authentication result = authenticateBlocking("_action", message, fallback); Authentication result = authenticateBlocking("_action", message, fallback);
assertThat(result, notNullValue()); assertThat(result, notNullValue());
@ -317,7 +319,7 @@ public class AuthenticationServiceTests extends ESTestCase {
User user1 = new User("username", "r1", "r2"); User user1 = new User("username", "r1", "r2");
when(firstRealm.token(threadContext)).thenReturn(token); when(firstRealm.token(threadContext)).thenReturn(token);
when(firstRealm.supports(token)).thenReturn(true); when(firstRealm.supports(token)).thenReturn(true);
when(firstRealm.authenticate(token)).thenReturn(user1); mockAuthenticate(firstRealm, token, user1);
Authentication result = authenticateBlocking(restRequest); Authentication result = authenticateBlocking(restRequest);
assertThat(result, notNullValue()); assertThat(result, notNullValue());
assertThat(result.getUser(), sameInstance(user1)); assertThat(result.getUser(), sameInstance(user1));
@ -330,7 +332,7 @@ public class AuthenticationServiceTests extends ESTestCase {
User user1 = new User("username", "r1", "r2"); User user1 = new User("username", "r1", "r2");
when(firstRealm.token(threadContext)).thenReturn(token); when(firstRealm.token(threadContext)).thenReturn(token);
when(firstRealm.supports(token)).thenReturn(true); when(firstRealm.supports(token)).thenReturn(true);
when(firstRealm.authenticate(token)).thenReturn(user1); mockAuthenticate(firstRealm, token, user1);
Authentication authentication = authenticateBlocking("_action", message, SystemUser.INSTANCE); Authentication authentication = authenticateBlocking("_action", message, SystemUser.INSTANCE);
assertThat(authentication, notNullValue()); assertThat(authentication, notNullValue());
assertThat(authentication.getUser(), sameInstance(user1)); assertThat(authentication.getUser(), sameInstance(user1));
@ -382,7 +384,7 @@ public class AuthenticationServiceTests extends ESTestCase {
User user1 = new User("username", "r1", "r2"); User user1 = new User("username", "r1", "r2");
when(firstRealm.supports(token)).thenReturn(true); when(firstRealm.supports(token)).thenReturn(true);
when(firstRealm.token(threadContext)).thenReturn(token); when(firstRealm.token(threadContext)).thenReturn(token);
when(firstRealm.authenticate(token)).thenReturn(user1); mockAuthenticate(firstRealm, token, user1);
Authentication authentication = authenticateBlocking("_action", message, SystemUser.INSTANCE); Authentication authentication = authenticateBlocking("_action", message, SystemUser.INSTANCE);
assertThat(authentication, notNullValue()); assertThat(authentication, notNullValue());
assertThat(authentication.getUser(), sameInstance(user1)); assertThat(authentication.getUser(), sameInstance(user1));
@ -572,7 +574,8 @@ public class AuthenticationServiceTests extends ESTestCase {
AuthenticationToken token = mock(AuthenticationToken.class); AuthenticationToken token = mock(AuthenticationToken.class);
when(secondRealm.token(threadContext)).thenReturn(token); when(secondRealm.token(threadContext)).thenReturn(token);
when(secondRealm.supports(token)).thenReturn(true); when(secondRealm.supports(token)).thenReturn(true);
when(secondRealm.authenticate(token)).thenThrow(authenticationError("realm doesn't like authenticate")); doThrow(authenticationError("realm doesn't like authenticate"))
.when(secondRealm).authenticate(eq(token), any(ActionListener.class));
try { try {
authenticateBlocking("_action", message, null); authenticateBlocking("_action", message, null);
fail("exception should bubble out"); fail("exception should bubble out");
@ -586,7 +589,8 @@ public class AuthenticationServiceTests extends ESTestCase {
AuthenticationToken token = mock(AuthenticationToken.class); AuthenticationToken token = mock(AuthenticationToken.class);
when(secondRealm.token(threadContext)).thenReturn(token); when(secondRealm.token(threadContext)).thenReturn(token);
when(secondRealm.supports(token)).thenReturn(true); when(secondRealm.supports(token)).thenReturn(true);
when(secondRealm.authenticate(token)).thenThrow(authenticationError("realm doesn't like authenticate")); doThrow(authenticationError("realm doesn't like authenticate"))
.when(secondRealm).authenticate(eq(token), any(ActionListener.class));
try { try {
authenticateBlocking(restRequest); authenticateBlocking(restRequest);
fail("exception should bubble out"); fail("exception should bubble out");
@ -601,8 +605,9 @@ public class AuthenticationServiceTests extends ESTestCase {
threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, "run_as"); threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, "run_as");
when(secondRealm.token(threadContext)).thenReturn(token); when(secondRealm.token(threadContext)).thenReturn(token);
when(secondRealm.supports(token)).thenReturn(true); when(secondRealm.supports(token)).thenReturn(true);
when(secondRealm.authenticate(token)).thenReturn(new User("lookup user", new String[]{"user"})); mockAuthenticate(secondRealm, token, new User("lookup user", new String[]{"user"}));
when(secondRealm.lookupUser("run_as")).thenThrow(authenticationError("realm doesn't want to lookup")); doThrow(authenticationError("realm doesn't want to lookup"))
.when(secondRealm).lookupUser(eq("run_as"), any(ActionListener.class));
when(secondRealm.userLookupSupported()).thenReturn(true); when(secondRealm.userLookupSupported()).thenReturn(true);
try { try {
@ -619,8 +624,9 @@ public class AuthenticationServiceTests extends ESTestCase {
threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, "run_as"); threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, "run_as");
when(secondRealm.token(threadContext)).thenReturn(token); when(secondRealm.token(threadContext)).thenReturn(token);
when(secondRealm.supports(token)).thenReturn(true); when(secondRealm.supports(token)).thenReturn(true);
when(secondRealm.authenticate(token)).thenReturn(new User("lookup user", new String[]{"user"})); mockAuthenticate(secondRealm, token, new User("lookup user", new String[]{"user"}));
when(secondRealm.lookupUser("run_as")).thenThrow(authenticationError("realm doesn't want to lookup")); doThrow(authenticationError("realm doesn't want to " + "lookup"))
.when(secondRealm).lookupUser(eq("run_as"), any(ActionListener.class));
when(secondRealm.userLookupSupported()).thenReturn(true); when(secondRealm.userLookupSupported()).thenReturn(true);
try { try {
@ -639,8 +645,12 @@ public class AuthenticationServiceTests extends ESTestCase {
when(secondRealm.supports(token)).thenReturn(true); when(secondRealm.supports(token)).thenReturn(true);
final User user = new User("lookup user", new String[]{"user"}, "lookup user", "lookup@foo.foo", final User user = new User("lookup user", new String[]{"user"}, "lookup user", "lookup@foo.foo",
Collections.singletonMap("foo", "bar"), true); Collections.singletonMap("foo", "bar"), true);
when(secondRealm.authenticate(token)).thenReturn(user); mockAuthenticate(secondRealm, token, user);
when(secondRealm.lookupUser("run_as")).thenReturn(new User("looked up user", new String[]{"some role"})); doAnswer((i) -> {
ActionListener listener = (ActionListener) i.getArguments()[1];
listener.onResponse(new User("looked up user", new String[]{"some role"}));
return null;
}).when(secondRealm).lookupUser(eq("run_as"), any(ActionListener.class));
when(secondRealm.userLookupSupported()).thenReturn(true); when(secondRealm.userLookupSupported()).thenReturn(true);
Authentication result; Authentication result;
@ -671,9 +681,13 @@ public class AuthenticationServiceTests extends ESTestCase {
threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, "run_as"); threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, "run_as");
when(secondRealm.token(threadContext)).thenReturn(token); when(secondRealm.token(threadContext)).thenReturn(token);
when(secondRealm.supports(token)).thenReturn(true); when(secondRealm.supports(token)).thenReturn(true);
when(secondRealm.authenticate(token)).thenReturn(new User("lookup user", new String[]{"user"})); mockAuthenticate(secondRealm, token, new User("lookup user", new String[]{"user"}));
when(firstRealm.userLookupSupported()).thenReturn(true); when(firstRealm.userLookupSupported()).thenReturn(true);
when(firstRealm.lookupUser("run_as")).thenReturn(new User("looked up user", new String[]{"some role"})); doAnswer((i) -> {
ActionListener listener = (ActionListener) i.getArguments()[1];
listener.onResponse(new User("looked up user", new String[]{"some role"}));
return null;
}).when(firstRealm).lookupUser(eq("run_as"), any(ActionListener.class));
when(firstRealm.userLookupSupported()).thenReturn(true); when(firstRealm.userLookupSupported()).thenReturn(true);
Authentication result; Authentication result;
@ -700,7 +714,7 @@ public class AuthenticationServiceTests extends ESTestCase {
threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, ""); threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, "");
when(secondRealm.token(threadContext)).thenReturn(token); when(secondRealm.token(threadContext)).thenReturn(token);
when(secondRealm.supports(token)).thenReturn(true); when(secondRealm.supports(token)).thenReturn(true);
when(secondRealm.authenticate(token)).thenReturn(user); mockAuthenticate(secondRealm, token, user);
when(secondRealm.userLookupSupported()).thenReturn(true); when(secondRealm.userLookupSupported()).thenReturn(true);
try { try {
@ -718,7 +732,7 @@ public class AuthenticationServiceTests extends ESTestCase {
threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, ""); threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, "");
when(secondRealm.token(threadContext)).thenReturn(token); when(secondRealm.token(threadContext)).thenReturn(token);
when(secondRealm.supports(token)).thenReturn(true); when(secondRealm.supports(token)).thenReturn(true);
when(secondRealm.authenticate(token)).thenReturn(user); mockAuthenticate(secondRealm, token, user);
when(secondRealm.userLookupSupported()).thenReturn(true); when(secondRealm.userLookupSupported()).thenReturn(true);
try { try {
@ -735,9 +749,12 @@ public class AuthenticationServiceTests extends ESTestCase {
threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, "run_as"); threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, "run_as");
when(secondRealm.token(threadContext)).thenReturn(token); when(secondRealm.token(threadContext)).thenReturn(token);
when(secondRealm.supports(token)).thenReturn(true); when(secondRealm.supports(token)).thenReturn(true);
when(secondRealm.authenticate(token)).thenReturn(new User("lookup user", new String[]{"user"})); mockAuthenticate(secondRealm, token, new User("lookup user", new String[]{"user"}));
when(secondRealm.lookupUser("run_as")) doAnswer((i) -> {
.thenReturn(new User("looked up user", new String[]{"some role"}, null, null, null, false)); ActionListener listener = (ActionListener) i.getArguments()[1];
listener.onResponse(new User("looked up user", new String[]{"some role"}, null, null, null, false));
return null;
}).when(secondRealm).lookupUser(eq("run_as"), any(ActionListener.class));
when(secondRealm.userLookupSupported()).thenReturn(true); when(secondRealm.userLookupSupported()).thenReturn(true);
User fallback = randomBoolean() ? SystemUser.INSTANCE : null; User fallback = randomBoolean() ? SystemUser.INSTANCE : null;
ElasticsearchSecurityException e = ElasticsearchSecurityException e =
@ -752,9 +769,12 @@ public class AuthenticationServiceTests extends ESTestCase {
threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, "run_as"); threadContext.putHeader(AuthenticationService.RUN_AS_USER_HEADER, "run_as");
when(secondRealm.token(threadContext)).thenReturn(token); when(secondRealm.token(threadContext)).thenReturn(token);
when(secondRealm.supports(token)).thenReturn(true); when(secondRealm.supports(token)).thenReturn(true);
when(secondRealm.authenticate(token)).thenReturn(new User("lookup user", new String[]{"user"})); mockAuthenticate(secondRealm, token, new User("lookup user", new String[]{"user"}));
when(secondRealm.lookupUser("run_as")) doAnswer((i) -> {
.thenReturn(new User("looked up user", new String[]{"some role"}, null, null, null, false)); ActionListener listener = (ActionListener) i.getArguments()[1];
listener.onResponse(new User("looked up user", new String[]{"some role"}, null, null, null, false));
return null;
}).when(secondRealm).lookupUser(eq("run_as"), any(ActionListener.class));
when(secondRealm.userLookupSupported()).thenReturn(true); when(secondRealm.userLookupSupported()).thenReturn(true);
ElasticsearchSecurityException e = ElasticsearchSecurityException e =
@ -782,6 +802,14 @@ public class AuthenticationServiceTests extends ESTestCase {
} }
} }
private void mockAuthenticate(Realm realm, AuthenticationToken token, User user) {
doAnswer((i) -> {
ActionListener listener = (ActionListener) i.getArguments()[1];
listener.onResponse(user);
return null;
}).when(realm).authenticate(eq(token), any(ActionListener.class));
}
private Authentication authenticateBlocking(RestRequest restRequest) { private Authentication authenticateBlocking(RestRequest restRequest) {
PlainActionFuture<Authentication> future = new PlainActionFuture<>(); PlainActionFuture<Authentication> future = new PlainActionFuture<>();
service.authenticate(restRequest, future); service.authenticate(restRequest, future);

View File

@ -12,6 +12,7 @@ import com.unboundid.ldap.sdk.LDAPException;
import com.unboundid.ldap.sdk.LDAPURL; import com.unboundid.ldap.sdk.LDAPURL;
import com.unboundid.ldap.sdk.schema.Schema; import com.unboundid.ldap.sdk.schema.Schema;
import org.elasticsearch.action.support.PlainActionFuture;
import org.elasticsearch.common.Strings; import org.elasticsearch.common.Strings;
import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.xpack.security.user.User; import org.elasticsearch.xpack.security.user.User;
@ -116,7 +117,9 @@ public class ActiveDirectoryRealmTests extends ESTestCase {
DnRoleMapper roleMapper = new DnRoleMapper(ActiveDirectoryRealm.TYPE, config, resourceWatcherService, null); DnRoleMapper roleMapper = new DnRoleMapper(ActiveDirectoryRealm.TYPE, config, resourceWatcherService, null);
ActiveDirectoryRealm realm = new ActiveDirectoryRealm(config, sessionFactory, roleMapper); ActiveDirectoryRealm realm = new ActiveDirectoryRealm(config, sessionFactory, roleMapper);
User user = realm.authenticate(new UsernamePasswordToken("CN=ironman", SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("CN=ironman", SecuredStringTests.build(PASSWORD)), future);
User user = future.actionGet();
assertThat(user, is(notNullValue())); assertThat(user, is(notNullValue()));
assertThat(user.roles(), arrayContaining(containsString("Avengers"))); assertThat(user.roles(), arrayContaining(containsString("Avengers")));
} }
@ -129,7 +132,9 @@ public class ActiveDirectoryRealmTests extends ESTestCase {
ActiveDirectoryRealm realm = new ActiveDirectoryRealm(config, sessionFactory, roleMapper); ActiveDirectoryRealm realm = new ActiveDirectoryRealm(config, sessionFactory, roleMapper);
// Thor does not have a UPN of form CN=Thor@ad.test.elasticsearch.com // Thor does not have a UPN of form CN=Thor@ad.test.elasticsearch.com
User user = realm.authenticate(new UsernamePasswordToken("CN=Thor", SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("CN=Thor", SecuredStringTests.build(PASSWORD)), future);
User user = future.actionGet();
assertThat(user, is(notNullValue())); assertThat(user, is(notNullValue()));
assertThat(user.roles(), arrayContaining(containsString("Avengers"))); assertThat(user.roles(), arrayContaining(containsString("Avengers")));
} }
@ -152,7 +157,9 @@ public class ActiveDirectoryRealmTests extends ESTestCase {
int count = randomIntBetween(2, 10); int count = randomIntBetween(2, 10);
for (int i = 0; i < count; i++) { for (int i = 0; i < count; i++) {
realm.authenticate(new UsernamePasswordToken("CN=ironman", SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("CN=ironman", SecuredStringTests.build(PASSWORD)), future);
future.actionGet();
} }
// verify one and only one session as further attempts should be returned from cache // verify one and only one session as further attempts should be returned from cache
@ -168,7 +175,9 @@ public class ActiveDirectoryRealmTests extends ESTestCase {
int count = randomIntBetween(2, 10); int count = randomIntBetween(2, 10);
for (int i = 0; i < count; i++) { for (int i = 0; i < count; i++) {
realm.authenticate(new UsernamePasswordToken("CN=ironman", SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("CN=ironman", SecuredStringTests.build(PASSWORD)), future);
future.actionGet();
} }
// verify one and only one session as second attempt should be returned from cache // verify one and only one session as second attempt should be returned from cache
@ -184,7 +193,9 @@ public class ActiveDirectoryRealmTests extends ESTestCase {
int count = randomIntBetween(2, 10); int count = randomIntBetween(2, 10);
for (int i = 0; i < count; i++) { for (int i = 0; i < count; i++) {
realm.authenticate(new UsernamePasswordToken("CN=ironman", SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("CN=ironman", SecuredStringTests.build(PASSWORD)), future);
future.actionGet();
} }
// verify one and only one session as further attempts should be returned from cache // verify one and only one session as further attempts should be returned from cache
@ -194,7 +205,9 @@ public class ActiveDirectoryRealmTests extends ESTestCase {
roleMapper.notifyRefresh(); roleMapper.notifyRefresh();
for (int i = 0; i < count; i++) { for (int i = 0; i < count; i++) {
realm.authenticate(new UsernamePasswordToken("CN=ironman", SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("CN=ironman", SecuredStringTests.build(PASSWORD)), future);
future.actionGet();
} }
verify(sessionFactory, times(2)).session(eq("CN=ironman"), any(SecuredString.class)); verify(sessionFactory, times(2)).session(eq("CN=ironman"), any(SecuredString.class));
@ -209,7 +222,9 @@ public class ActiveDirectoryRealmTests extends ESTestCase {
DnRoleMapper roleMapper = new DnRoleMapper(ActiveDirectoryRealm.TYPE, config, resourceWatcherService, null); DnRoleMapper roleMapper = new DnRoleMapper(ActiveDirectoryRealm.TYPE, config, resourceWatcherService, null);
ActiveDirectoryRealm realm = new ActiveDirectoryRealm(config, sessionFactory, roleMapper); ActiveDirectoryRealm realm = new ActiveDirectoryRealm(config, sessionFactory, roleMapper);
User user = realm.authenticate(new UsernamePasswordToken("CN=ironman", SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("CN=ironman", SecuredStringTests.build(PASSWORD)), future);
User user = future.actionGet();
assertThat(user, is(notNullValue())); assertThat(user, is(notNullValue()));
assertThat(user.roles(), arrayContaining(equalTo("group_role"))); assertThat(user.roles(), arrayContaining(equalTo("group_role")));
} }
@ -223,7 +238,9 @@ public class ActiveDirectoryRealmTests extends ESTestCase {
DnRoleMapper roleMapper = new DnRoleMapper(ActiveDirectoryRealm.TYPE, config, resourceWatcherService, null); DnRoleMapper roleMapper = new DnRoleMapper(ActiveDirectoryRealm.TYPE, config, resourceWatcherService, null);
ActiveDirectoryRealm realm = new ActiveDirectoryRealm(config, sessionFactory, roleMapper); ActiveDirectoryRealm realm = new ActiveDirectoryRealm(config, sessionFactory, roleMapper);
User user = realm.authenticate(new UsernamePasswordToken("CN=Thor", SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("CN=Thor", SecuredStringTests.build(PASSWORD)), future);
User user = future.actionGet();
assertThat(user, is(notNullValue())); assertThat(user, is(notNullValue()));
assertThat(user.roles(), arrayContainingInAnyOrder(equalTo("group_role"), equalTo("user_role"))); assertThat(user.roles(), arrayContainingInAnyOrder(equalTo("group_role"), equalTo("user_role")));
} }

View File

@ -25,6 +25,7 @@ import org.junit.Before;
import java.util.Collection; import java.util.Collection;
import java.util.Collections; import java.util.Collections;
import java.util.Map; import java.util.Map;
import java.util.Map.Entry;
import static org.hamcrest.Matchers.contains; import static org.hamcrest.Matchers.contains;
import static org.hamcrest.Matchers.containsInAnyOrder; import static org.hamcrest.Matchers.containsInAnyOrder;
@ -33,6 +34,7 @@ import static org.hamcrest.Matchers.empty;
import static org.hamcrest.Matchers.is; import static org.hamcrest.Matchers.is;
import static org.hamcrest.Matchers.nullValue; import static org.hamcrest.Matchers.nullValue;
import static org.mockito.Matchers.any; import static org.mockito.Matchers.any;
import static org.mockito.Matchers.eq;
import static org.mockito.Mockito.doAnswer; import static org.mockito.Mockito.doAnswer;
import static org.mockito.Mockito.mock; import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.times; import static org.mockito.Mockito.times;
@ -62,29 +64,38 @@ public class ReservedRealmTests extends ESTestCase {
new ReservedRealm(mock(Environment.class), Settings.EMPTY, usersStore, new AnonymousUser(Settings.EMPTY)); new ReservedRealm(mock(Environment.class), Settings.EMPTY, usersStore, new AnonymousUser(Settings.EMPTY));
final String principal = randomFrom(ElasticUser.NAME, KibanaUser.NAME); final String principal = randomFrom(ElasticUser.NAME, KibanaUser.NAME);
ElasticsearchSecurityException expected = expectThrows(ElasticsearchSecurityException.class, PlainActionFuture<User> listener = new PlainActionFuture<>();
() -> reservedRealm.doAuthenticate(new UsernamePasswordToken(principal, DEFAULT_PASSWORD))); reservedRealm.doAuthenticate(new UsernamePasswordToken(principal, DEFAULT_PASSWORD), listener);
ElasticsearchSecurityException expected = expectThrows(ElasticsearchSecurityException.class, listener::actionGet);
assertThat(expected.getMessage(), containsString("failed to authenticate user [" + principal)); assertThat(expected.getMessage(), containsString("failed to authenticate user [" + principal));
verify(usersStore).started(); verify(usersStore).started();
verifyNoMoreInteractions(usersStore); verifyNoMoreInteractions(usersStore);
} }
public void testDefaultPasswordAuthentication() throws Throwable { public void testDefaultPasswordAuthentication() throws Throwable {
final User expected = randomFrom(new ElasticUser(true), new KibanaUser(true));
final String principal = expected.principal();
final boolean securityIndexExists = randomBoolean(); final boolean securityIndexExists = randomBoolean();
if (securityIndexExists) { if (securityIndexExists) {
when(usersStore.securityIndexExists()).thenReturn(true); when(usersStore.securityIndexExists()).thenReturn(true);
doAnswer((i) -> {
ActionListener listener = (ActionListener) i.getArguments()[1];
listener.onResponse(null);
return null;
}).when(usersStore).getReservedUserInfo(eq(principal), any(ActionListener.class));
} }
final ReservedRealm reservedRealm = final ReservedRealm reservedRealm =
new ReservedRealm(mock(Environment.class), Settings.EMPTY, usersStore, new AnonymousUser(Settings.EMPTY)); new ReservedRealm(mock(Environment.class), Settings.EMPTY, usersStore, new AnonymousUser(Settings.EMPTY));
final User expected = randomFrom(new ElasticUser(true), new KibanaUser(true));
final String principal = expected.principal();
final User authenticated = reservedRealm.doAuthenticate(new UsernamePasswordToken(principal, DEFAULT_PASSWORD));
PlainActionFuture<User> listener = new PlainActionFuture<>();
reservedRealm.doAuthenticate(new UsernamePasswordToken(principal, DEFAULT_PASSWORD), listener);
final User authenticated = listener.actionGet();
assertEquals(expected, authenticated); assertEquals(expected, authenticated);
verify(usersStore).started(); verify(usersStore).started();
verify(usersStore).securityIndexExists(); verify(usersStore).securityIndexExists();
if (securityIndexExists) { if (securityIndexExists) {
verify(usersStore).getReservedUserInfo(principal); verify(usersStore).getReservedUserInfo(eq(principal), any(ActionListener.class));
} }
verifyNoMoreInteractions(usersStore); verifyNoMoreInteractions(usersStore);
} }
@ -99,7 +110,9 @@ public class ReservedRealmTests extends ESTestCase {
final User expected = randomFrom(new ElasticUser(true), new KibanaUser(true)); final User expected = randomFrom(new ElasticUser(true), new KibanaUser(true));
final String principal = expected.principal(); final String principal = expected.principal();
final User authenticated = reservedRealm.doAuthenticate(new UsernamePasswordToken(principal, DEFAULT_PASSWORD)); PlainActionFuture<User> listener = new PlainActionFuture<>();
reservedRealm.doAuthenticate(new UsernamePasswordToken(principal, DEFAULT_PASSWORD), listener);
final User authenticated = listener.actionGet();
assertNull(authenticated); assertNull(authenticated);
verifyZeroInteractions(usersStore); verifyZeroInteractions(usersStore);
} }
@ -111,22 +124,33 @@ public class ReservedRealmTests extends ESTestCase {
final String principal = expectedUser.principal(); final String principal = expectedUser.principal();
final SecuredString newPassword = new SecuredString("foobar".toCharArray()); final SecuredString newPassword = new SecuredString("foobar".toCharArray());
when(usersStore.securityIndexExists()).thenReturn(true); when(usersStore.securityIndexExists()).thenReturn(true);
when(usersStore.getReservedUserInfo(principal)).thenReturn(new ReservedUserInfo(Hasher.BCRYPT.hash(newPassword), true)); doAnswer((i) -> {
ActionListener callback = (ActionListener) i.getArguments()[1];
callback.onResponse(new ReservedUserInfo(Hasher.BCRYPT.hash(newPassword), true));
return null;
}).when(usersStore).getReservedUserInfo(eq(principal), any(ActionListener.class));
// test default password // test default password
ElasticsearchSecurityException expected = expectThrows(ElasticsearchSecurityException.class, final PlainActionFuture<User> listener = new PlainActionFuture<>();
() -> reservedRealm.doAuthenticate(new UsernamePasswordToken(principal, DEFAULT_PASSWORD))); reservedRealm.doAuthenticate(new UsernamePasswordToken(principal, DEFAULT_PASSWORD), listener);
ElasticsearchSecurityException expected = expectThrows(ElasticsearchSecurityException.class, listener::actionGet);
assertThat(expected.getMessage(), containsString("failed to authenticate user [" + principal)); assertThat(expected.getMessage(), containsString("failed to authenticate user [" + principal));
// the realm assumes it owns the hashed password so it fills it with 0's // the realm assumes it owns the hashed password so it fills it with 0's
when(usersStore.getReservedUserInfo(principal)).thenReturn(new ReservedUserInfo(Hasher.BCRYPT.hash(newPassword), true)); doAnswer((i) -> {
ActionListener callback = (ActionListener) i.getArguments()[1];
callback.onResponse(new ReservedUserInfo(Hasher.BCRYPT.hash(newPassword), true));
return null;
}).when(usersStore).getReservedUserInfo(eq(principal), any(ActionListener.class));
// test new password // test new password
final User authenticated = reservedRealm.doAuthenticate(new UsernamePasswordToken(principal, newPassword)); final PlainActionFuture<User> authListener = new PlainActionFuture<>();
reservedRealm.doAuthenticate(new UsernamePasswordToken(principal, newPassword), authListener);
final User authenticated = authListener.actionGet();
assertEquals(expectedUser, authenticated); assertEquals(expectedUser, authenticated);
verify(usersStore, times(2)).started(); verify(usersStore, times(2)).started();
verify(usersStore, times(2)).securityIndexExists(); verify(usersStore, times(2)).securityIndexExists();
verify(usersStore, times(2)).getReservedUserInfo(principal); verify(usersStore, times(2)).getReservedUserInfo(eq(principal), any(ActionListener.class));
verifyNoMoreInteractions(usersStore); verifyNoMoreInteractions(usersStore);
} }
@ -136,12 +160,16 @@ public class ReservedRealmTests extends ESTestCase {
final User expectedUser = randomFrom(new ElasticUser(true), new KibanaUser(true)); final User expectedUser = randomFrom(new ElasticUser(true), new KibanaUser(true));
final String principal = expectedUser.principal(); final String principal = expectedUser.principal();
final User user = reservedRealm.doLookupUser(principal); PlainActionFuture<User> listener = new PlainActionFuture<>();
reservedRealm.doLookupUser(principal, listener);
final User user = listener.actionGet();
assertEquals(expectedUser, user); assertEquals(expectedUser, user);
verify(usersStore).started(); verify(usersStore).started();
verify(usersStore).securityIndexExists(); verify(usersStore).securityIndexExists();
final User doesntExist = reservedRealm.doLookupUser("foobar"); PlainActionFuture<User> future = new PlainActionFuture<>();
reservedRealm.doLookupUser("foobar", future);
final User doesntExist = future.actionGet();
assertThat(doesntExist, nullValue()); assertThat(doesntExist, nullValue());
verifyNoMoreInteractions(usersStore); verifyNoMoreInteractions(usersStore);
} }
@ -153,7 +181,9 @@ public class ReservedRealmTests extends ESTestCase {
final User expectedUser = randomFrom(new ElasticUser(true), new KibanaUser(true)); final User expectedUser = randomFrom(new ElasticUser(true), new KibanaUser(true));
final String principal = expectedUser.principal(); final String principal = expectedUser.principal();
final User user = reservedRealm.doLookupUser(principal); PlainActionFuture<User> listener = new PlainActionFuture<>();
reservedRealm.doLookupUser(principal, listener);
final User user = listener.actionGet();
assertNull(user); assertNull(user);
verifyZeroInteractions(usersStore); verifyZeroInteractions(usersStore);
} }
@ -165,15 +195,20 @@ public class ReservedRealmTests extends ESTestCase {
final String principal = expectedUser.principal(); final String principal = expectedUser.principal();
when(usersStore.securityIndexExists()).thenReturn(true); when(usersStore.securityIndexExists()).thenReturn(true);
final RuntimeException e = new RuntimeException("store threw"); final RuntimeException e = new RuntimeException("store threw");
when(usersStore.getReservedUserInfo(principal)).thenThrow(e); doAnswer((i) -> {
ActionListener callback = (ActionListener) i.getArguments()[1];
callback.onFailure(e);
return null;
}).when(usersStore).getReservedUserInfo(eq(principal), any(ActionListener.class));
ElasticsearchSecurityException securityException = PlainActionFuture<User> future = new PlainActionFuture<>();
expectThrows(ElasticsearchSecurityException.class, () -> reservedRealm.lookupUser(principal)); reservedRealm.lookupUser(principal, future);
ElasticsearchSecurityException securityException = expectThrows(ElasticsearchSecurityException.class, future::actionGet);
assertThat(securityException.getMessage(), containsString("failed to lookup")); assertThat(securityException.getMessage(), containsString("failed to lookup"));
verify(usersStore).started(); verify(usersStore).started();
verify(usersStore).securityIndexExists(); verify(usersStore).securityIndexExists();
verify(usersStore).getReservedUserInfo(principal); verify(usersStore).getReservedUserInfo(eq(principal), any(ActionListener.class));
verifyNoMoreInteractions(usersStore); verifyNoMoreInteractions(usersStore);
} }
@ -226,18 +261,17 @@ public class ReservedRealmTests extends ESTestCase {
new ReservedRealm(mock(Environment.class), Settings.EMPTY, usersStore, new AnonymousUser(Settings.EMPTY)); new ReservedRealm(mock(Environment.class), Settings.EMPTY, usersStore, new AnonymousUser(Settings.EMPTY));
// maybe cache a successful auth // maybe cache a successful auth
if (randomBoolean()) { if (randomBoolean()) {
User user = reservedRealm.authenticate( PlainActionFuture<User> future = new PlainActionFuture<>();
new UsernamePasswordToken(ElasticUser.NAME, new SecuredString("changeme".toCharArray()))); reservedRealm.authenticate(new UsernamePasswordToken(ElasticUser.NAME, new SecuredString("changeme".toCharArray())), future);
User user = future.actionGet();
assertEquals(new ElasticUser(true), user); assertEquals(new ElasticUser(true), user);
} }
try { PlainActionFuture<User> future = new PlainActionFuture<>();
reservedRealm.authenticate(new UsernamePasswordToken(ElasticUser.NAME, new SecuredString("foobar".toCharArray()))); reservedRealm.authenticate(new UsernamePasswordToken(ElasticUser.NAME, new SecuredString("foobar".toCharArray())), future);
fail("authentication should throw an exception otherwise we may allow others to impersonate reserved users..."); ElasticsearchSecurityException e = expectThrows(ElasticsearchSecurityException.class, future::actionGet);
} catch (ElasticsearchSecurityException e) {
assertThat(e.getMessage(), containsString("failed to authenticate")); assertThat(e.getMessage(), containsString("failed to authenticate"));
} }
}
/* /*
* NativeUserStore#getAllReservedUserInfo is pkg private we can't mock it otherwise * NativeUserStore#getAllReservedUserInfo is pkg private we can't mock it otherwise
@ -247,5 +281,12 @@ public class ReservedRealmTests extends ESTestCase {
((ActionListener) i.getArguments()[0]).onResponse(collection); ((ActionListener) i.getArguments()[0]).onResponse(collection);
return null; return null;
}).when(usersStore).getAllReservedUserInfo(any(ActionListener.class)); }).when(usersStore).getAllReservedUserInfo(any(ActionListener.class));
for (Entry<String, ReservedUserInfo> entry : collection.entrySet()) {
doAnswer((i) -> {
((ActionListener) i.getArguments()[1]).onResponse(entry.getValue());
return null;
}).when(usersStore).getReservedUserInfo(eq(entry.getKey()), any(ActionListener.class));
}
} }
} }

View File

@ -5,6 +5,7 @@
*/ */
package org.elasticsearch.xpack.security.authc.file; package org.elasticsearch.xpack.security.authc.file;
import org.elasticsearch.action.support.PlainActionFuture;
import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.common.util.concurrent.ThreadContext; import org.elasticsearch.common.util.concurrent.ThreadContext;
import org.elasticsearch.xpack.security.authc.RealmConfig; import org.elasticsearch.xpack.security.authc.RealmConfig;
@ -50,7 +51,9 @@ public class FileRealmTests extends ESTestCase {
when(userRolesStore.roles("user1")).thenReturn(new String[] { "role1", "role2" }); when(userRolesStore.roles("user1")).thenReturn(new String[] { "role1", "role2" });
RealmConfig config = new RealmConfig("file-test", Settings.EMPTY, globalSettings); RealmConfig config = new RealmConfig("file-test", Settings.EMPTY, globalSettings);
FileRealm realm = new FileRealm(config, userPasswdStore, userRolesStore); FileRealm realm = new FileRealm(config, userPasswdStore, userRolesStore);
User user = realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123"))); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123")), future);
User user = future.actionGet();
assertThat(user, notNullValue()); assertThat(user, notNullValue());
assertThat(user.principal(), equalTo("user1")); assertThat(user.principal(), equalTo("user1"));
assertThat(user.roles(), notNullValue()); assertThat(user.roles(), notNullValue());
@ -66,8 +69,12 @@ public class FileRealmTests extends ESTestCase {
when(userPasswdStore.verifyPassword("user1", SecuredStringTests.build("test123"))).thenReturn(true); when(userPasswdStore.verifyPassword("user1", SecuredStringTests.build("test123"))).thenReturn(true);
when(userRolesStore.roles("user1")).thenReturn(new String[]{"role1", "role2"}); when(userRolesStore.roles("user1")).thenReturn(new String[]{"role1", "role2"});
FileRealm realm = new FileRealm(config, userPasswdStore, userRolesStore); FileRealm realm = new FileRealm(config, userPasswdStore, userRolesStore);
User user1 = realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123"))); PlainActionFuture<User> future = new PlainActionFuture<>();
User user2 = realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123"))); realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123")), future);
User user1 = future.actionGet();
future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123")), future);
User user2 = future.actionGet();
assertThat(user1, sameInstance(user2)); assertThat(user1, sameInstance(user2));
} }
@ -78,18 +85,34 @@ public class FileRealmTests extends ESTestCase {
doReturn(true).when(userPasswdStore).verifyPassword("user1", SecuredStringTests.build("test123")); doReturn(true).when(userPasswdStore).verifyPassword("user1", SecuredStringTests.build("test123"));
doReturn(new String[] { "role1", "role2" }).when(userRolesStore).roles("user1"); doReturn(new String[] { "role1", "role2" }).when(userRolesStore).roles("user1");
FileRealm realm = new FileRealm(config, userPasswdStore, userRolesStore); FileRealm realm = new FileRealm(config, userPasswdStore, userRolesStore);
User user1 = realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123"))); PlainActionFuture<User> future = new PlainActionFuture<>();
User user2 = realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123"))); realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123")), future);
User user1 = future.actionGet();
future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123")), future);
User user2 = future.actionGet();
assertThat(user1, sameInstance(user2)); assertThat(user1, sameInstance(user2));
userPasswdStore.notifyRefresh(); userPasswdStore.notifyRefresh();
User user3 = realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123")));
future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123")), future);
User user3 = future.actionGet();
assertThat(user2, not(sameInstance(user3))); assertThat(user2, not(sameInstance(user3)));
User user4 = realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123"))); future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123")), future);
User user4 = future.actionGet();
assertThat(user3, sameInstance(user4)); assertThat(user3, sameInstance(user4));
userRolesStore.notifyRefresh(); userRolesStore.notifyRefresh();
User user5 = realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123")));
future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123")), future);
User user5 = future.actionGet();
assertThat(user4, not(sameInstance(user5))); assertThat(user4, not(sameInstance(user5)));
User user6 = realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123"))); future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("user1", SecuredStringTests.build("test123")), future);
User user6 = future.actionGet();
assertThat(user5, sameInstance(user6)); assertThat(user5, sameInstance(user6));
} }
@ -115,7 +138,9 @@ public class FileRealmTests extends ESTestCase {
RealmConfig config = new RealmConfig("file-test", Settings.EMPTY, globalSettings); RealmConfig config = new RealmConfig("file-test", Settings.EMPTY, globalSettings);
FileRealm realm = new FileRealm(config, userPasswdStore, userRolesStore); FileRealm realm = new FileRealm(config, userPasswdStore, userRolesStore);
User user = realm.lookupUser("user1"); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.lookupUser("user1", future);
User user = future.actionGet();
assertThat(user, notNullValue()); assertThat(user, notNullValue());
assertThat(user.principal(), equalTo("user1")); assertThat(user.principal(), equalTo("user1"));
@ -130,8 +155,12 @@ public class FileRealmTests extends ESTestCase {
RealmConfig config = new RealmConfig("file-test", Settings.EMPTY, globalSettings); RealmConfig config = new RealmConfig("file-test", Settings.EMPTY, globalSettings);
FileRealm realm = new FileRealm(config, userPasswdStore, userRolesStore); FileRealm realm = new FileRealm(config, userPasswdStore, userRolesStore);
User user = realm.lookupUser("user1"); PlainActionFuture<User> future = new PlainActionFuture<>();
User user1 = realm.lookupUser("user1"); realm.lookupUser("user1", future);
User user = future.actionGet();
future = new PlainActionFuture<>();
realm.lookupUser("user1", future);
User user1 = future.actionGet();
assertThat(user, sameInstance(user1)); assertThat(user, sameInstance(user1));
verify(userPasswdStore).userExists("user1"); verify(userPasswdStore).userExists("user1");
verify(userRolesStore).roles("user1"); verify(userRolesStore).roles("user1");
@ -144,18 +173,34 @@ public class FileRealmTests extends ESTestCase {
doReturn(true).when(userPasswdStore).userExists("user1"); doReturn(true).when(userPasswdStore).userExists("user1");
doReturn(new String[] { "role1", "role2" }).when(userRolesStore).roles("user1"); doReturn(new String[] { "role1", "role2" }).when(userRolesStore).roles("user1");
FileRealm realm = new FileRealm(config, userPasswdStore, userRolesStore); FileRealm realm = new FileRealm(config, userPasswdStore, userRolesStore);
User user1 = realm.lookupUser("user1"); PlainActionFuture<User> future = new PlainActionFuture<>();
User user2 = realm.lookupUser("user1"); realm.lookupUser("user1", future);
User user1 = future.actionGet();
future = new PlainActionFuture<>();
realm.lookupUser("user1", future);
User user2 = future.actionGet();
assertThat(user1, sameInstance(user2)); assertThat(user1, sameInstance(user2));
userPasswdStore.notifyRefresh(); userPasswdStore.notifyRefresh();
User user3 = realm.lookupUser("user1");
future = new PlainActionFuture<>();
realm.lookupUser("user1", future);
User user3 = future.actionGet();
assertThat(user2, not(sameInstance(user3))); assertThat(user2, not(sameInstance(user3)));
User user4 = realm.lookupUser("user1"); future = new PlainActionFuture<>();
realm.lookupUser("user1", future);
User user4 = future.actionGet();
assertThat(user3, sameInstance(user4)); assertThat(user3, sameInstance(user4));
userRolesStore.notifyRefresh(); userRolesStore.notifyRefresh();
User user5 = realm.lookupUser("user1");
future = new PlainActionFuture<>();
realm.lookupUser("user1", future);
User user5 = future.actionGet();
assertThat(user4, not(sameInstance(user5))); assertThat(user4, not(sameInstance(user5)));
User user6 = realm.lookupUser("user1"); future = new PlainActionFuture<>();
realm.lookupUser("user1", future);
User user6 = future.actionGet();
assertThat(user5, sameInstance(user6)); assertThat(user5, sameInstance(user6));
} }

View File

@ -5,6 +5,7 @@
*/ */
package org.elasticsearch.xpack.security.authc.ldap; package org.elasticsearch.xpack.security.authc.ldap;
import org.elasticsearch.action.support.PlainActionFuture;
import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.xpack.security.authc.RealmConfig; import org.elasticsearch.xpack.security.authc.RealmConfig;
import org.elasticsearch.xpack.security.authc.ldap.support.LdapSearchScope; import org.elasticsearch.xpack.security.authc.ldap.support.LdapSearchScope;
@ -69,7 +70,9 @@ public class LdapRealmTests extends LdapTestCase {
LdapSessionFactory ldapFactory = new LdapSessionFactory(config, null); LdapSessionFactory ldapFactory = new LdapSessionFactory(config, null);
LdapRealm ldap = new LdapRealm(config, ldapFactory, buildGroupAsRoleMapper(resourceWatcherService)); LdapRealm ldap = new LdapRealm(config, ldapFactory, buildGroupAsRoleMapper(resourceWatcherService));
User user = ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD)), future);
User user = future.actionGet();
assertThat(user, notNullValue()); assertThat(user, notNullValue());
assertThat(user.roles(), arrayContaining("HMS Victory")); assertThat(user.roles(), arrayContaining("HMS Victory"));
} }
@ -85,7 +88,9 @@ public class LdapRealmTests extends LdapTestCase {
LdapSessionFactory ldapFactory = new LdapSessionFactory(config, null); LdapSessionFactory ldapFactory = new LdapSessionFactory(config, null);
LdapRealm ldap = new LdapRealm(config, ldapFactory, buildGroupAsRoleMapper(resourceWatcherService)); LdapRealm ldap = new LdapRealm(config, ldapFactory, buildGroupAsRoleMapper(resourceWatcherService));
User user = ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD)), future);
User user = future.actionGet();
assertThat(user, notNullValue()); assertThat(user, notNullValue());
assertThat(user.roles(), arrayContaining("HMS Victory")); assertThat(user.roles(), arrayContaining("HMS Victory"));
} }
@ -101,8 +106,12 @@ public class LdapRealmTests extends LdapTestCase {
LdapSessionFactory ldapFactory = new LdapSessionFactory(config, null); LdapSessionFactory ldapFactory = new LdapSessionFactory(config, null);
ldapFactory = spy(ldapFactory); ldapFactory = spy(ldapFactory);
LdapRealm ldap = new LdapRealm(config, ldapFactory, buildGroupAsRoleMapper(resourceWatcherService)); LdapRealm ldap = new LdapRealm(config, ldapFactory, buildGroupAsRoleMapper(resourceWatcherService));
ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
ldap.authenticate( new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD))); ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD)), future);
future.actionGet();
future = new PlainActionFuture<>();
ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD)), future);
future.actionGet();
//verify one and only one session -> caching is working //verify one and only one session -> caching is working
verify(ldapFactory, times(1)).session(anyString(), any(SecuredString.class)); verify(ldapFactory, times(1)).session(anyString(), any(SecuredString.class));
@ -120,15 +129,21 @@ public class LdapRealmTests extends LdapTestCase {
DnRoleMapper roleMapper = buildGroupAsRoleMapper(resourceWatcherService); DnRoleMapper roleMapper = buildGroupAsRoleMapper(resourceWatcherService);
ldapFactory = spy(ldapFactory); ldapFactory = spy(ldapFactory);
LdapRealm ldap = new LdapRealm(config, ldapFactory, roleMapper); LdapRealm ldap = new LdapRealm(config, ldapFactory, roleMapper);
ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD))); ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD)), future);
future.actionGet();
future = new PlainActionFuture<>();
ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD)), future);
future.actionGet();
//verify one and only one session -> caching is working //verify one and only one session -> caching is working
verify(ldapFactory, times(1)).session(anyString(), any(SecuredString.class)); verify(ldapFactory, times(1)).session(anyString(), any(SecuredString.class));
roleMapper.notifyRefresh(); roleMapper.notifyRefresh();
ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD))); future = new PlainActionFuture<>();
ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD)), future);
future.actionGet();
//we need to session again //we need to session again
verify(ldapFactory, times(2)).session(anyString(), any(SecuredString.class)); verify(ldapFactory, times(2)).session(anyString(), any(SecuredString.class));
@ -146,8 +161,12 @@ public class LdapRealmTests extends LdapTestCase {
LdapSessionFactory ldapFactory = new LdapSessionFactory(config, null); LdapSessionFactory ldapFactory = new LdapSessionFactory(config, null);
ldapFactory = spy(ldapFactory); ldapFactory = spy(ldapFactory);
LdapRealm ldap = new LdapRealm(config, ldapFactory, buildGroupAsRoleMapper(resourceWatcherService)); LdapRealm ldap = new LdapRealm(config, ldapFactory, buildGroupAsRoleMapper(resourceWatcherService));
ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD))); ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD)), future);
future.actionGet();
future = new PlainActionFuture<>();
ldap.authenticate(new UsernamePasswordToken(VALID_USERNAME, SecuredStringTests.build(PASSWORD)), future);
future.actionGet();
//verify two and only two binds -> caching is disabled //verify two and only two binds -> caching is disabled
verify(ldapFactory, times(2)).session(anyString(), any(SecuredString.class)); verify(ldapFactory, times(2)).session(anyString(), any(SecuredString.class));
@ -219,7 +238,9 @@ public class LdapRealmTests extends LdapTestCase {
LdapSessionFactory ldapFactory = new LdapSessionFactory(config, null); LdapSessionFactory ldapFactory = new LdapSessionFactory(config, null);
LdapRealm ldap = new LdapRealm(config, ldapFactory, new DnRoleMapper(LdapRealm.TYPE, config, resourceWatcherService, null)); LdapRealm ldap = new LdapRealm(config, ldapFactory, new DnRoleMapper(LdapRealm.TYPE, config, resourceWatcherService, null));
User user = ldap.authenticate(new UsernamePasswordToken("Horatio Hornblower", SecuredStringTests.build(PASSWORD))); PlainActionFuture<User> future = new PlainActionFuture<>();
ldap.authenticate(new UsernamePasswordToken("Horatio Hornblower", SecuredStringTests.build(PASSWORD)), future);
User user = future.actionGet();
assertThat(user, notNullValue()); assertThat(user, notNullValue());
assertThat(user.roles(), arrayContaining("avenger")); assertThat(user.roles(), arrayContaining("avenger"));
} }

View File

@ -5,6 +5,7 @@
*/ */
package org.elasticsearch.xpack.security.authc.pki; package org.elasticsearch.xpack.security.authc.pki;
import org.elasticsearch.action.support.PlainActionFuture;
import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.common.util.concurrent.ThreadContext; import org.elasticsearch.common.util.concurrent.ThreadContext;
import org.elasticsearch.env.Environment; import org.elasticsearch.env.Environment;
@ -83,7 +84,9 @@ public class PkiRealmTests extends ESTestCase {
PkiRealm realm = new PkiRealm(new RealmConfig("", Settings.EMPTY, globalSettings), roleMapper, sslService); PkiRealm realm = new PkiRealm(new RealmConfig("", Settings.EMPTY, globalSettings), roleMapper, sslService);
when(roleMapper.resolveRoles(anyString(), anyList())).thenReturn(Collections.<String>emptySet()); when(roleMapper.resolveRoles(anyString(), anyList())).thenReturn(Collections.<String>emptySet());
User user = realm.authenticate(token); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(token, future);
User user = future.actionGet();
assertThat(user, is(notNullValue())); assertThat(user, is(notNullValue()));
assertThat(user.principal(), is("Elasticsearch Test Node")); assertThat(user.principal(), is("Elasticsearch Test Node"));
assertThat(user.roles(), is(notNullValue())); assertThat(user.roles(), is(notNullValue()));
@ -100,7 +103,9 @@ public class PkiRealmTests extends ESTestCase {
threadContext.putTransient(PkiRealm.PKI_CERT_HEADER_NAME, new X509Certificate[] { certificate }); threadContext.putTransient(PkiRealm.PKI_CERT_HEADER_NAME, new X509Certificate[] { certificate });
X509AuthenticationToken token = realm.token(threadContext); X509AuthenticationToken token = realm.token(threadContext);
User user = realm.authenticate(token); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(token, future);
User user = future.actionGet();
assertThat(user, is(notNullValue())); assertThat(user, is(notNullValue()));
assertThat(user.principal(), is("elasticsearch")); assertThat(user.principal(), is("elasticsearch"));
assertThat(user.roles(), is(notNullValue())); assertThat(user.roles(), is(notNullValue()));
@ -121,7 +126,9 @@ public class PkiRealmTests extends ESTestCase {
threadContext.putTransient(PkiRealm.PKI_CERT_HEADER_NAME, new X509Certificate[] { certificate }); threadContext.putTransient(PkiRealm.PKI_CERT_HEADER_NAME, new X509Certificate[] { certificate });
X509AuthenticationToken token = realm.token(threadContext); X509AuthenticationToken token = realm.token(threadContext);
User user = realm.authenticate(token); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(token, future);
User user = future.actionGet();
assertThat(user, is(notNullValue())); assertThat(user, is(notNullValue()));
assertThat(user.principal(), is("Elasticsearch Test Node")); assertThat(user.principal(), is("Elasticsearch Test Node"));
assertThat(user.roles(), is(notNullValue())); assertThat(user.roles(), is(notNullValue()));
@ -143,7 +150,9 @@ public class PkiRealmTests extends ESTestCase {
threadContext.putTransient(PkiRealm.PKI_CERT_HEADER_NAME, new X509Certificate[] { certificate }); threadContext.putTransient(PkiRealm.PKI_CERT_HEADER_NAME, new X509Certificate[] { certificate });
X509AuthenticationToken token = realm.token(threadContext); X509AuthenticationToken token = realm.token(threadContext);
User user = realm.authenticate(token); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(token, future);
User user = future.actionGet();
assertThat(user, is(nullValue())); assertThat(user, is(nullValue()));
} }

View File

@ -5,6 +5,8 @@
*/ */
package org.elasticsearch.xpack.security.authc.support; package org.elasticsearch.xpack.security.authc.support;
import org.elasticsearch.action.ActionListener;
import org.elasticsearch.action.support.PlainActionFuture;
import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.common.unit.TimeValue; import org.elasticsearch.common.unit.TimeValue;
import org.elasticsearch.xpack.security.authc.Realm; import org.elasticsearch.xpack.security.authc.Realm;
@ -19,6 +21,7 @@ import java.util.concurrent.CountDownLatch;
import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicInteger;
import static org.hamcrest.Matchers.arrayContaining; import static org.hamcrest.Matchers.arrayContaining;
import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.is; import static org.hamcrest.Matchers.is;
import static org.hamcrest.Matchers.not; import static org.hamcrest.Matchers.not;
import static org.hamcrest.Matchers.nullValue; import static org.hamcrest.Matchers.nullValue;
@ -47,13 +50,13 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
RealmConfig config = new RealmConfig("test_realm", settings, globalSettings); RealmConfig config = new RealmConfig("test_realm", settings, globalSettings);
CachingUsernamePasswordRealm realm = new CachingUsernamePasswordRealm("test", config) { CachingUsernamePasswordRealm realm = new CachingUsernamePasswordRealm("test", config) {
@Override @Override
protected User doAuthenticate(UsernamePasswordToken token) { protected void doAuthenticate(UsernamePasswordToken token, ActionListener<User> listener) {
return new User("username", new String[] { "r1", "r2", "r3" }); listener.onResponse(new User("username", new String[] { "r1", "r2", "r3" }));
} }
@Override @Override
protected User doLookupUser(String username) { protected void doLookupUser(String username, ActionListener<User> listener) {
throw new UnsupportedOperationException("this method should not be called"); listener.onFailure(new UnsupportedOperationException("this method should not be called"));
} }
@Override @Override
@ -68,14 +71,27 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
public void testAuthCache() { public void testAuthCache() {
AlwaysAuthenticateCachingRealm realm = new AlwaysAuthenticateCachingRealm(globalSettings); AlwaysAuthenticateCachingRealm realm = new AlwaysAuthenticateCachingRealm(globalSettings);
SecuredString pass = SecuredStringTests.build("pass"); SecuredString pass = SecuredStringTests.build("pass");
realm.authenticate(new UsernamePasswordToken("a", pass)); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("b", pass)); realm.authenticate(new UsernamePasswordToken("a", pass), future);
realm.authenticate(new UsernamePasswordToken("c", pass)); future.actionGet();
future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("b", pass), future);
future.actionGet();
future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("c", pass), future);
future.actionGet();
assertThat(realm.authInvocationCounter.intValue(), is(3)); assertThat(realm.authInvocationCounter.intValue(), is(3));
realm.authenticate(new UsernamePasswordToken("a", pass));
realm.authenticate(new UsernamePasswordToken("b", pass)); future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("c", pass)); realm.authenticate(new UsernamePasswordToken("a", pass), future);
future.actionGet();
future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("b", pass), future);
future.actionGet();
future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("c", pass), future);
future.actionGet();
assertThat(realm.authInvocationCounter.intValue(), is(3)); assertThat(realm.authInvocationCounter.intValue(), is(3));
assertThat(realm.lookupInvocationCounter.intValue(), is(0)); assertThat(realm.lookupInvocationCounter.intValue(), is(0));
@ -83,14 +99,26 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
public void testLookupCache() { public void testLookupCache() {
AlwaysAuthenticateCachingRealm realm = new AlwaysAuthenticateCachingRealm(globalSettings); AlwaysAuthenticateCachingRealm realm = new AlwaysAuthenticateCachingRealm(globalSettings);
realm.lookupUser("a"); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.lookupUser("b"); realm.lookupUser("a", future);
realm.lookupUser("c"); future.actionGet();
future = new PlainActionFuture<>();
realm.lookupUser("b", future);
future.actionGet();
future = new PlainActionFuture<>();
realm.lookupUser("c", future);
future.actionGet();
assertThat(realm.lookupInvocationCounter.intValue(), is(3)); assertThat(realm.lookupInvocationCounter.intValue(), is(3));
realm.lookupUser("a"); future = new PlainActionFuture<>();
realm.lookupUser("b"); realm.lookupUser("a", future);
realm.lookupUser("c"); future.actionGet();
future = new PlainActionFuture<>();
realm.lookupUser("b", future);
future.actionGet();
future = new PlainActionFuture<>();
realm.lookupUser("c", future);
future.actionGet();
assertThat(realm.authInvocationCounter.intValue(), is(0)); assertThat(realm.authInvocationCounter.intValue(), is(0));
assertThat(realm.lookupInvocationCounter.intValue(), is(3)); assertThat(realm.lookupInvocationCounter.intValue(), is(3));
@ -99,25 +127,33 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
public void testLookupAndAuthCache() { public void testLookupAndAuthCache() {
AlwaysAuthenticateCachingRealm realm = new AlwaysAuthenticateCachingRealm(globalSettings); AlwaysAuthenticateCachingRealm realm = new AlwaysAuthenticateCachingRealm(globalSettings);
// lookup first // lookup first
User lookedUp = realm.lookupUser("a"); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.lookupUser("a", future);
User lookedUp = future.actionGet();
assertThat(realm.lookupInvocationCounter.intValue(), is(1)); assertThat(realm.lookupInvocationCounter.intValue(), is(1));
assertThat(realm.authInvocationCounter.intValue(), is(0)); assertThat(realm.authInvocationCounter.intValue(), is(0));
assertThat(lookedUp.roles(), arrayContaining("lookupRole1", "lookupRole2")); assertThat(lookedUp.roles(), arrayContaining("lookupRole1", "lookupRole2"));
// now authenticate // now authenticate
User user = realm.authenticate(new UsernamePasswordToken("a", SecuredStringTests.build("pass"))); future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("a", SecuredStringTests.build("pass")), future);
User user = future.actionGet();
assertThat(realm.lookupInvocationCounter.intValue(), is(1)); assertThat(realm.lookupInvocationCounter.intValue(), is(1));
assertThat(realm.authInvocationCounter.intValue(), is(1)); assertThat(realm.authInvocationCounter.intValue(), is(1));
assertThat(user.roles(), arrayContaining("testRole1", "testRole2")); assertThat(user.roles(), arrayContaining("testRole1", "testRole2"));
assertThat(user, not(sameInstance(lookedUp))); assertThat(user, not(sameInstance(lookedUp)));
// authenticate a different user first // authenticate a different user first
user = realm.authenticate(new UsernamePasswordToken("b", SecuredStringTests.build("pass"))); future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("b", SecuredStringTests.build("pass")), future);
user = future.actionGet();
assertThat(realm.lookupInvocationCounter.intValue(), is(1)); assertThat(realm.lookupInvocationCounter.intValue(), is(1));
assertThat(realm.authInvocationCounter.intValue(), is(2)); assertThat(realm.authInvocationCounter.intValue(), is(2));
assertThat(user.roles(), arrayContaining("testRole1", "testRole2")); assertThat(user.roles(), arrayContaining("testRole1", "testRole2"));
//now lookup b //now lookup b
lookedUp = realm.lookupUser("b"); future = new PlainActionFuture<>();
realm.lookupUser("b", future);
lookedUp = future.actionGet();
assertThat(realm.lookupInvocationCounter.intValue(), is(1)); assertThat(realm.lookupInvocationCounter.intValue(), is(1));
assertThat(realm.authInvocationCounter.intValue(), is(2)); assertThat(realm.authInvocationCounter.intValue(), is(2));
assertThat(user, sameInstance(lookedUp)); assertThat(user, sameInstance(lookedUp));
@ -130,47 +166,69 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
SecuredString pass1 = SecuredStringTests.build("pass"); SecuredString pass1 = SecuredStringTests.build("pass");
SecuredString pass2 = SecuredStringTests.build("password"); SecuredString pass2 = SecuredStringTests.build("password");
realm.authenticate(new UsernamePasswordToken(user, pass1)); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken(user, pass1)); realm.authenticate(new UsernamePasswordToken(user, pass1), future);
future.actionGet();
future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken(user, pass1), future);
future.actionGet();
assertThat(realm.authInvocationCounter.intValue(), is(1)); assertThat(realm.authInvocationCounter.intValue(), is(1));
realm.authenticate(new UsernamePasswordToken(user, pass2)); future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken(user, pass2)); realm.authenticate(new UsernamePasswordToken(user, pass2), future);
future.actionGet();
future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken(user, pass2), future);
future.actionGet();
assertThat(realm.authInvocationCounter.intValue(), is(2)); assertThat(realm.authInvocationCounter.intValue(), is(2));
} }
public void testAuthenticateContract() throws Exception { public void testAuthenticateContract() throws Exception {
Realm realm = new FailingAuthenticationRealm(Settings.EMPTY, globalSettings); Realm realm = new FailingAuthenticationRealm(Settings.EMPTY, globalSettings);
User user = realm.authenticate(new UsernamePasswordToken("user", SecuredStringTests.build("pass"))); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.authenticate(new UsernamePasswordToken("user", SecuredStringTests.build("pass")), future);
User user = future.actionGet();
assertThat(user , nullValue()); assertThat(user , nullValue());
realm = new ThrowingAuthenticationRealm(Settings.EMPTY, globalSettings); realm = new ThrowingAuthenticationRealm(Settings.EMPTY, globalSettings);
user = realm.authenticate(new UsernamePasswordToken("user", SecuredStringTests.build("pass"))); future = new PlainActionFuture<>();
assertThat(user , nullValue()); realm.authenticate(new UsernamePasswordToken("user", SecuredStringTests.build("pass")), future);
RuntimeException e = expectThrows(RuntimeException.class, future::actionGet);
assertThat(e.getMessage() , containsString("whatever exception"));
} }
public void testLookupContract() throws Exception { public void testLookupContract() throws Exception {
Realm realm = new FailingAuthenticationRealm(Settings.EMPTY, globalSettings); Realm realm = new FailingAuthenticationRealm(Settings.EMPTY, globalSettings);
User user = realm.lookupUser("user"); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.lookupUser("user", future);
User user = future.actionGet();
assertThat(user , nullValue()); assertThat(user , nullValue());
realm = new ThrowingAuthenticationRealm(Settings.EMPTY, globalSettings); realm = new ThrowingAuthenticationRealm(Settings.EMPTY, globalSettings);
user = realm.lookupUser("user"); future = new PlainActionFuture<>();
assertThat(user , nullValue()); realm.lookupUser("user", future);
RuntimeException e = expectThrows(RuntimeException.class, future::actionGet);
assertThat(e.getMessage() , containsString("lookup exception"));
} }
public void testThatLookupIsNotCalledIfNotSupported() throws Exception { public void testThatLookupIsNotCalledIfNotSupported() throws Exception {
LookupNotSupportedRealm realm = new LookupNotSupportedRealm(globalSettings); LookupNotSupportedRealm realm = new LookupNotSupportedRealm(globalSettings);
assertThat(realm.userLookupSupported(), is(false)); assertThat(realm.userLookupSupported(), is(false));
User user = realm.lookupUser("a"); PlainActionFuture<User> future = new PlainActionFuture<>();
realm.lookupUser("a", future);
User user = future.actionGet();
assertThat(user, is(nullValue())); assertThat(user, is(nullValue()));
assertThat(realm.lookupInvocationCounter.intValue(), is(0)); assertThat(realm.lookupInvocationCounter.intValue(), is(0));
// try to lookup more // try to lookup more
realm.lookupUser("b"); future = new PlainActionFuture<>();
realm.lookupUser("c"); realm.lookupUser("b", future);
future.actionGet();
future = new PlainActionFuture<>();
realm.lookupUser("c", future);
future.actionGet();
assertThat(realm.lookupInvocationCounter.intValue(), is(0)); assertThat(realm.lookupInvocationCounter.intValue(), is(0));
} }
@ -184,17 +242,18 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
RealmConfig config = new RealmConfig("test_realm", Settings.EMPTY, globalSettings); RealmConfig config = new RealmConfig("test_realm", Settings.EMPTY, globalSettings);
final CachingUsernamePasswordRealm realm = new CachingUsernamePasswordRealm("test", config) { final CachingUsernamePasswordRealm realm = new CachingUsernamePasswordRealm("test", config) {
@Override @Override
protected User doAuthenticate(UsernamePasswordToken token) { protected void doAuthenticate(UsernamePasswordToken token, ActionListener<User> listener) {
// do something slow // do something slow
if (BCrypt.checkpw(token.credentials(), passwordHash)) { if (BCrypt.checkpw(token.credentials(), passwordHash)) {
return new User(username, new String[]{"r1", "r2", "r3"}); listener.onResponse(new User(username, new String[]{"r1", "r2", "r3"}));
} else {
listener.onResponse(null);
} }
return null;
} }
@Override @Override
protected User doLookupUser(String username) { protected void doLookupUser(String username, ActionListener<User> listener) {
throw new UnsupportedOperationException("this method should not be called"); listener.onFailure(new UnsupportedOperationException("this method should not be called"));
} }
@Override @Override
@ -217,12 +276,17 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
latch.await(); latch.await();
for (int i = 0; i < numberOfIterations; i++) { for (int i = 0; i < numberOfIterations; i++) {
UsernamePasswordToken token = new UsernamePasswordToken(username, invalidPassword ? randomPassword : password); UsernamePasswordToken token = new UsernamePasswordToken(username, invalidPassword ? randomPassword : password);
User user = realm.authenticate(token);
realm.authenticate(token, ActionListener.wrap((user) -> {
if (invalidPassword && user != null) { if (invalidPassword && user != null) {
throw new RuntimeException("invalid password led to an authenticated user: " + user.toString()); throw new RuntimeException("invalid password led to an authenticated user: " + user.toString());
} else if (invalidPassword == false && user == null) { } else if (invalidPassword == false && user == null) {
throw new RuntimeException("proper password led to a null user!"); throw new RuntimeException("proper password led to a null user!");
} }
}, (e) -> {
logger.error("caught exception", e);
fail("unexpected exception");
}));
} }
} catch (InterruptedException e) {} } catch (InterruptedException e) {}
@ -245,13 +309,13 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
RealmConfig config = new RealmConfig("test_realm", Settings.EMPTY, globalSettings); RealmConfig config = new RealmConfig("test_realm", Settings.EMPTY, globalSettings);
final CachingUsernamePasswordRealm realm = new CachingUsernamePasswordRealm("test", config) { final CachingUsernamePasswordRealm realm = new CachingUsernamePasswordRealm("test", config) {
@Override @Override
protected User doAuthenticate(UsernamePasswordToken token) { protected void doAuthenticate(UsernamePasswordToken token, ActionListener<User> listener) {
throw new UnsupportedOperationException("authenticate should not be called!"); listener.onFailure(new UnsupportedOperationException("authenticate should not be called!"));
} }
@Override @Override
protected User doLookupUser(String username) { protected void doLookupUser(String username, ActionListener<User> listener) {
return new User(username, new String[]{"r1", "r2", "r3"}); listener.onResponse(new User(username, new String[]{"r1", "r2", "r3"}));
} }
@Override @Override
@ -272,10 +336,14 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
try { try {
latch.await(); latch.await();
for (int i = 0; i < numberOfIterations; i++) { for (int i = 0; i < numberOfIterations; i++) {
User user = realm.lookupUser(username); realm.lookupUser(username, ActionListener.wrap((user) -> {
if (user == null) { if (user == null) {
throw new RuntimeException("failed to lookup user"); throw new RuntimeException("failed to lookup user");
} }
}, (e) -> {
logger.error("caught exception", e);
fail("unexpected exception");
}));
} }
} catch (InterruptedException e) {} } catch (InterruptedException e) {}
@ -298,20 +366,20 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
super("failing", new RealmConfig("failing-test", settings, global)); super("failing", new RealmConfig("failing-test", settings, global));
} }
@Override
protected User doAuthenticate(UsernamePasswordToken token) {
return null;
}
@Override
protected User doLookupUser(String username) {
return null;
}
@Override @Override
public boolean userLookupSupported() { public boolean userLookupSupported() {
return true; return true;
} }
@Override
protected void doAuthenticate(UsernamePasswordToken token, ActionListener<User> listener) {
listener.onResponse(null);
}
@Override
protected void doLookupUser(String username, ActionListener<User> listener) {
listener.onResponse(null);
}
} }
static class ThrowingAuthenticationRealm extends CachingUsernamePasswordRealm { static class ThrowingAuthenticationRealm extends CachingUsernamePasswordRealm {
@ -321,13 +389,13 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
} }
@Override @Override
protected User doAuthenticate(UsernamePasswordToken token) { protected void doAuthenticate(UsernamePasswordToken token, ActionListener<User> listener) {
throw new RuntimeException("whatever exception"); listener.onFailure(new RuntimeException("whatever exception"));
} }
@Override @Override
protected User doLookupUser(String username) { protected void doLookupUser(String username, ActionListener<User> listener) {
throw new RuntimeException("lookup exception"); listener.onFailure(new RuntimeException("lookup exception"));
} }
@Override @Override
@ -346,15 +414,15 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
} }
@Override @Override
protected User doAuthenticate(UsernamePasswordToken token) { protected void doAuthenticate(UsernamePasswordToken token, ActionListener<User> listener) {
authInvocationCounter.incrementAndGet(); authInvocationCounter.incrementAndGet();
return new User(token.principal(), new String[] { "testRole1", "testRole2" }); listener.onResponse(new User(token.principal(), new String[] { "testRole1", "testRole2" }));
} }
@Override @Override
protected User doLookupUser(String username) { protected void doLookupUser(String username, ActionListener<User> listener) {
lookupInvocationCounter.incrementAndGet(); lookupInvocationCounter.incrementAndGet();
return new User(username, new String[] { "lookupRole1", "lookupRole2" }); listener.onResponse(new User(username, new String[] { "lookupRole1", "lookupRole2" }));
} }
@Override @Override
@ -373,15 +441,15 @@ public class CachingUsernamePasswordRealmTests extends ESTestCase {
} }
@Override @Override
protected User doAuthenticate(UsernamePasswordToken token) { protected void doAuthenticate(UsernamePasswordToken token, ActionListener<User> listener) {
authInvocationCounter.incrementAndGet(); authInvocationCounter.incrementAndGet();
return new User(token.principal(), new String[] { "testRole1", "testRole2" }); listener.onResponse(new User(token.principal(), new String[] { "testRole1", "testRole2" }));
} }
@Override @Override
protected User doLookupUser(String username) { protected void doLookupUser(String username, ActionListener<User> listener) {
lookupInvocationCounter.incrementAndGet(); lookupInvocationCounter.incrementAndGet();
throw new UnsupportedOperationException("don't call lookup if lookup isn't supported!!!"); listener.onFailure(new UnsupportedOperationException("don't call lookup if lookup isn't supported!!!"));
} }
@Override @Override