All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.elasticsearch.xpack.security.authc.TokenService Maven / Gradle / Ivy

There is a newer version: 8.17.0
Show newest version
/*
 * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
 * or more contributor license agreements. Licensed under the Elastic License
 * 2.0; you may not use this file except in compliance with the Elastic License
 * 2.0.
 */

package org.elasticsearch.xpack.security.authc;

import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.apache.lucene.util.BytesRef;
import org.apache.lucene.util.BytesRefBuilder;
import org.apache.lucene.util.UnicodeUtil;
import org.elasticsearch.ElasticsearchException;
import org.elasticsearch.ElasticsearchSecurityException;
import org.elasticsearch.ExceptionsHelper;
import org.elasticsearch.TransportVersion;
import org.elasticsearch.TransportVersions;
import org.elasticsearch.action.ActionListener;
import org.elasticsearch.action.DocWriteRequest.OpType;
import org.elasticsearch.action.DocWriteResponse;
import org.elasticsearch.action.DocWriteResponse.Result;
import org.elasticsearch.action.bulk.BackoffPolicy;
import org.elasticsearch.action.bulk.BulkItemResponse;
import org.elasticsearch.action.bulk.BulkRequestBuilder;
import org.elasticsearch.action.bulk.BulkResponse;
import org.elasticsearch.action.get.GetRequest;
import org.elasticsearch.action.get.GetResponse;
import org.elasticsearch.action.index.IndexRequest;
import org.elasticsearch.action.index.TransportIndexAction;
import org.elasticsearch.action.search.SearchRequest;
import org.elasticsearch.action.search.SearchResponse;
import org.elasticsearch.action.support.ContextPreservingActionListener;
import org.elasticsearch.action.support.TransportActions;
import org.elasticsearch.action.support.WriteRequest.RefreshPolicy;
import org.elasticsearch.action.update.UpdateRequest;
import org.elasticsearch.action.update.UpdateRequestBuilder;
import org.elasticsearch.action.update.UpdateResponse;
import org.elasticsearch.client.internal.Client;
import org.elasticsearch.cluster.ClusterState;
import org.elasticsearch.cluster.ClusterStateUpdateTask;
import org.elasticsearch.cluster.service.ClusterService;
import org.elasticsearch.common.Priority;
import org.elasticsearch.common.Strings;
import org.elasticsearch.common.bytes.BytesReference;
import org.elasticsearch.common.cache.Cache;
import org.elasticsearch.common.cache.CacheBuilder;
import org.elasticsearch.common.io.stream.BytesStreamOutput;
import org.elasticsearch.common.io.stream.InputStreamStreamInput;
import org.elasticsearch.common.io.stream.OutputStreamStreamOutput;
import org.elasticsearch.common.io.stream.StreamInput;
import org.elasticsearch.common.io.stream.StreamOutput;
import org.elasticsearch.common.settings.SecureString;
import org.elasticsearch.common.settings.Setting;
import org.elasticsearch.common.settings.Setting.Property;
import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.common.util.Maps;
import org.elasticsearch.common.util.concurrent.AbstractRunnable;
import org.elasticsearch.common.util.concurrent.ThreadContext;
import org.elasticsearch.core.Nullable;
import org.elasticsearch.core.Streams;
import org.elasticsearch.core.SuppressForbidden;
import org.elasticsearch.core.TimeValue;
import org.elasticsearch.core.Tuple;
import org.elasticsearch.index.IndexNotFoundException;
import org.elasticsearch.index.engine.VersionConflictEngineException;
import org.elasticsearch.index.query.BoolQueryBuilder;
import org.elasticsearch.index.query.QueryBuilders;
import org.elasticsearch.index.seqno.SequenceNumbers;
import org.elasticsearch.indices.IndexClosedException;
import org.elasticsearch.license.LicenseUtils;
import org.elasticsearch.license.XPackLicenseState;
import org.elasticsearch.rest.RestStatus;
import org.elasticsearch.search.SearchHit;
import org.elasticsearch.xcontent.XContentBuilder;
import org.elasticsearch.xcontent.XContentFactory;
import org.elasticsearch.xcontent.XContentType;
import org.elasticsearch.xpack.core.XPackField;
import org.elasticsearch.xpack.core.XPackPlugin;
import org.elasticsearch.xpack.core.XPackSettings;
import org.elasticsearch.xpack.core.security.ScrollHelper;
import org.elasticsearch.xpack.core.security.SecurityContext;
import org.elasticsearch.xpack.core.security.authc.Authentication;
import org.elasticsearch.xpack.core.security.authc.KeyAndTimestamp;
import org.elasticsearch.xpack.core.security.authc.TokenMetadata;
import org.elasticsearch.xpack.core.security.authc.support.AuthenticationContextSerializer;
import org.elasticsearch.xpack.core.security.authc.support.Hasher;
import org.elasticsearch.xpack.core.security.authc.support.TokensInvalidationResult;
import org.elasticsearch.xpack.security.Security;
import org.elasticsearch.xpack.security.support.FeatureNotEnabledException;
import org.elasticsearch.xpack.security.support.FeatureNotEnabledException.Feature;
import org.elasticsearch.xpack.security.support.SecurityIndexManager;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.Closeable;
import java.io.IOException;
import java.io.OutputStream;
import java.io.UncheckedIOException;
import java.nio.ByteBuffer;
import java.nio.charset.StandardCharsets;
import java.security.GeneralSecurityException;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.security.SecureRandom;
import java.security.spec.InvalidKeySpecException;
import java.time.Clock;
import java.time.DateTimeException;
import java.time.Instant;
import java.time.ZoneOffset;
import java.time.temporal.ChronoUnit;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Base64;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;
import java.util.function.Consumer;
import java.util.function.Predicate;
import java.util.function.Supplier;

import javax.crypto.Cipher;
import javax.crypto.CipherInputStream;
import javax.crypto.CipherOutputStream;
import javax.crypto.NoSuchPaddingException;
import javax.crypto.SecretKey;
import javax.crypto.SecretKeyFactory;
import javax.crypto.spec.GCMParameterSpec;
import javax.crypto.spec.PBEKeySpec;
import javax.crypto.spec.SecretKeySpec;

import static org.elasticsearch.action.support.TransportActions.isShardNotAvailableException;
import static org.elasticsearch.common.hash.MessageDigests.sha256;
import static org.elasticsearch.core.Strings.format;
import static org.elasticsearch.gateway.GatewayService.STATE_NOT_RECOVERED_BLOCK;
import static org.elasticsearch.search.SearchService.DEFAULT_KEEPALIVE_SETTING;
import static org.elasticsearch.xpack.core.ClientHelper.SECURITY_ORIGIN;
import static org.elasticsearch.xpack.core.ClientHelper.executeAsyncWithOrigin;
import static org.elasticsearch.xpack.security.support.SecurityIndexManager.Availability.PRIMARY_SHARDS;
import static org.elasticsearch.xpack.security.support.SecurityIndexManager.Availability.SEARCH_SHARDS;

/**
 * Service responsible for the creation, validation, and other management of {@link UserToken}
 * objects for authentication
 */
public class TokenService {

    /**
     * The parameters below are used to generate the cryptographic key that is used to encrypt the
     * values returned by this service. These parameters are based off of the
     * OWASP Password Storage
     * Cheat Sheet and the 
     * NIST Digital Identity Guidelines
     */
    static final int TOKEN_SERVICE_KEY_ITERATIONS = 100000;
    static final int TOKENS_ENCRYPTION_KEY_ITERATIONS = 1024;
    private static final String KDF_ALGORITHM = "PBKDF2withHMACSHA512";
    static final int SALT_BYTES = 32;
    private static final int KEY_BYTES = 64;
    static final int IV_BYTES = 12;
    private static final int VERSION_BYTES = 4;
    private static final String ENCRYPTION_CIPHER = "AES/GCM/NoPadding";
    private static final String EXPIRED_TOKEN_WWW_AUTH_VALUE = String.format(Locale.ROOT, """
        Bearer realm="%s", error="invalid_token", error_description="The access token expired\"""", XPackField.SECURITY);
    private static final BackoffPolicy DEFAULT_BACKOFF = BackoffPolicy.exponentialBackoff();

    public static final String THREAD_POOL_NAME = XPackField.SECURITY + "-token-key";
    public static final Setting TOKEN_EXPIRATION = Setting.timeSetting(
        "xpack.security.authc.token.timeout",
        TimeValue.timeValueMinutes(20L),
        TimeValue.timeValueSeconds(1L),
        TimeValue.timeValueHours(1L),
        Property.NodeScope
    );
    public static final Setting DELETE_INTERVAL = Setting.timeSetting(
        "xpack.security.authc.token.delete.interval",
        TimeValue.timeValueMinutes(30L),
        Property.NodeScope
    );
    public static final Setting DELETE_TIMEOUT = Setting.timeSetting(
        "xpack.security.authc.token.delete.timeout",
        TimeValue.MINUS_ONE,
        Property.NodeScope
    );

    static final String TOKEN_DOC_TYPE = "token";
    static final int RAW_TOKEN_BYTES_LENGTH = 16;
    static final int RAW_TOKEN_DOC_ID_BYTES_LENGTH = 8;
    static final int RAW_TOKEN_BYTES_TOTAL_LENGTH = RAW_TOKEN_BYTES_LENGTH + RAW_TOKEN_DOC_ID_BYTES_LENGTH;
    // UUIDs are 16 bytes encoded base64 without padding, therefore the length is (16 / 3) * 4 + ((16 % 3) * 8 + 5) / 6 chars
    private static final int TOKEN_LENGTH = 22;
    private static final String TOKEN_DOC_ID_PREFIX = TOKEN_DOC_TYPE + "_";
    static final int LEGACY_MINIMUM_BYTES = VERSION_BYTES + SALT_BYTES + IV_BYTES + 1;
    static final int MINIMUM_BYTES = VERSION_BYTES + TOKEN_LENGTH + 1;
    static final int LEGACY_MINIMUM_BASE64_BYTES = Double.valueOf(Math.ceil((4 * LEGACY_MINIMUM_BYTES) / 3)).intValue();
    public static final int MINIMUM_BASE64_BYTES = Double.valueOf(Math.ceil((4 * MINIMUM_BYTES) / 3)).intValue();
    static final TransportVersion VERSION_HASHED_TOKENS = TransportVersions.V_7_2_0;
    static final TransportVersion VERSION_TOKENS_INDEX_INTRODUCED = TransportVersions.V_7_2_0;
    static final TransportVersion VERSION_ACCESS_TOKENS_AS_UUIDS = TransportVersions.V_7_2_0;
    static final TransportVersion VERSION_MULTIPLE_CONCURRENT_REFRESHES = TransportVersions.V_7_2_0;
    static final TransportVersion VERSION_CLIENT_AUTH_FOR_REFRESH = TransportVersions.V_8_2_0;
    static final TransportVersion VERSION_GET_TOKEN_DOC_FOR_REFRESH = TransportVersions.V_8_10_X;

    private static final Logger logger = LogManager.getLogger(TokenService.class);

    private final SecureRandom secureRandom = new SecureRandom();
    private final Settings settings;
    private final ClusterService clusterService;
    private final Clock clock;
    private final TimeValue expirationDelay;
    private final TimeValue deleteInterval;
    private final Client client;
    private final SecurityIndexManager securityMainIndex;
    private final SecurityIndexManager securityTokensIndex;
    private final ExpiredTokenRemover expiredTokenRemover;
    private final boolean enabled;
    private final XPackLicenseState licenseState;
    private final SecurityContext securityContext;
    private volatile TokenKeys keyCache;
    private volatile long lastExpirationRunMs;
    private final AtomicLong createdTimeStamps = new AtomicLong(-1);

    /**
     * Creates a new token service
     */
    @SuppressWarnings("this-escape")
    public TokenService(
        Settings settings,
        Clock clock,
        Client client,
        XPackLicenseState licenseState,
        SecurityContext securityContext,
        SecurityIndexManager securityMainIndex,
        SecurityIndexManager securityTokensIndex,
        ClusterService clusterService
    ) throws GeneralSecurityException {
        byte[] saltArr = new byte[SALT_BYTES];
        secureRandom.nextBytes(saltArr);
        final SecureString tokenPassphrase = generateTokenKey();
        this.settings = settings;
        this.clock = clock.withZone(ZoneOffset.UTC);
        this.expirationDelay = TOKEN_EXPIRATION.get(settings);
        this.client = client;
        this.licenseState = licenseState;
        this.securityContext = securityContext;
        this.securityMainIndex = securityMainIndex;
        this.securityTokensIndex = securityTokensIndex;
        this.lastExpirationRunMs = client.threadPool().relativeTimeInMillis();
        this.deleteInterval = DELETE_INTERVAL.get(settings);
        this.enabled = isTokenServiceEnabled(settings);
        this.expiredTokenRemover = new ExpiredTokenRemover(settings, client, this.securityMainIndex, securityTokensIndex);
        ensureEncryptionCiphersSupported();
        KeyAndCache keyAndCache = new KeyAndCache(
            new KeyAndTimestamp(tokenPassphrase, createdTimeStamps.incrementAndGet()),
            new BytesKey(saltArr)
        );
        keyCache = new TokenKeys(Collections.singletonMap(keyAndCache.getKeyHash(), keyAndCache), keyAndCache.getKeyHash());
        this.clusterService = clusterService;
        initialize(clusterService);
        getTokenMetadata();
    }

    /**
     * Creates an access token and optionally a refresh token as well, based on the provided authentication and metadata with
     * auto-generated values. The created tokens are stored in the security index for versions up to
     * {@link #VERSION_TOKENS_INDEX_INTRODUCED} and to a specific security tokens index for later versions.
     */
    public void createOAuth2Tokens(
        Authentication authentication,
        Authentication originatingClientAuth,
        Map metadata,
        boolean includeRefreshToken,
        ActionListener listener
    ) {
        // the created token is compatible with the oldest node version in the cluster
        final TransportVersion tokenVersion = getTokenVersionCompatibility();
        Tuple newTokenBytes = getRandomTokenBytes(tokenVersion, includeRefreshToken);
        createOAuth2Tokens(newTokenBytes.v1(), newTokenBytes.v2(), tokenVersion, authentication, originatingClientAuth, metadata, listener);
    }

    /**
     * Creates an access token and optionally a refresh token as well from predefined values, based on the provided authentication and
     * metadata. The created tokens are stored in the security index for versions up to {@link #VERSION_TOKENS_INDEX_INTRODUCED} and to a
     * specific security tokens index for later versions.
     */
    // public for testing
    public void createOAuth2Tokens(
        byte[] accessTokenBytes,
        @Nullable byte[] refreshTokenBytes,
        Authentication authentication,
        Authentication originatingClientAuth,
        Map metadata,
        ActionListener listener
    ) {
        // the created token is compatible with the oldest node version in the cluster
        final TransportVersion tokenVersion = getTokenVersionCompatibility();
        createOAuth2Tokens(accessTokenBytes, refreshTokenBytes, tokenVersion, authentication, originatingClientAuth, metadata, listener);
    }

    /**
     * Create an access token and optionally a refresh token as well from predefined values, based on the provided authentication and
     * metadata.
     *
     * @param accessTokenBytes      The predefined seed value for the access token. This will then be
     *                              
    *
  • Encrypted before stored for versions before {@link #VERSION_TOKENS_INDEX_INTRODUCED}
  • *
  • Hashed before stored for versions after {@link #VERSION_TOKENS_INDEX_INTRODUCED}
  • *
  • Stored in the security index for versions up to {@link #VERSION_TOKENS_INDEX_INTRODUCED}
  • *
  • Stored in a specific security tokens index for versions after * {@link #VERSION_TOKENS_INDEX_INTRODUCED}
  • *
  • Prepended with a version ID and Base64 encoded before returned to the caller of the APIs
  • *
* @param refreshTokenBytes The predefined seed value for the access token. This will then be *
    *
  • Hashed before stored for versions after {@link #VERSION_TOKENS_INDEX_INTRODUCED}
  • *
  • Stored in the security index for versions up to {@link #VERSION_TOKENS_INDEX_INTRODUCED}
  • *
  • Stored in a specific security tokens index for versions after * {@link #VERSION_TOKENS_INDEX_INTRODUCED}
  • *
  • Prepended with a version ID and encoded with Base64 before returned to the caller of the APIs * for versions after {@link #VERSION_TOKENS_INDEX_INTRODUCED}
  • *
* @param tokenVersion The version of the nodes with which these tokens will be compatible. * @param authentication The authentication object representing the user for which the tokens are created * @param originatingClientAuth The authentication object representing the client that called the related API * @param metadata A map with metadata to be stored in the token document * @param listener The listener to call upon completion with a {@link CreateTokenResult} containing the * serialized access token, serialized refresh token and authentication for which the token is created * as these will be returned to the client */ private void createOAuth2Tokens( byte[] accessTokenBytes, @Nullable byte[] refreshTokenBytes, TransportVersion tokenVersion, Authentication authentication, Authentication originatingClientAuth, Map metadata, ActionListener listener ) { ensureEnabled(); if (authentication == null) { listener.onFailure(traceLog("create token", new IllegalArgumentException("authentication must be provided"))); } else if (originatingClientAuth == null) { listener.onFailure( traceLog("create token", new IllegalArgumentException("originating client authentication must be provided")) ); } else { final Authentication tokenAuth = authentication.token().maybeRewriteForOlderVersion(tokenVersion); final String accessTokenToStore; final String refreshTokenToStore; final String refreshTokenToReturn; final String documentId; final BytesReference tokenDocument; try { final String userTokenId; if (tokenVersion.onOrAfter(VERSION_GET_TOKEN_DOC_FOR_REFRESH)) { assert accessTokenBytes.length == RAW_TOKEN_BYTES_TOTAL_LENGTH; MessageDigest userTokenIdDigest = sha256(); userTokenIdDigest.update(accessTokenBytes, RAW_TOKEN_BYTES_LENGTH, RAW_TOKEN_DOC_ID_BYTES_LENGTH); userTokenId = Base64.getUrlEncoder().withoutPadding().encodeToString(userTokenIdDigest.digest()); accessTokenToStore = Base64.getUrlEncoder().withoutPadding().encodeToString(sha256().digest(accessTokenBytes)); if (refreshTokenBytes != null) { assert refreshTokenBytes.length == RAW_TOKEN_BYTES_TOTAL_LENGTH; assert Arrays.equals( refreshTokenBytes, RAW_TOKEN_BYTES_LENGTH, RAW_TOKEN_BYTES_TOTAL_LENGTH, accessTokenBytes, RAW_TOKEN_BYTES_LENGTH, RAW_TOKEN_BYTES_TOTAL_LENGTH ) : "the last bytes of paired access and refresh tokens should be the same"; refreshTokenToStore = Base64.getUrlEncoder().withoutPadding().encodeToString(sha256().digest(refreshTokenBytes)); refreshTokenToReturn = prependVersionAndEncodeRefreshToken(tokenVersion, refreshTokenBytes); } else { refreshTokenToStore = refreshTokenToReturn = null; } } else if (tokenVersion.onOrAfter(VERSION_HASHED_TOKENS)) { assert accessTokenBytes.length == RAW_TOKEN_BYTES_LENGTH; userTokenId = hashTokenString(Base64.getUrlEncoder().withoutPadding().encodeToString(accessTokenBytes)); accessTokenToStore = null; if (refreshTokenBytes != null) { assert refreshTokenBytes.length == RAW_TOKEN_BYTES_LENGTH; refreshTokenToStore = hashTokenString(Base64.getUrlEncoder().withoutPadding().encodeToString(refreshTokenBytes)); refreshTokenToReturn = prependVersionAndEncodeRefreshToken(tokenVersion, refreshTokenBytes); } else { refreshTokenToStore = refreshTokenToReturn = null; } } else { assert accessTokenBytes.length == RAW_TOKEN_BYTES_LENGTH; userTokenId = Base64.getUrlEncoder().withoutPadding().encodeToString(accessTokenBytes); accessTokenToStore = null; if (refreshTokenBytes != null) { assert refreshTokenBytes.length == RAW_TOKEN_BYTES_LENGTH; refreshTokenToStore = refreshTokenToReturn = Base64.getUrlEncoder() .withoutPadding() .encodeToString(refreshTokenBytes); } else { refreshTokenToStore = refreshTokenToReturn = null; } } UserToken userToken = new UserToken(userTokenId, tokenVersion, tokenAuth, getExpirationTime(), metadata); tokenDocument = createTokenDocument(userToken, accessTokenToStore, refreshTokenToStore, originatingClientAuth); documentId = getTokenDocumentId(userToken); } catch (IOException e) { logger.error("Could not encode access or refresh token", e); listener.onFailure(traceLog("create token", e)); return; } final RefreshPolicy tokenCreationRefreshPolicy = tokenVersion.onOrAfter(VERSION_GET_TOKEN_DOC_FOR_REFRESH) ? RefreshPolicy.NONE : RefreshPolicy.WAIT_UNTIL; final SecurityIndexManager tokensIndex = getTokensIndexForVersion(tokenVersion); logger.debug( () -> format( "Using refresh policy [%s] when creating token doc [%s] in the security index [%s]", tokenCreationRefreshPolicy, documentId, tokensIndex.aliasName() ) ); final IndexRequest indexTokenRequest = client.prepareIndex(tokensIndex.aliasName()) .setId(documentId) .setOpType(OpType.CREATE) .setSource(tokenDocument, XContentType.JSON) .setRefreshPolicy(tokenCreationRefreshPolicy) .request(); tokensIndex.prepareIndexIfNeededThenExecute( ex -> listener.onFailure(traceLog("prepare tokens index [" + tokensIndex.aliasName() + "]", documentId, ex)), () -> executeAsyncWithOrigin( client, SECURITY_ORIGIN, TransportIndexAction.TYPE, indexTokenRequest, ActionListener.wrap(indexResponse -> { if (indexResponse.getResult() == Result.CREATED) { String accessTokenToReturn = prependVersionAndEncodeAccessToken(tokenVersion, accessTokenBytes); listener.onResponse(new CreateTokenResult(accessTokenToReturn, refreshTokenToReturn, authentication)); } else { listener.onFailure( traceLog("create token", new ElasticsearchException("failed to create token document [{}]", indexResponse)) ); } }, listener::onFailure) ) ); } } /** * Hashes an access or refresh token String so that it can safely be persisted in the index. We don't salt * the values as these are v4 UUIDs that have enough entropy by themselves. */ // public for testing public static String hashTokenString(String accessTokenString) { return new String(Hasher.SHA256.hash(new SecureString(accessTokenString.toCharArray()))); } /** * If the token is non-null, then it is validated, which might include authenticated decryption and * verification that the token has not been revoked or is expired. */ void tryAuthenticateToken(SecureString token, ActionListener listener) { if (shouldTryRealm() && token != null) { decodeToken(token.toString(), true, listener.delegateResponse((l, e) -> { if (isShardNotAvailableException(e)) { l.onResponse(null); } else { l.onFailure(e); } })); } else { listener.onResponse(null); } } /** * Reads the authentication and metadata from the given token. * This method does not validate whether the token is expired or not. */ public void getAuthenticationAndMetadata(String token, ActionListener>> listener) { decodeToken(token, false, ActionListener.wrap(userToken -> { if (userToken == null) { listener.onFailure(new ElasticsearchSecurityException("supplied token is not valid")); } else { listener.onResponse(new Tuple<>(userToken.getAuthentication(), userToken.getMetadata())); } }, listener::onFailure)); } /** * Retrieves the {@code UserToken} given the {@param tokenId} and {@param tokenVersion}. * If passed in, {@param storedAccessToken} is verified against the value stored in the token document. * A mismatch here indicates that the token doc represents a different token from the one the client * is referring to. * If {@param validateUserToken} is {@code true} the {@code UserToken} is also checked to not be * invalidated or expired. */ @SuppressWarnings("unchecked") private void getAndValidateUserToken( String tokenId, TransportVersion tokenVersion, @Nullable String storedAccessToken, boolean validateUserToken, ActionListener listener ) { getTokenDocById(tokenId, tokenVersion, storedAccessToken, null, ActionListener.wrap(doc -> { if (doc == null) { listener.onResponse(null); } else { Map accessTokenSource = (Map) doc.sourceAsMap().get("access_token"); if (validateUserToken) { Boolean invalidated = (Boolean) accessTokenSource.get("invalidated"); if (invalidated == null) { listener.onFailure(new IllegalStateException("token document is missing invalidated field")); } else if (invalidated) { listener.onFailure(expiredTokenException()); } else { UserToken userToken = UserToken.fromSourceMap((Map) accessTokenSource.get("user_token")); if (clock.instant().isAfter(userToken.getExpirationTime())) { listener.onFailure(traceLog("validate token", userToken.getId(), expiredTokenException())); } else { listener.onResponse(userToken); } } } else { listener.onResponse(UserToken.fromSourceMap((Map) accessTokenSource.get("user_token"))); } } }, listener::onFailure)); } /** * Retrieves a token doc given the {@param tokenId} and {@param tokenVersion}. * If passed in, {@param storedAccessToken} and {@param storedRefreshToken} are verified against the values stored * in the token document. A mismatch here indicates that the token doc represents a different token from the one * that the client is referring to, even though the client knows (or guessed) the {@param tokenId}. In other words, * the client knows only part of the secret token (some part of it is wrong). */ @SuppressWarnings("unchecked") private void getTokenDocById( String tokenId, TransportVersion tokenVersion, @Nullable String storedAccessToken, @Nullable String storedRefreshToken, ActionListener listener ) { final SecurityIndexManager tokensIndex = getTokensIndexForVersion(tokenVersion); final SecurityIndexManager frozenTokensIndex = tokensIndex.defensiveCopy(); if (frozenTokensIndex.isAvailable(PRIMARY_SHARDS) == false) { logger.warn("failed to get access token [{}] because index [{}] is not available", tokenId, tokensIndex.aliasName()); listener.onFailure(frozenTokensIndex.getUnavailableReason(PRIMARY_SHARDS)); return; } final GetRequest getRequest = client.prepareGet(tokensIndex.aliasName(), getTokenDocumentId(tokenId)).request(); final Consumer onFailure = ex -> listener.onFailure(traceLog("get token from id", tokenId, ex)); tokensIndex.checkIndexVersionThenExecute( ex -> listener.onFailure(traceLog("prepare tokens index [" + tokensIndex.aliasName() + "]", tokenId, ex)), () -> executeAsyncWithOrigin( client.threadPool().getThreadContext(), SECURITY_ORIGIN, getRequest, ActionListener.wrap(response -> { if (response.isExists() == false) { // The chances of a random token string decoding to something that we can read is minimal, so // we assume that this was a token we have created but is now expired/revoked and deleted logger.trace("The token [{}] probably expired and has already been deleted", tokenId); listener.onResponse(null); return; } Map accessSource = (Map) response.getSource().get("access_token"); Map refreshSource = (Map) response.getSource().get("refresh_token"); boolean versionGetForRefresh = tokenVersion.onOrAfter(VERSION_GET_TOKEN_DOC_FOR_REFRESH); if (accessSource == null) { onFailure.accept(new IllegalStateException("token document is missing the access_token field")); } else if (accessSource.containsKey("user_token") == false) { onFailure.accept(new IllegalStateException("token document is missing the user_token field")); } else if (versionGetForRefresh && accessSource.containsKey("token") == false) { onFailure.accept(new IllegalStateException("token document is missing the user_token.token field")); } else if (versionGetForRefresh && refreshSource != null && refreshSource.containsKey("token") == false) { onFailure.accept(new IllegalStateException("token document is missing the refresh_token.token field")); } else if (storedAccessToken != null && storedAccessToken.equals(accessSource.get("token")) == false) { logger.error( "The stored access token [{}] for token doc id [{}] could not be verified", storedAccessToken, tokenId ); listener.onResponse(null); } else if (storedRefreshToken != null && (refreshSource == null || storedRefreshToken.equals(refreshSource.get("token")) == false)) { logger.error( "The stored refresh token [{}] for token doc id [{}] could not be verified", storedRefreshToken, tokenId ); listener.onResponse(null); } else { listener.onResponse(new Doc(response)); } }, e -> { // if the index or the shard is not there / available we assume that // the token is not valid if (isShardNotAvailableException(e)) { logger.warn("failed to get token doc [{}] because index [{}] is not available", tokenId, tokensIndex.aliasName()); } else { logger.error(() -> "failed to get token doc [" + tokenId + "]", e); } listener.onFailure(e); }), client::get ) ); } /** * If needed, for tokens that were created in a pre {@code #VERSION_ACCESS_TOKENS_UUIDS} cluster, it asynchronously decodes the token to * get the token document id. The process for this is asynchronous as we may need to compute a key, which can be computationally * expensive so this should not block the current thread, which is typically a network thread. A second reason for being asynchronous is * that we can restrain the amount of resources consumed by the key computation to a single thread. For tokens created in an after * {@code #VERSION_ACCESS_TOKENS_UUIDS} cluster, the token is just the token document Id so this is used directly without decryption * */ void decodeToken(String token, boolean validateUserToken, ActionListener listener) { final byte[] bytes = token.getBytes(StandardCharsets.UTF_8); try (StreamInput in = new InputStreamStreamInput(Base64.getDecoder().wrap(new ByteArrayInputStream(bytes)), bytes.length)) { final TransportVersion version = TransportVersion.readVersion(in); in.setTransportVersion(version); if (version.onOrAfter(VERSION_GET_TOKEN_DOC_FOR_REFRESH)) { byte[] accessTokenBytes = in.readByteArray(); if (accessTokenBytes.length != RAW_TOKEN_BYTES_TOTAL_LENGTH) { logger.debug( "invalid token, received size [{}] bytes is different from expect size [{}]", accessTokenBytes.length, RAW_TOKEN_BYTES_TOTAL_LENGTH ); listener.onResponse(null); return; } MessageDigest userTokenIdDigest = sha256(); userTokenIdDigest.update(accessTokenBytes, RAW_TOKEN_BYTES_LENGTH, RAW_TOKEN_DOC_ID_BYTES_LENGTH); final String userTokenId = Base64.getUrlEncoder().withoutPadding().encodeToString(userTokenIdDigest.digest()); final String storedAccessToken = Base64.getUrlEncoder().withoutPadding().encodeToString(sha256().digest(accessTokenBytes)); getAndValidateUserToken(userTokenId, version, storedAccessToken, validateUserToken, listener); } else if (version.onOrAfter(VERSION_ACCESS_TOKENS_AS_UUIDS)) { // The token was created in a > VERSION_ACCESS_TOKENS_UUIDS cluster if (in.available() < MINIMUM_BYTES) { logger.debug("invalid token, smaller than [{}] bytes", MINIMUM_BYTES); listener.onResponse(null); return; } final String accessToken = in.readString(); final String userTokenId = hashTokenString(accessToken); getAndValidateUserToken(userTokenId, version, null, validateUserToken, listener); } else { // The token was created in a < VERSION_ACCESS_TOKENS_UUIDS cluster so we need to decrypt it to get the tokenId if (in.available() < LEGACY_MINIMUM_BYTES) { logger.debug("invalid token, smaller than [{}] bytes", LEGACY_MINIMUM_BYTES); listener.onResponse(null); return; } final BytesKey decodedSalt = new BytesKey(in.readByteArray()); final BytesKey passphraseHash = new BytesKey(in.readByteArray()); final byte[] iv = in.readByteArray(); final BytesStreamOutput out = new BytesStreamOutput(); Streams.copy(in, out); final byte[] encryptedTokenId = BytesReference.toBytes(out.bytes()); final KeyAndCache keyAndCache = keyCache.get(passphraseHash); if (keyAndCache != null) { getKeyAsync(decodedSalt, keyAndCache, ActionListener.wrap(decodeKey -> { if (decodeKey != null) { try { final Cipher cipher = getDecryptionCipher(iv, decodeKey, version, decodedSalt); final String tokenId = decryptTokenId(encryptedTokenId, cipher, version); getAndValidateUserToken(tokenId, version, null, validateUserToken, listener); } catch (IOException | GeneralSecurityException e) { // could happen with a token that is not ours logger.warn("invalid token", e); listener.onResponse(null); } } else { // could happen with a token that is not ours listener.onResponse(null); } }, listener::onFailure)); } else { logger.debug(() -> format("invalid key %s key: %s", passphraseHash, keyCache.cache.keySet())); listener.onResponse(null); } } } catch (Exception e) { // could happen with a token that is not ours logger.debug("built in token service unable to decode token", e); listener.onResponse(null); } } /** * This method performs the steps necessary to invalidate an access token so that it may no longer be * used. The process of invalidation involves performing an update to the token document and setting * the {@code access_token.invalidated} field to {@code true} */ public void invalidateAccessToken(String accessToken, ActionListener listener) { ensureEnabled(); if (Strings.isNullOrEmpty(accessToken)) { listener.onFailure(traceLog("invalidate access token", new IllegalArgumentException("access token must be provided"))); } else { maybeStartTokenRemover(); final Iterator backoff = DEFAULT_BACKOFF.iterator(); decodeToken(accessToken, false, ActionListener.wrap(userToken -> { if (userToken == null) { // The chances of a random token string decoding to something that we can read is minimal, so // we assume that this was a token we have created but is now expired/revoked and deleted logger.trace("The access token [{}] is expired and already deleted", accessToken); listener.onResponse(TokensInvalidationResult.emptyResult(RestStatus.NOT_FOUND)); } else { indexInvalidation(Collections.singleton(userToken), backoff, "access_token", null, listener); } }, e -> { if (e instanceof IndexNotFoundException || e instanceof IndexClosedException) { listener.onFailure(new ElasticsearchSecurityException("failed to invalidate token", RestStatus.BAD_REQUEST)); } else { listener.onFailure(unableToPerformAction(e)); } })); } } /** * This method invalidates a refresh token so that it may no longer be used. Invalidation involves performing an update to the token * document and setting the refresh_token.invalidated field to true * * @param refreshToken The string representation of the refresh token * @param listener the listener to notify upon completion */ public void invalidateRefreshToken(String refreshToken, ActionListener listener) { ensureEnabled(); if (Strings.isNullOrEmpty(refreshToken)) { logger.trace("No refresh token provided"); listener.onFailure(new IllegalArgumentException("refresh token must be provided")); } else { maybeStartTokenRemover(); final Iterator backoff = DEFAULT_BACKOFF.iterator(); findTokenFromRefreshToken(refreshToken, backoff, ActionListener.wrap(tokenDoc -> { if (tokenDoc == null) { logger.debug("could not find token document for refresh token"); listener.onResponse(TokensInvalidationResult.emptyResult(RestStatus.NOT_FOUND)); } else { final Tuple parsedTokens = parseTokenAndRefreshStatus(tokenDoc.sourceAsMap()); final UserToken userToken = parsedTokens.v1(); final RefreshTokenStatus refresh = parsedTokens.v2(); if (refresh.isInvalidated()) { listener.onResponse(new TokensInvalidationResult(List.of(), List.of(userToken.getId()), null, RestStatus.OK)); } else { indexInvalidation(Collections.singletonList(userToken), backoff, "refresh_token", null, listener); } } }, e -> { if (e instanceof IndexNotFoundException || e instanceof IndexClosedException) { listener.onFailure(new ElasticsearchSecurityException("failed to invalidate token", RestStatus.BAD_REQUEST)); } else { listener.onFailure(unableToPerformAction(e)); } })); } } /** * Invalidates all access and refresh tokens for a given {@code realmName} and/or of a given * {@code username} so that they may no longer be usable. * * @param realmName the realm of which the tokens should be invalidated * @param username the username for which the tokens should be invalidated * @param filter An optional {@code Predicate} to further test and filter the tokens to invalidate. * The predicate tests the token doc source. * @param listener the listener to notify upon completion */ public void invalidateActiveTokens( @Nullable String realmName, @Nullable String username, @Nullable Predicate> filter, ActionListener listener ) { ensureEnabled(); maybeStartTokenRemover(); if (Strings.isNullOrEmpty(realmName) && Strings.isNullOrEmpty(username)) { logger.trace("No realm name or username provided"); listener.onFailure(new IllegalArgumentException("realm name or username must be provided")); } else { if (Strings.hasText(username)) { filter = filter == null ? isOfUser(username) : filter.and(isOfUser(username)); } searchActiveTokens(realmName, filter, ActionListener.wrap(tokenTuples -> { if (tokenTuples.isEmpty()) { logger.warn("No tokens to invalidate for realm [{}] and username [{}]", realmName, username); listener.onResponse(TokensInvalidationResult.emptyResult(RestStatus.OK)); } else { invalidateTokens(tokenTuples, listener); } }, listener::onFailure)); } } /** * Invalidates a collection of access and refresh token pairs. It invalidates tokens in "bulk", first all the refresh * tokens, then all the access tokens. */ private void invalidateTokens(Collection> tokenTuples, ActionListener listener) { // Invalidate the refresh tokens first so that they cannot be used to get new // access tokens while we invalidate the access tokens we currently know about final List userTokens = new ArrayList<>(); final List tokensWithRefresh = new ArrayList<>(); tokenTuples.forEach(t -> { userTokens.add(t.v1()); if (t.v2() != null) { tokensWithRefresh.add(t.v1()); } }); final Iterator backoff = DEFAULT_BACKOFF.iterator(); if (false == tokensWithRefresh.isEmpty()) { indexInvalidation( tokensWithRefresh, backoff, "refresh_token", null, ActionListener.wrap(result -> indexInvalidation(userTokens, backoff, "access_token", result, listener), listener::onFailure) ); } else { indexInvalidation(userTokens, backoff, "access_token", null, listener); } } /** * Invalidates access and/or refresh tokens associated to a user token (coexisting in the same token document) */ private void indexInvalidation( Collection userTokens, Iterator backoff, String srcPrefix, @Nullable TokensInvalidationResult previousResult, ActionListener listener ) { final Set idsOfRecentTokens = new HashSet<>(); final Set idsOfOlderTokens = new HashSet<>(); boolean anyOlderTokensBeforeRefreshViaGet = false; for (UserToken userToken : userTokens) { if (userToken.getTransportVersion().onOrAfter(VERSION_TOKENS_INDEX_INTRODUCED)) { idsOfRecentTokens.add(userToken.getId()); } else { idsOfOlderTokens.add(userToken.getId()); } anyOlderTokensBeforeRefreshViaGet |= userToken.getTransportVersion().before(VERSION_GET_TOKEN_DOC_FOR_REFRESH); } final RefreshPolicy tokensInvalidationRefreshPolicy = anyOlderTokensBeforeRefreshViaGet ? RefreshPolicy.WAIT_UNTIL : RefreshPolicy.NONE; if (false == idsOfOlderTokens.isEmpty()) { indexInvalidation( idsOfOlderTokens, securityMainIndex, backoff, srcPrefix, previousResult, RefreshPolicy.WAIT_UNTIL, ActionListener.wrap(newResult -> { if (false == idsOfRecentTokens.isEmpty()) { // carry-over result of the invalidation for the tokens security index indexInvalidation( idsOfRecentTokens, securityTokensIndex, backoff, srcPrefix, newResult, tokensInvalidationRefreshPolicy, listener ); } else { listener.onResponse(newResult); } }, listener::onFailure) ); } else { indexInvalidation( idsOfRecentTokens, securityTokensIndex, backoff, srcPrefix, previousResult, tokensInvalidationRefreshPolicy, listener ); } } /** * Performs the actual invalidation of a collection of tokens. In case of recoverable errors ( see * {@link TransportActions#isShardNotAvailableException} ) the UpdateRequests to mark the tokens as invalidated are retried using * an exponential backoff policy. * * @param tokenIds the tokens to invalidate * @param tokensIndexManager the manager for the index where the tokens are stored * @param backoff the amount of time to delay between attempts * @param srcPrefix the prefix to use when constructing the doc to update, either refresh_token or access_token depending on * what type of tokens should be invalidated * @param previousResult if this not the initial attempt for invalidation, it contains the result of invalidating * tokens up to the point of the retry. This result is added to the result of the current attempt * @param refreshPolicy the {@link RefreshPolicy} for the indexing op * @param listener the listener to notify upon completion */ private void indexInvalidation( Collection tokenIds, SecurityIndexManager tokensIndexManager, Iterator backoff, String srcPrefix, @Nullable TokensInvalidationResult previousResult, RefreshPolicy refreshPolicy, ActionListener listener ) { if (tokenIds.isEmpty()) { logger.warn("No [{}] tokens provided for invalidation", srcPrefix); listener.onFailure(invalidGrantException("No tokens provided for invalidation")); } else { BulkRequestBuilder bulkRequestBuilder = client.prepareBulk(); for (String tokenId : tokenIds) { UpdateRequest request = client.prepareUpdate(tokensIndexManager.aliasName(), getTokenDocumentId(tokenId)) .setDoc(srcPrefix, Collections.singletonMap("invalidated", true)) .setFetchSource(srcPrefix, null) .request(); bulkRequestBuilder.add(request); } logger.debug( () -> format( "Using refresh policy [%s] when updating token docs [%s] for invalidation in the security index [%s]", refreshPolicy, tokenIds, tokensIndexManager.aliasName() ) ); bulkRequestBuilder.setRefreshPolicy(refreshPolicy); tokensIndexManager.prepareIndexIfNeededThenExecute( ex -> listener.onFailure(traceLog("prepare index [" + tokensIndexManager.aliasName() + "]", ex)), () -> executeAsyncWithOrigin( client.threadPool().getThreadContext(), SECURITY_ORIGIN, bulkRequestBuilder.request(), ActionListener.wrap(bulkResponse -> { ArrayList retryTokenDocIds = new ArrayList<>(); ArrayList failedRequestResponses = new ArrayList<>(); ArrayList previouslyInvalidated = new ArrayList<>(); ArrayList invalidated = new ArrayList<>(); if (null != previousResult) { failedRequestResponses.addAll((previousResult.getErrors())); previouslyInvalidated.addAll(previousResult.getPreviouslyInvalidatedTokens()); invalidated.addAll(previousResult.getInvalidatedTokens()); } for (BulkItemResponse bulkItemResponse : bulkResponse.getItems()) { if (bulkItemResponse.isFailed()) { Throwable cause = bulkItemResponse.getFailure().getCause(); final String failedTokenDocId = getTokenIdFromDocumentId(bulkItemResponse.getFailure().getId()); if (isShardNotAvailableException(cause)) { retryTokenDocIds.add(failedTokenDocId); } else { traceLog("invalidate access token", failedTokenDocId, cause); failedRequestResponses.add(new ElasticsearchException("Error invalidating " + srcPrefix + ": ", cause)); } } else { UpdateResponse updateResponse = bulkItemResponse.getResponse(); if (updateResponse.getResult() == DocWriteResponse.Result.UPDATED) { logger.debug( () -> format("Invalidated [%s] for doc [%s]", srcPrefix, updateResponse.getGetResult().getId()) ); invalidated.add(updateResponse.getGetResult().getId()); } else if (updateResponse.getResult() == DocWriteResponse.Result.NOOP) { previouslyInvalidated.add(updateResponse.getGetResult().getId()); } } } if (retryTokenDocIds.isEmpty() == false && backoff.hasNext()) { logger.debug( "failed to invalidate [{}] tokens out of [{}], retrying to invalidate these too", retryTokenDocIds.size(), tokenIds.size() ); final TokensInvalidationResult incompleteResult = new TokensInvalidationResult( invalidated, previouslyInvalidated, failedRequestResponses, RestStatus.OK ); client.threadPool() .schedule( () -> indexInvalidation( retryTokenDocIds, tokensIndexManager, backoff, srcPrefix, incompleteResult, refreshPolicy, listener ), backoff.next(), client.threadPool().generic() ); } else { if (retryTokenDocIds.isEmpty() == false) { logger.warn( "failed to invalidate [{}] tokens out of [{}] after all retries", retryTokenDocIds.size(), tokenIds.size() ); for (String retryTokenDocId : retryTokenDocIds) { failedRequestResponses.add( new ElasticsearchException( "Error invalidating [{}] with doc id [{}] after retries exhausted", srcPrefix, retryTokenDocId ) ); } } final TokensInvalidationResult result = new TokensInvalidationResult( invalidated, previouslyInvalidated, failedRequestResponses, RestStatus.OK ); listener.onResponse(result); } }, e -> { Throwable cause = ExceptionsHelper.unwrapCause(e); traceLog("invalidate tokens", cause); if (isShardNotAvailableException(cause) && backoff.hasNext()) { logger.debug("failed to invalidate tokens, retrying "); client.threadPool() .schedule( () -> indexInvalidation( tokenIds, tokensIndexManager, backoff, srcPrefix, previousResult, refreshPolicy, listener ), backoff.next(), client.threadPool().generic() ); } else { listener.onFailure(e); } }), client::bulk ) ); } } /** * Called by the transport action in order to start the process of refreshing a token. * * @param refreshToken The refresh token as provided by the client * @param listener The listener to call upon completion with a {@link CreateTokenResult} containing the * serialized access token, serialized refresh token and authentication for which the token is created * as these will be returned to the client */ public void refreshToken(String refreshToken, ActionListener listener) { ensureEnabled(); final Instant refreshRequested = clock.instant(); final Iterator backoff = DEFAULT_BACKOFF.iterator(); final Consumer onFailure = ex -> listener.onFailure(traceLog("find token by refresh token", refreshToken, ex)); findTokenFromRefreshToken(refreshToken, backoff, ActionListener.wrap(tokenDoc -> { if (tokenDoc == null) { logger.warn("could not find token document for refresh token"); onFailure.accept(invalidGrantException("could not refresh the requested token")); } else { final Authentication clientAuth = securityContext.getAuthentication(); innerRefresh(refreshToken, tokenDoc, clientAuth, backoff, refreshRequested, listener); } }, e -> listener.onFailure(invalidGrantException("could not refresh the requested token")))); } /** * Infers the format and version of the passed in {@code refreshToken}. Delegates the actual search of the token document to * {@code #findTokenFromRefreshToken(String, SecurityIndexManager, Iterator, ActionListener)} . */ private void findTokenFromRefreshToken(String refreshToken, Iterator backoff, ActionListener listener) { if (refreshToken.length() == TOKEN_LENGTH) { // first check if token has the old format before the new version-prepended one logger.debug( "Assuming an unversioned refresh token [{}], generated for node versions" + " prior to the introduction of the version-header format.", refreshToken ); findTokenFromRefreshToken(refreshToken, securityMainIndex, backoff, listener); } else { final byte[] bytes = refreshToken.getBytes(StandardCharsets.UTF_8); try (StreamInput in = new InputStreamStreamInput(Base64.getDecoder().wrap(new ByteArrayInputStream(bytes)), bytes.length)) { final TransportVersion version = TransportVersion.readVersion(in); in.setTransportVersion(version); if (version.onOrAfter(VERSION_GET_TOKEN_DOC_FOR_REFRESH)) { final byte[] refreshTokenBytes = in.readByteArray(); if (refreshTokenBytes.length != RAW_TOKEN_BYTES_TOTAL_LENGTH) { logger.debug( "Refresh token with version [{}] has length [{}] but expect length is [{}].", version, refreshTokenBytes.length, RAW_TOKEN_BYTES_LENGTH ); listener.onResponse(null); } else { MessageDigest userTokenIdDigest = sha256(); userTokenIdDigest.update(refreshTokenBytes, RAW_TOKEN_BYTES_LENGTH, RAW_TOKEN_DOC_ID_BYTES_LENGTH); final String userTokenId = Base64.getUrlEncoder().withoutPadding().encodeToString(userTokenIdDigest.digest()); final String storedRefreshToken = Base64.getUrlEncoder() .withoutPadding() .encodeToString(sha256().digest(refreshTokenBytes)); getTokenDocById(userTokenId, version, null, storedRefreshToken, listener); } } else if (version.onOrAfter(VERSION_HASHED_TOKENS)) { final String unencodedRefreshToken = in.readString(); if (unencodedRefreshToken.length() != TOKEN_LENGTH) { logger.debug("Decoded refresh token [{}] with version [{}] is invalid.", unencodedRefreshToken, version); listener.onResponse(null); } else { final String hashedRefreshToken = hashTokenString(unencodedRefreshToken); findTokenFromRefreshToken(hashedRefreshToken, securityTokensIndex, backoff, listener); } } else { logger.debug("Unrecognized refresh token version [{}].", version); listener.onResponse(null); } } catch (IOException e) { logger.debug(() -> "Could not decode refresh token [" + refreshToken + "].", e); listener.onResponse(null); } } } /** * Performs an asynchronous search request for the token document that contains the {@code refreshToken} and calls the {@code listener} * with the resulting {@link SearchResponse}. In case of recoverable errors the {@code SearchRequest} is retried using an exponential * backoff policy. This method requires the tokens index where the token document, pointed to by the refresh token, resides. */ private void findTokenFromRefreshToken( String refreshToken, SecurityIndexManager tokensIndexManager, Iterator backoff, ActionListener listener ) { final Consumer onFailure = ex -> listener.onFailure(traceLog("find token by refresh token", refreshToken, ex)); final Consumer maybeRetryOnFailure = ex -> { if (backoff.hasNext()) { final TimeValue backofTimeValue = backoff.next(); logger.debug("retrying after [{}] back off", backofTimeValue); client.threadPool() .schedule( () -> findTokenFromRefreshToken(refreshToken, tokensIndexManager, backoff, listener), backofTimeValue, client.threadPool().generic() ); } else { logger.warn("failed to find token from refresh token after all retries"); onFailure.accept(ex); } }; final SecurityIndexManager frozenTokensIndex = tokensIndexManager.defensiveCopy(); if (frozenTokensIndex.indexExists() == false) { logger.warn("index [{}] does not exist so we can't find token from refresh token", frozenTokensIndex.aliasName()); listener.onFailure(new IndexNotFoundException(frozenTokensIndex.aliasName())); } else if (frozenTokensIndex.isAvailable(SEARCH_SHARDS) == false) { logger.debug("index [{}] is not available to find token from refresh token, retrying", frozenTokensIndex.aliasName()); maybeRetryOnFailure.accept(frozenTokensIndex.getUnavailableReason(SEARCH_SHARDS)); } else { final SearchRequest request = client.prepareSearch(tokensIndexManager.aliasName()) .setQuery( QueryBuilders.boolQuery() .filter(QueryBuilders.termQuery("doc_type", TOKEN_DOC_TYPE)) .filter(QueryBuilders.termQuery("refresh_token.token", refreshToken)) ) .seqNoAndPrimaryTerm(true) .request(); tokensIndexManager.checkIndexVersionThenExecute( listener::onFailure, () -> executeAsyncWithOrigin( client.threadPool().getThreadContext(), SECURITY_ORIGIN, request, ActionListener.wrap(searchResponse -> { if (searchResponse.isTimedOut()) { logger.debug("find token from refresh token response timed out, retrying"); maybeRetryOnFailure.accept(invalidGrantException("could not refresh the requested token")); } else if (searchResponse.getHits().getHits().length > 1) { listener.onFailure(new IllegalStateException("multiple tokens share the same refresh token")); } else if (searchResponse.getHits().getHits().length < 1) { logger.debug("could not find token document for refresh token"); listener.onResponse(null); } else { listener.onResponse(new Doc(searchResponse.getHits().getAt(0))); } }, e -> { if (isShardNotAvailableException(e)) { logger.debug("find token from refresh token request failed because of unavailable shards, retrying"); maybeRetryOnFailure.accept(invalidGrantException("could not refresh the requested token")); } else { onFailure.accept(e); } }), client::search ) ); } } /** * Performs the actual refresh of the token with retries in case of certain exceptions that may be recoverable. The refresh involves two * steps: First, we check if the token document is still valid for refresh. Then, in the case that the token has been refreshed within * the previous 30 seconds, we do not create a new token document but instead retrieve the one that was created by the original refresh * and return an access token and refresh token based on that. Otherwise this token document gets its refresh_token marked as refreshed, * while also storing the Instant when it was refreshed along with a pointer to the new token document that holds the refresh_token that * supersedes this one. The new document that contains the new access token and refresh token is created and finally the new access * token and refresh token are returned to the listener. */ private void innerRefresh( String refreshToken, Doc tokenDoc, Authentication clientAuth, Iterator backoff, Instant refreshRequested, ActionListener listener ) { logger.debug("Attempting to refresh token stored in token document [{}]", tokenDoc.id()); final Consumer onFailure = ex -> listener.onFailure(traceLog("refresh token", tokenDoc.id(), ex)); final Tuple> checkRefreshResult; try { checkRefreshResult = checkTokenDocumentForRefresh(refreshRequested, clientAuth, tokenDoc.sourceAsMap()); } catch (DateTimeException | IllegalStateException e) { onFailure.accept(new ElasticsearchSecurityException("invalid token document", e)); return; } if (checkRefreshResult.v2().isPresent()) { onFailure.accept(checkRefreshResult.v2().get()); return; } final RefreshTokenStatus refreshTokenStatus = checkRefreshResult.v1(); final SecurityIndexManager refreshedTokenIndex = getTokensIndexForVersion(refreshTokenStatus.getTransportVersion()); if (refreshTokenStatus.isRefreshed()) { logger.debug( "Token document [{}] was recently refreshed, when a new token document was generated. Reusing that result.", tokenDoc.id() ); final Tuple parsedTokens = parseTokensFromDocument(tokenDoc.sourceAsMap(), null); Authentication authentication = parsedTokens.v1().getAuthentication(); decryptAndReturnSupersedingTokens(refreshToken, refreshTokenStatus, refreshedTokenIndex, authentication, listener); } else { final TransportVersion newTokenVersion = getTokenVersionCompatibility(); final Tuple newTokenBytes = getRandomTokenBytes(newTokenVersion, true); final Map updateMap = new HashMap<>(); updateMap.put("refreshed", true); if (newTokenVersion.onOrAfter(VERSION_MULTIPLE_CONCURRENT_REFRESHES)) { updateMap.put("refresh_time", clock.instant().toEpochMilli()); try { final byte[] iv = getRandomBytes(IV_BYTES); final byte[] salt = getRandomBytes(SALT_BYTES); String encryptedAccessAndRefreshToken = encryptSupersedingTokens( newTokenBytes.v1(), newTokenBytes.v2(), refreshToken, iv, salt ); updateMap.put("superseding.encrypted_tokens", encryptedAccessAndRefreshToken); updateMap.put("superseding.encryption_iv", Base64.getEncoder().encodeToString(iv)); updateMap.put("superseding.encryption_salt", Base64.getEncoder().encodeToString(salt)); } catch (GeneralSecurityException e) { logger.warn("could not encrypt access token and refresh token string", e); onFailure.accept(invalidGrantException("could not refresh the requested token")); } } assert tokenDoc.seqNo() != SequenceNumbers.UNASSIGNED_SEQ_NO : "expected an assigned sequence number"; assert tokenDoc.primaryTerm() != SequenceNumbers.UNASSIGNED_PRIMARY_TERM : "expected an assigned primary term"; final RefreshPolicy tokenRefreshUpdateRefreshPolicy = refreshTokenStatus.getTransportVersion() .onOrAfter(VERSION_GET_TOKEN_DOC_FOR_REFRESH) ? RefreshPolicy.NONE : RefreshPolicy.IMMEDIATE; logger.debug( () -> format( "Using refresh policy [%s] when updating token doc [%s] for refresh in the security index [%s]", tokenRefreshUpdateRefreshPolicy, tokenDoc.id(), refreshedTokenIndex.aliasName() ) ); final UpdateRequestBuilder updateRequest = client.prepareUpdate(refreshedTokenIndex.aliasName(), tokenDoc.id()) .setDoc("refresh_token", updateMap) .setFetchSource(logger.isDebugEnabled()) .setRefreshPolicy(tokenRefreshUpdateRefreshPolicy) .setIfSeqNo(tokenDoc.seqNo()) .setIfPrimaryTerm(tokenDoc.primaryTerm()); refreshedTokenIndex.prepareIndexIfNeededThenExecute( ex -> listener.onFailure(traceLog("prepare index [" + refreshedTokenIndex.aliasName() + "]", ex)), () -> executeAsyncWithOrigin( client.threadPool().getThreadContext(), SECURITY_ORIGIN, updateRequest.request(), ActionListener.wrap(updateResponse -> { if (updateResponse.getResult() == DocWriteResponse.Result.UPDATED) { logger.debug( () -> format("updated the original token document to %s", updateResponse.getGetResult().sourceAsMap()) ); final Tuple parsedTokens = parseTokensFromDocument(tokenDoc.sourceAsMap(), null); final UserToken toRefreshUserToken = parsedTokens.v1(); createOAuth2Tokens( newTokenBytes.v1(), newTokenBytes.v2(), newTokenVersion, toRefreshUserToken.getAuthentication(), clientAuth, toRefreshUserToken.getMetadata(), listener ); } else if (backoff.hasNext()) { logger.info( "failed to update the original token document [{}], the update result was [{}]. Retrying", tokenDoc.id(), updateResponse.getResult() ); client.threadPool() .schedule( () -> innerRefresh(refreshToken, tokenDoc, clientAuth, backoff, refreshRequested, listener), backoff.next(), client.threadPool().generic() ); } else { logger.info( "failed to update the original token document [{}] after all retries, the update result was [{}]. ", tokenDoc.id(), updateResponse.getResult() ); listener.onFailure(invalidGrantException("could not refresh the requested token")); } }, e -> { Throwable cause = ExceptionsHelper.unwrapCause(e); if (cause instanceof VersionConflictEngineException) { // The document has been updated by another thread, get it again. logger.debug("version conflict while updating document [{}], attempting to get it again", tokenDoc.id()); getTokenDocAsync(tokenDoc.id(), refreshedTokenIndex, true, new ActionListener<>() { @Override public void onResponse(GetResponse response) { if (response.isExists()) { assert response.getId().equals(tokenDoc.id()); innerRefresh(refreshToken, new Doc(response), clientAuth, backoff, refreshRequested, listener); } else { logger.warn("could not find token document [{}] for refresh", tokenDoc.id()); onFailure.accept(invalidGrantException("could not refresh the requested token")); } } @Override public void onFailure(Exception e) { if (isShardNotAvailableException(e)) { if (backoff.hasNext()) { logger.info("could not get token document [{}] for refresh, retrying", tokenDoc.id()); client.threadPool() .schedule( () -> getTokenDocAsync(tokenDoc.id(), refreshedTokenIndex, true, this), backoff.next(), client.threadPool().generic() ); } else { logger.warn("could not get token document [{}] for refresh after all retries", tokenDoc.id()); onFailure.accept(invalidGrantException("could not refresh the requested token")); } } else { onFailure.accept(e); } } }); } else if (isShardNotAvailableException(e)) { if (backoff.hasNext()) { logger.debug("failed to update the original token document [{}], retrying", tokenDoc.id()); client.threadPool() .schedule( () -> innerRefresh(refreshToken, tokenDoc, clientAuth, backoff, refreshRequested, listener), backoff.next(), client.threadPool().generic() ); } else { logger.warn("failed to update the original token document [{}], after all retries", tokenDoc.id()); onFailure.accept(invalidGrantException("could not refresh the requested token")); } } else { onFailure.accept(e); } }), client::update ) ); } } /** * Decrypts the values of the superseding access token and the refresh token, using a key derived from the superseded refresh token. * It verifies that the token document for the access token it decrypted exists first, before calling the listener. It * encodes the version and serializes the tokens before calling the listener, in the same manner as {@link #createOAuth2Tokens } does. * * @param refreshToken The refresh token that the user sent in the request, used to derive the decryption key * @param refreshTokenStatus The {@link RefreshTokenStatus} containing information about the superseding tokens as retrieved from the * index * @param tokensIndex the manager for the index where the tokens are stored * @param authentication The authentication object representing the user for which the tokens are created * @param listener The listener to call upon completion with a {@link CreateTokenResult} containing the * serialized access token, serialized refresh token and authentication for which the token is created * as these will be returned to the client */ void decryptAndReturnSupersedingTokens( String refreshToken, RefreshTokenStatus refreshTokenStatus, SecurityIndexManager tokensIndex, Authentication authentication, ActionListener listener ) { final byte[] iv = Base64.getDecoder().decode(refreshTokenStatus.getIv()); final byte[] salt = Base64.getDecoder().decode(refreshTokenStatus.getSalt()); final byte[] encryptedSupersedingTokens = Base64.getDecoder().decode(refreshTokenStatus.getSupersedingTokens()); try { Cipher cipher = getDecryptionCipher(iv, refreshToken, salt); final String supersedingTokens = new String(cipher.doFinal(encryptedSupersedingTokens), StandardCharsets.UTF_8); final String[] decryptedTokens = supersedingTokens.split("\\|"); if (decryptedTokens.length != 2) { logger.warn("Decrypted tokens string is not correctly formatted"); listener.onFailure(invalidGrantException("could not refresh the requested token")); } else { // We expect this to protect against race conditions that manifest within few ms final Iterator backoff = BackoffPolicy.exponentialBackoff(TimeValue.timeValueMillis(10), 8).iterator(); final String tokenDocId; if (refreshTokenStatus.getTransportVersion().onOrAfter(VERSION_GET_TOKEN_DOC_FOR_REFRESH)) { MessageDigest userTokenIdDigest = sha256(); userTokenIdDigest.update( Base64.getUrlDecoder().decode(decryptedTokens[0]), RAW_TOKEN_BYTES_LENGTH, RAW_TOKEN_DOC_ID_BYTES_LENGTH ); tokenDocId = getTokenDocumentId(Base64.getUrlEncoder().withoutPadding().encodeToString(userTokenIdDigest.digest())); } else { tokenDocId = getTokenDocumentId(hashTokenString(decryptedTokens[0])); } final Consumer onFailure = ex -> listener.onFailure( traceLog("decrypt and get superseding token", tokenDocId, ex) ); final Consumer> maybeRetryGet = actionListener -> { if (backoff.hasNext()) { logger.info("could not get token document [{}] that should have been created, retrying", tokenDocId); client.threadPool() .schedule( () -> getTokenDocAsync(tokenDocId, tokensIndex, false, actionListener), backoff.next(), client.threadPool().generic() ); } else { logger.warn("could not get token document [{}] that should have been created after all retries", tokenDocId); onFailure.accept(invalidGrantException("could not refresh the requested token")); } }; getTokenDocAsync(tokenDocId, tokensIndex, false, new ActionListener<>() { @Override public void onResponse(GetResponse response) { if (response.isExists()) { try { logger.debug( "Found superseding document: index=[{}] id=[{}] primTerm=[{}] seqNo=[{}]", response.getIndex(), response.getId(), response.getPrimaryTerm(), response.getSeqNo() ); listener.onResponse( new CreateTokenResult( prependVersionAndEncodeAccessToken( refreshTokenStatus.getTransportVersion(), Base64.getUrlDecoder().decode(decryptedTokens[0]) ), prependVersionAndEncodeRefreshToken( refreshTokenStatus.getTransportVersion(), Base64.getUrlDecoder().decode(decryptedTokens[1]) ), authentication ) ); } catch (GeneralSecurityException | IOException e) { logger.warn("Could not format stored superseding token values", e); onFailure.accept(invalidGrantException("could not refresh the requested token")); } } else { maybeRetryGet.accept(this); } } @Override public void onFailure(Exception e) { if (isShardNotAvailableException(e)) { maybeRetryGet.accept(this); } else { onFailure.accept(e); } } }); } } catch (GeneralSecurityException e) { logger.warn("Could not get stored superseding token values", e); listener.onFailure(invalidGrantException("could not refresh the requested token")); } } /* * Encrypts the values of the superseding access token and the refresh token, using a key derived from the superseded refresh token. * The tokens are concatenated to a string separated with `|` before encryption so that we only perform one encryption operation * and that we only need to store one field */ String encryptSupersedingTokens( byte[] supersedingAccessTokenBytes, byte[] supersedingRefreshTokenBytes, String refreshToken, byte[] iv, byte[] salt ) throws GeneralSecurityException { Cipher cipher = getEncryptionCipher(iv, refreshToken, salt); final String supersedingAccessToken = Base64.getUrlEncoder().withoutPadding().encodeToString(supersedingAccessTokenBytes); final String supersedingRefreshToken = Base64.getUrlEncoder().withoutPadding().encodeToString(supersedingRefreshTokenBytes); final String supersedingTokens = supersedingAccessToken + "|" + supersedingRefreshToken; return Base64.getEncoder().encodeToString(cipher.doFinal(supersedingTokens.getBytes(StandardCharsets.UTF_8))); } private void getTokenDocAsync( String tokenDocId, SecurityIndexManager tokensIndex, boolean fetchSource, ActionListener listener ) { final GetRequest getRequest = client.prepareGet(tokensIndex.aliasName(), tokenDocId).setFetchSource(fetchSource).request(); tokensIndex.checkIndexVersionThenExecute( ex -> listener.onFailure(traceLog("prepare tokens index [" + tokensIndex.aliasName() + "]", tokenDocId, ex)), () -> executeAsyncWithOrigin(client.threadPool().getThreadContext(), SECURITY_ORIGIN, getRequest, listener, client::get) ); } // public for tests public TransportVersion getTokenVersionCompatibility() { // newly minted tokens are compatible with the min transport version in the cluster return clusterService.state().getMinTransportVersion(); } public static Boolean isTokenServiceEnabled(Settings settings) { return XPackSettings.TOKEN_SERVICE_ENABLED_SETTING.get(settings); } /** * A refresh token has a fixed maximum lifetime of {@code ExpiredTokenRemover#MAXIMUM_TOKEN_LIFETIME_HOURS} hours. This checks if the * token document represents a valid token wrt this time interval. */ private static Optional checkTokenDocumentExpired(Instant refreshRequested, Map src) { final Long creationEpochMilli = (Long) src.get("creation_time"); if (creationEpochMilli == null) { throw new IllegalStateException("token document is missing creation time value"); } else { final Instant creationTime = Instant.ofEpochMilli(creationEpochMilli); if (refreshRequested.isAfter(creationTime.plus(ExpiredTokenRemover.MAXIMUM_TOKEN_LIFETIME_HOURS, ChronoUnit.HOURS))) { return Optional.of(invalidGrantException("token document has expired")); } else { return Optional.empty(); } } } /** * Parses the {@code RefreshTokenStatus} from the token document and throws an exception if the document is malformed. Returns the * parsed {@code RefreshTokenStatus} together with an {@code Optional} validation exception that encapsulates the various logic about * when and by who a token can be refreshed. */ private static Tuple> checkTokenDocumentForRefresh( Instant refreshRequested, Authentication clientAuth, Map source ) throws IllegalStateException, DateTimeException { final RefreshTokenStatus refreshTokenStatus = parseTokenAndRefreshStatus(source).v2(); final ElasticsearchSecurityException validationException = checkTokenDocumentExpired(refreshRequested, source).orElseGet(() -> { if (refreshTokenStatus.isInvalidated()) { return invalidGrantException("token has been invalidated"); } else { return checkClientCanRefresh(refreshTokenStatus, clientAuth).orElse( checkMultipleRefreshes(refreshRequested, refreshTokenStatus).orElse(null) ); } }); return new Tuple<>(refreshTokenStatus, Optional.ofNullable(validationException)); } private static Tuple parseTokenAndRefreshStatus(Map source) { final RefreshTokenStatus refreshTokenStatus = RefreshTokenStatus.fromSourceMap(getRefreshTokenSourceMap(source)); final UserToken userToken = UserToken.fromSourceMap(getUserTokenSourceMap(source)); refreshTokenStatus.setTransportVersion(userToken.getTransportVersion()); return new Tuple<>(userToken, refreshTokenStatus); } /** * Refresh tokens are bound to be used only by the client that originally created them. This check validates this condition, given the * {@code Authentication} of the client that attempted the refresh operation. */ private static Optional checkClientCanRefresh( RefreshTokenStatus refreshToken, Authentication clientAuthentication ) { if (refreshToken.getAssociatedAuthentication() != null) { // this is the newer method to validate that the client refreshing the token indeed owns the token if (clientAuthentication.canAccessResourcesOf(refreshToken.getAssociatedAuthentication())) { return Optional.empty(); } else { logger.warn( "Token was originally created by [{}] but [{}] attempted to refresh it", refreshToken.getAssociatedAuthentication(), clientAuthentication ); return Optional.of(invalidGrantException("tokens must be refreshed by the creating client")); } } else { // falback to the previous method if (clientAuthentication.getEffectiveSubject().getUser().principal().equals(refreshToken.getAssociatedUser()) == false) { logger.warn( "Token was originally created by [{}] but [{}] attempted to refresh it", refreshToken.getAssociatedUser(), clientAuthentication.getEffectiveSubject().getUser().principal() ); return Optional.of(invalidGrantException("tokens must be refreshed by the creating client")); } else if (clientAuthentication.getAuthenticatingSubject() .getRealm() .getName() .equals(refreshToken.getAssociatedRealm()) == false) { logger.warn( "[{}] created the refresh token while authenticated by [{}] but is now authenticated by [{}]", refreshToken.getAssociatedUser(), refreshToken.getAssociatedRealm(), clientAuthentication.getAuthenticatingSubject().getRealm().getName() ); return Optional.of(invalidGrantException("tokens must be refreshed by the creating client")); } else { return Optional.empty(); } } } private static Map getRefreshTokenSourceMap(Map source) { @SuppressWarnings("unchecked") final Map refreshTokenSource = (Map) source.get("refresh_token"); if (refreshTokenSource == null || refreshTokenSource.isEmpty()) { throw new IllegalStateException("token document is missing the refresh_token object"); } return refreshTokenSource; } private static Map getUserTokenSourceMap(Map source) { @SuppressWarnings("unchecked") final Map accessTokenSource = (Map) source.get("access_token"); if (accessTokenSource == null || accessTokenSource.isEmpty()) { throw new IllegalStateException("token document is missing the access_token object"); } @SuppressWarnings("unchecked") final Map userTokenSource = (Map) accessTokenSource.get("user_token"); if (userTokenSource == null || userTokenSource.isEmpty()) { throw new IllegalStateException("token document is missing the user token info"); } return userTokenSource; } /** * Checks if the token can be refreshed once more. If a token has previously been refreshed, it can only by refreshed again inside a * short span of time (30 s). * * @return An {@code Optional} containing the exception in case this refresh token cannot be reused, or an empty Optional if * refreshing is allowed. */ private static Optional checkMultipleRefreshes( Instant refreshRequested, RefreshTokenStatus refreshTokenStatus ) { if (refreshTokenStatus.isRefreshed()) { if (refreshTokenStatus.getTransportVersion().onOrAfter(VERSION_MULTIPLE_CONCURRENT_REFRESHES)) { if (refreshRequested.isAfter(refreshTokenStatus.getRefreshInstant().plus(30L, ChronoUnit.SECONDS))) { return Optional.of(invalidGrantException("token has already been refreshed more than 30 seconds in the past")); } if (refreshRequested.isBefore(refreshTokenStatus.getRefreshInstant().minus(30L, ChronoUnit.SECONDS))) { return Optional.of( invalidGrantException("token has been refreshed more than 30 seconds in the future, clock skew too great") ); } } else { return Optional.of(invalidGrantException("token has already been refreshed")); } } return Optional.empty(); } /** * Search for refresh and access tokens that have not been invalidated or expired, and, optionally, that were issued against * the specified realm and pass the predicate test on the token doc source (also optional). * Note that searches on the tokens index is discouraged, so please avoid using this method, if possible. * * @param realmName The name of the realm for which to get the tokens * @param filter an optional Predicate to test the source of the found documents against * @param listener The listener to notify upon completion */ private void searchActiveTokens( @Nullable String realmName, @Nullable Predicate> filter, ActionListener>> listener ) { sourceIndicesWithTokensAndRun(ActionListener.wrap(indicesWithTokens -> { if (indicesWithTokens.isEmpty()) { listener.onResponse(Collections.emptyList()); } else { final Instant now = clock.instant(); final BoolQueryBuilder boolQuery = QueryBuilders.boolQuery().filter(QueryBuilders.termQuery("doc_type", TOKEN_DOC_TYPE)); if (Strings.isNullOrEmpty(realmName) == false) { boolQuery.filter(QueryBuilders.termQuery("access_token.realm", realmName)); } boolQuery.filter( QueryBuilders.boolQuery() .should( QueryBuilders.boolQuery() .must(QueryBuilders.termQuery("access_token.invalidated", false)) .must(QueryBuilders.rangeQuery("access_token.user_token.expiration_time").gte(now.toEpochMilli())) ) .should( QueryBuilders.boolQuery() .must(QueryBuilders.termQuery("refresh_token.invalidated", false)) .must( QueryBuilders.rangeQuery("creation_time") .gte( now.toEpochMilli() - TimeValue.timeValueHours(ExpiredTokenRemover.MAXIMUM_TOKEN_LIFETIME_HOURS) .millis() ) ) ) ); final Supplier supplier = client.threadPool().getThreadContext().newRestorableContext(false); try (ThreadContext.StoredContext ignore = client.threadPool().getThreadContext().stashWithOrigin(SECURITY_ORIGIN)) { // before searching, refresh the tokens index because tokens created/refreshed/invalidated post // version {@code #VERSION_GET_TOKEN_DOC_FOR_REFRESH} use {@code RefreshPolicy#NONE} client.admin() .indices() .prepareRefresh(securityTokensIndex.aliasName()) .execute(ActionListener.wrap(refreshResponse -> { if (refreshResponse.getFailedShards() > 0 && logger.isDebugEnabled()) { logger.debug("Failure to refresh tokens index {}", Arrays.toString(refreshResponse.getShardFailures())); } final SearchRequest request = client.prepareSearch(indicesWithTokens.toArray(new String[0])) .setScroll(DEFAULT_KEEPALIVE_SETTING.get(settings)) .setQuery(boolQuery) .setVersion(false) .setSize(1000) .setFetchSource(true) .request(); ScrollHelper.fetchAllByEntity( client, request, new ContextPreservingActionListener<>(supplier, listener), (SearchHit hit) -> filterAndParseHit(hit, filter) ); }, listener::onFailure)); } } }, listener::onFailure)); } /** * Security tokens were traditionally stored on the main security index but after version {@code #VERSION_TOKENS_INDEX_INTRODUCED} they * have been stored on a dedicated separate index. This move has been implemented without requiring user intervention, so the newly * created tokens started to be created in the new index, while the old tokens were still usable out of the main security index, subject * to their maximum lifetime of {@code ExpiredTokenRemover#MAXIMUM_TOKEN_LIFETIME_HOURS} hours. Once the dedicated tokens index has been * automatically created, all the onwards created tokens will be stored inside it. This function returns the list of the indices names * that might contain tokens. Unless there are availability or version issues, the dedicated tokens index always contains tokens. The * main security index might contain tokens if the tokens index has not been created yet, or if it has been created recently so * that there might still be tokens that have not yet exceeded their maximum lifetime. */ private void sourceIndicesWithTokensAndRun(ActionListener> listener) { final List indicesWithTokens = new ArrayList<>(2); final SecurityIndexManager frozenTokensIndex = securityTokensIndex.defensiveCopy(); if (frozenTokensIndex.indexExists()) { // an existing tokens index always contains tokens (if available and version allows) if (false == frozenTokensIndex.isAvailable(SEARCH_SHARDS)) { listener.onFailure(frozenTokensIndex.getUnavailableReason(SEARCH_SHARDS)); return; } if (false == frozenTokensIndex.isIndexUpToDate()) { listener.onFailure( new IllegalStateException( "Index [" + frozenTokensIndex.aliasName() + "] is not on the current version. Features relying on the index" + " will not be available until the upgrade API is run on the index" ) ); return; } indicesWithTokens.add(frozenTokensIndex.aliasName()); } final SecurityIndexManager frozenMainIndex = securityMainIndex.defensiveCopy(); if (frozenMainIndex.indexExists()) { // main security index _might_ contain tokens if the tokens index has been created recently if (false == frozenTokensIndex.indexExists() || frozenTokensIndex.getCreationTime() .isAfter(clock.instant().minus(ExpiredTokenRemover.MAXIMUM_TOKEN_LIFETIME_HOURS, ChronoUnit.HOURS))) { if (false == frozenMainIndex.isAvailable(SEARCH_SHARDS)) { listener.onFailure(frozenMainIndex.getUnavailableReason(SEARCH_SHARDS)); return; } if (false == frozenMainIndex.isIndexUpToDate()) { listener.onFailure( new IllegalStateException( "Index [" + frozenMainIndex.aliasName() + "] is not on the current version. Features relying on the index" + " will not be available until the upgrade API is run on the index" ) ); return; } indicesWithTokens.add(frozenMainIndex.aliasName()); } } listener.onResponse(indicesWithTokens); } private BytesReference createTokenDocument( UserToken userToken, @Nullable String accessTokenToStore, @Nullable String refreshTokenToStore, @Nullable Authentication originatingClientAuth ) { final Instant creationTime = getCreationTime(userToken.getExpirationTime()); return createTokenDocument(userToken, accessTokenToStore, refreshTokenToStore, originatingClientAuth, creationTime); } static BytesReference createTokenDocument( UserToken userToken, @Nullable String accessTokenToStore, @Nullable String refreshTokenToStore, @Nullable Authentication originatingClientAuth, Instant creationTime ) { assert refreshTokenToStore == null || originatingClientAuth != null : "non-null refresh token " + refreshTokenToStore + " requires non-null client authn " + originatingClientAuth; try (XContentBuilder builder = XContentFactory.jsonBuilder()) { builder.startObject(); builder.field("doc_type", TOKEN_DOC_TYPE); builder.field("creation_time", creationTime.toEpochMilli()); if (refreshTokenToStore != null) { builder.startObject("refresh_token") .field("token", refreshTokenToStore) .field("invalidated", false) .field("refreshed", false) .startObject("client") .field("type", "unassociated_client"); if (userToken.getTransportVersion().onOrAfter(VERSION_CLIENT_AUTH_FOR_REFRESH)) { builder.field( "authentication", originatingClientAuth.maybeRewriteForOlderVersion(userToken.getTransportVersion()).encode() ); } else { builder.field("user", originatingClientAuth.getEffectiveSubject().getUser().principal()) .field("realm", originatingClientAuth.getAuthenticatingSubject().getRealm().getName()); if (originatingClientAuth.getAuthenticatingSubject().getRealm().getDomain() != null) { builder.field("realm_domain", originatingClientAuth.getAuthenticatingSubject().getRealm().getDomain()); } } builder.endObject().endObject(); } final Authentication.RealmRef userTokenEffectiveRealm = userToken.getAuthentication().getEffectiveSubject().getRealm(); builder.startObject("access_token") .field("invalidated", false) .field("user_token", userToken) .field("realm", userTokenEffectiveRealm.getName()); if (userTokenEffectiveRealm.getDomain() != null) { builder.field("realm_domain", userTokenEffectiveRealm.getDomain()); } if (accessTokenToStore != null) { builder.field("token", accessTokenToStore); } builder.endObject().endObject(); return BytesReference.bytes(builder); } catch (IOException e) { throw new RuntimeException("Unexpected exception when constructing a JSON document.", e); } } private static Predicate> isOfUser(String username) { return source -> { String auth = (String) source.get("authentication"); Integer version = (Integer) source.get("version"); TransportVersion authVersion = TransportVersion.fromId(version); try (StreamInput in = StreamInput.wrap(Base64.getDecoder().decode(auth))) { in.setTransportVersion(authVersion); Authentication authentication = new Authentication(in); return authentication.getEffectiveSubject().getUser().principal().equals(username); } catch (IOException e) { throw new UncheckedIOException(e); } }; } private static Tuple filterAndParseHit(SearchHit hit, @Nullable Predicate> filter) throws IllegalStateException, DateTimeException { final Map source = hit.getSourceAsMap(); if (source == null) { throw new IllegalStateException("token document did not have source but source should have been fetched"); } return parseTokensFromDocument(source, filter); } /** * Parses a token document into a Tuple of a {@link UserToken} and a String representing the corresponding refresh_token * * @param source The token document source as retrieved * @param filter an optional Predicate to test the source of the UserToken against * @return A {@link Tuple} of access-token and refresh-token-id or null if a Predicate is defined and the userToken source doesn't * satisfy it */ private static Tuple parseTokensFromDocument( Map source, @Nullable Predicate> filter ) throws IllegalStateException, DateTimeException { @SuppressWarnings("unchecked") final Map refreshTokenMap = (Map) source.get("refresh_token"); final String hashedRefreshToken = refreshTokenMap != null ? (String) refreshTokenMap.get("token") : null; @SuppressWarnings("unchecked") final Map userTokenSource = (Map) ((Map) source.get("access_token")).get( "user_token" ); if (null != filter && filter.test(userTokenSource) == false) { return null; } final UserToken userToken = UserToken.fromSourceMap(userTokenSource); return new Tuple<>(userToken, hashedRefreshToken); } private static String getTokenDocumentId(UserToken userToken) { return getTokenDocumentId(userToken.getId()); } private static String getTokenDocumentId(String id) { return TOKEN_DOC_ID_PREFIX + id; } private static String getTokenIdFromDocumentId(String docId) { if (docId.startsWith(TOKEN_DOC_ID_PREFIX) == false) { throw new IllegalStateException("TokenDocument ID [" + docId + "] has unexpected value"); } else { return docId.substring(TOKEN_DOC_ID_PREFIX.length()); } } private boolean shouldTryRealm() { // Check license without tracking because this is just checking if we should *try* the realm - if this realm doesn't match, // the next realm in the list will be checked, and that's not "using the feature" return enabled && Security.TOKEN_SERVICE_FEATURE.checkWithoutTracking(licenseState); } private void ensureEnabled() { if (Security.TOKEN_SERVICE_FEATURE.check(licenseState) == false) { throw LicenseUtils.newComplianceException("security tokens"); } if (enabled == false) { throw new FeatureNotEnabledException(Feature.TOKEN_SERVICE, "security tokens are not enabled"); } } /** * In version {@code #VERSION_TOKENS_INDEX_INTRODUCED} security tokens were moved into a separate index, away from the other entities in * the main security index, due to their ephemeral nature. They moved "seamlessly" - without manual user intervention. In this way, new * tokens are created in the new index, while the existing ones were left in place - to be accessed from the old index - and due to be * removed automatically by the {@code ExpiredTokenRemover} periodic job. Therefore, in general, when searching for a token we need to * consider both the new and the old indices. */ private SecurityIndexManager getTokensIndexForVersion(TransportVersion version) { if (version.onOrAfter(VERSION_TOKENS_INDEX_INTRODUCED)) { return securityTokensIndex; } else { return securityMainIndex; } } public TimeValue getExpirationDelay() { return expirationDelay; } Instant getExpirationTime() { return clock.instant().plusSeconds(expirationDelay.getSeconds()); } private Instant getCreationTime(Instant expire) { return expire.minusSeconds(expirationDelay.getSeconds()); } private void maybeStartTokenRemover() { if (client.threadPool().relativeTimeInMillis() - lastExpirationRunMs > deleteInterval.getMillis()) { expiredTokenRemover.submit(client.threadPool()); lastExpirationRunMs = client.threadPool().relativeTimeInMillis(); } } public String prependVersionAndEncodeAccessToken(TransportVersion version, byte[] accessTokenBytes) throws IOException, GeneralSecurityException { if (version.onOrAfter(VERSION_GET_TOKEN_DOC_FOR_REFRESH)) { try (BytesStreamOutput out = new BytesStreamOutput(VERSION_BYTES + RAW_TOKEN_BYTES_TOTAL_LENGTH)) { out.setTransportVersion(version); TransportVersion.writeVersion(version, out); out.writeByteArray(accessTokenBytes); return Base64.getEncoder().encodeToString(out.bytes().toBytesRef().bytes); } } else if (version.onOrAfter(VERSION_ACCESS_TOKENS_AS_UUIDS)) { try (BytesStreamOutput out = new BytesStreamOutput(MINIMUM_BASE64_BYTES)) { out.setTransportVersion(version); TransportVersion.writeVersion(version, out); out.writeString(Base64.getUrlEncoder().withoutPadding().encodeToString(accessTokenBytes)); return Base64.getEncoder().encodeToString(out.bytes().toBytesRef().bytes); } } else { // we know that the minimum length is larger than the default of the ByteArrayOutputStream so set the size to this explicitly try ( ByteArrayOutputStream os = new ByteArrayOutputStream(LEGACY_MINIMUM_BASE64_BYTES); OutputStream base64 = Base64.getEncoder().wrap(os); StreamOutput out = new OutputStreamStreamOutput(base64) ) { out.setTransportVersion(version); KeyAndCache keyAndCache = keyCache.activeKeyCache; TransportVersion.writeVersion(version, out); out.writeByteArray(keyAndCache.getSalt().bytes); out.writeByteArray(keyAndCache.getKeyHash().bytes); final byte[] initializationVector = getRandomBytes(IV_BYTES); out.writeByteArray(initializationVector); try ( CipherOutputStream encryptedOutput = new CipherOutputStream( out, getEncryptionCipher(initializationVector, keyAndCache, version) ); StreamOutput encryptedStreamOutput = new OutputStreamStreamOutput(encryptedOutput) ) { encryptedStreamOutput.setTransportVersion(version); encryptedStreamOutput.writeString(Base64.getUrlEncoder().withoutPadding().encodeToString(accessTokenBytes)); // StreamOutput needs to be closed explicitly because it wraps CipherOutputStream encryptedStreamOutput.close(); return new String(os.toByteArray(), StandardCharsets.UTF_8); } } } } public static String prependVersionAndEncodeRefreshToken(TransportVersion version, byte[] refreshTokenBytes) throws IOException { if (version.onOrAfter(VERSION_GET_TOKEN_DOC_FOR_REFRESH)) { try (BytesStreamOutput out = new BytesStreamOutput(VERSION_BYTES + RAW_TOKEN_BYTES_TOTAL_LENGTH)) { out.setTransportVersion(version); TransportVersion.writeVersion(version, out); out.writeByteArray(refreshTokenBytes); return Base64.getEncoder().encodeToString(out.bytes().toBytesRef().bytes); } } else { try (BytesStreamOutput out = new BytesStreamOutput(VERSION_BYTES + TOKEN_LENGTH)) { out.setTransportVersion(version); TransportVersion.writeVersion(version, out); out.writeString(Base64.getUrlEncoder().withoutPadding().encodeToString(refreshTokenBytes)); return Base64.getEncoder().encodeToString(out.bytes().toBytesRef().bytes); } } } private static void ensureEncryptionCiphersSupported() throws NoSuchPaddingException, NoSuchAlgorithmException { Cipher.getInstance(ENCRYPTION_CIPHER); SecretKeyFactory.getInstance(KDF_ALGORITHM); } // Package private for testing Cipher getEncryptionCipher(byte[] iv, KeyAndCache keyAndCache, TransportVersion version) throws GeneralSecurityException { Cipher cipher = Cipher.getInstance(ENCRYPTION_CIPHER); BytesKey salt = keyAndCache.getSalt(); try { cipher.init(Cipher.ENCRYPT_MODE, keyAndCache.getOrComputeKey(salt), new GCMParameterSpec(128, iv), secureRandom); } catch (ExecutionException e) { throw new ElasticsearchSecurityException("Failed to compute secret key for active salt", e); } cipher.updateAAD(ByteBuffer.allocate(4).putInt(version.id()).array()); cipher.updateAAD(salt.bytes); return cipher; } /** * Initialize the encryption cipher using the provided password to derive the encryption key. */ Cipher getEncryptionCipher(byte[] iv, String password, byte[] salt) throws GeneralSecurityException { SecretKey key = computeSecretKey(password.toCharArray(), salt, TOKENS_ENCRYPTION_KEY_ITERATIONS); Cipher cipher = Cipher.getInstance(ENCRYPTION_CIPHER); cipher.init(Cipher.ENCRYPT_MODE, key, new GCMParameterSpec(128, iv), secureRandom); cipher.updateAAD(salt); return cipher; } private void getKeyAsync(BytesKey decodedSalt, KeyAndCache keyAndCache, ActionListener listener) { final SecretKey decodeKey = keyAndCache.getKey(decodedSalt); if (decodeKey != null) { listener.onResponse(decodeKey); } else { /* As a measure of protected against DOS, we can pass requests requiring a key * computation off to a single thread executor. For normal usage, the initial * request(s) that require a key computation will be delayed and there will be * some additional latency. */ client.threadPool().executor(THREAD_POOL_NAME).submit(new KeyComputingRunnable(decodedSalt, keyAndCache, listener)); } } private static String decryptTokenId(byte[] encryptedTokenId, Cipher cipher, TransportVersion version) throws IOException { try ( ByteArrayInputStream bais = new ByteArrayInputStream(encryptedTokenId); CipherInputStream cis = new CipherInputStream(bais, cipher); StreamInput decryptedInput = new InputStreamStreamInput(cis) ) { decryptedInput.setTransportVersion(version); return decryptedInput.readString(); } } private Cipher getDecryptionCipher(byte[] iv, SecretKey key, TransportVersion version, BytesKey salt) throws GeneralSecurityException { Cipher cipher = Cipher.getInstance(ENCRYPTION_CIPHER); cipher.init(Cipher.DECRYPT_MODE, key, new GCMParameterSpec(128, iv), secureRandom); cipher.updateAAD(ByteBuffer.allocate(4).putInt(version.id()).array()); cipher.updateAAD(salt.bytes); return cipher; } /** * Initialize the decryption cipher using the provided password to derive the decryption key. */ private Cipher getDecryptionCipher(byte[] iv, String password, byte[] salt) throws GeneralSecurityException { SecretKey key = computeSecretKey(password.toCharArray(), salt, TOKENS_ENCRYPTION_KEY_ITERATIONS); Cipher cipher = Cipher.getInstance(ENCRYPTION_CIPHER); cipher.init(Cipher.DECRYPT_MODE, key, new GCMParameterSpec(128, iv), secureRandom); cipher.updateAAD(salt); return cipher; } // public to be used in tests public Tuple getRandomTokenBytes(boolean includeRefreshToken) { return getRandomTokenBytes(getTokenVersionCompatibility(), includeRefreshToken); } Tuple getRandomTokenBytes(TransportVersion version, boolean includeRefreshToken) { if (version.onOrAfter(VERSION_GET_TOKEN_DOC_FOR_REFRESH)) { byte[] accessTokenBytes = getRandomBytes(RAW_TOKEN_BYTES_TOTAL_LENGTH); if (includeRefreshToken) { byte[] refreshTokenBytes = new byte[RAW_TOKEN_BYTES_TOTAL_LENGTH]; System.arraycopy(getRandomBytes(RAW_TOKEN_BYTES_LENGTH), 0, refreshTokenBytes, 0, RAW_TOKEN_BYTES_LENGTH); // access and refresh tokens are paired by having a common fixed-length suffix System.arraycopy( accessTokenBytes, RAW_TOKEN_BYTES_LENGTH, refreshTokenBytes, RAW_TOKEN_BYTES_LENGTH, RAW_TOKEN_DOC_ID_BYTES_LENGTH ); return new Tuple<>(accessTokenBytes, refreshTokenBytes); } else { return new Tuple<>(accessTokenBytes, null); } } else { if (includeRefreshToken) { return new Tuple<>(getRandomBytes(RAW_TOKEN_BYTES_LENGTH), getRandomBytes(RAW_TOKEN_BYTES_LENGTH)); } else { return new Tuple<>(getRandomBytes(RAW_TOKEN_BYTES_LENGTH), null); } } } byte[] getRandomBytes(int length) { final byte[] bytes = new byte[length]; secureRandom.nextBytes(bytes); return bytes; } /** * Generates a secret key based off of the provided password and salt. * This method can be computationally expensive. */ static SecretKey computeSecretKey(char[] rawPassword, byte[] salt, int iterations) throws NoSuchAlgorithmException, InvalidKeySpecException { SecretKeyFactory secretKeyFactory = SecretKeyFactory.getInstance(KDF_ALGORITHM); PBEKeySpec keySpec = new PBEKeySpec(rawPassword, salt, iterations, 128); SecretKey tmp = secretKeyFactory.generateSecret(keySpec); return new SecretKeySpec(tmp.getEncoded(), "AES"); } /** * Creates an {@link ElasticsearchSecurityException} that indicates the token was expired. It * is up to the client to re-authenticate and obtain a new token. The format for this response * is defined in */ private static ElasticsearchSecurityException expiredTokenException() { ElasticsearchSecurityException e = new ElasticsearchSecurityException("token expired", RestStatus.UNAUTHORIZED); e.addHeader("WWW-Authenticate", EXPIRED_TOKEN_WWW_AUTH_VALUE); return e; } /** * Creates an {@link ElasticsearchSecurityException} that indicates the request contained an invalid grant */ private static ElasticsearchSecurityException invalidGrantException(String detail) { ElasticsearchSecurityException e = new ElasticsearchSecurityException("invalid_grant", RestStatus.BAD_REQUEST); e.addHeader("error_description", detail); return e; } private static ElasticsearchSecurityException unableToPerformAction(@Nullable Throwable cause) { return new ElasticsearchSecurityException("unable to perform requested action", RestStatus.SERVICE_UNAVAILABLE, cause); } /** * Logs an exception concerning a specific Token at TRACE level (if enabled) */ private static E traceLog(String action, String identifier, E exception) { if (logger.isTraceEnabled()) { if (exception instanceof final ElasticsearchException esEx) { final Object detail = esEx.getHeader("error_description"); if (detail != null) { logger.trace(() -> format("Failure in [%s] for id [%s] - [%s]", action, identifier, detail), esEx); } else { logger.trace(() -> format("Failure in [%s] for id [%s]", action, identifier), esEx); } } else { logger.trace(() -> format("Failure in [%s] for id [%s]", action, identifier), exception); } } return exception; } /** * Logs an exception at TRACE level (if enabled) */ private static E traceLog(String action, E exception) { if (logger.isTraceEnabled()) { if (exception instanceof final ElasticsearchException esEx) { final Object detail = esEx.getHeader("error_description"); if (detail != null) { logger.trace(() -> format("Failure in [%s] - [%s]", action, detail), esEx); } else { logger.trace(() -> "Failure in [" + action + "]", esEx); } } else { logger.trace(() -> "Failure in [" + action + "]", exception); } } return exception; } static boolean isExpiredTokenException(ElasticsearchSecurityException e) { final List headers = e.getHeader("WWW-Authenticate"); return headers != null && headers.stream().anyMatch(EXPIRED_TOKEN_WWW_AUTH_VALUE::equals); } boolean isExpirationInProgress() { return expiredTokenRemover.isExpirationInProgress(); } public static final class CreateTokenResult { private final String accessToken; private final String refreshToken; private final Authentication authentication; public CreateTokenResult(String accessToken, String refreshToken, Authentication authentication) { this.accessToken = accessToken; this.refreshToken = refreshToken; this.authentication = authentication; } public String getAccessToken() { return accessToken; } public String getRefreshToken() { return refreshToken; } public Authentication getAuthentication() { return authentication; } } private static class KeyComputingRunnable extends AbstractRunnable { private final BytesKey decodedSalt; private final KeyAndCache keyAndCache; private final ActionListener listener; KeyComputingRunnable(BytesKey decodedSalt, KeyAndCache keyAndCache, ActionListener listener) { this.decodedSalt = decodedSalt; this.keyAndCache = keyAndCache; this.listener = listener; } @Override protected void doRun() { try { final SecretKey computedKey = keyAndCache.getOrComputeKey(decodedSalt); listener.onResponse(computedKey); } catch (ExecutionException e) { if (e.getCause() != null && (e.getCause() instanceof GeneralSecurityException || e.getCause() instanceof IOException || e.getCause() instanceof IllegalArgumentException)) { // this could happen if another realm supports the Bearer token so we should // see if another realm can use this token! logger.debug("unable to decode bearer token", e); listener.onResponse(null); } else { listener.onFailure(e); } } } @Override public void onFailure(Exception e) { listener.onFailure(e); } } /** * Returns the current in-use metdata of this {@link TokenService} */ public synchronized TokenMetadata getTokenMetadata() { return newTokenMetadata(keyCache.currentTokenKeyHash, keyCache.cache.values()); } private static TokenMetadata newTokenMetadata(BytesKey activeTokenKey, Iterable iterable) { List list = new ArrayList<>(); for (KeyAndCache v : iterable) { list.add(v.keyAndTimestamp); } return new TokenMetadata(list, activeTokenKey.bytes); } /** * Refreshes the current in-use metadata. */ synchronized void refreshMetadata(TokenMetadata metadata) { BytesKey currentUsedKeyHash = new BytesKey(metadata.getCurrentKeyHash()); byte[] saltArr = new byte[SALT_BYTES]; Map map = Maps.newMapWithExpectedSize(metadata.getKeys().size()); long maxTimestamp = createdTimeStamps.get(); for (KeyAndTimestamp key : metadata.getKeys()) { secureRandom.nextBytes(saltArr); KeyAndCache keyAndCache = new KeyAndCache(key, new BytesKey(saltArr)); maxTimestamp = Math.max(keyAndCache.keyAndTimestamp.getTimestamp(), maxTimestamp); if (keyCache.cache.containsKey(keyAndCache.getKeyHash()) == false) { map.put(keyAndCache.getKeyHash(), keyAndCache); } else { map.put(keyAndCache.getKeyHash(), keyCache.get(keyAndCache.getKeyHash())); // maintain the cache we already have } } if (map.containsKey(currentUsedKeyHash) == false) { // this won't leak any secrets it's only exposing the current set of hashes throw new IllegalStateException("Current key is not in the map: " + map.keySet() + " key: " + currentUsedKeyHash); } createdTimeStamps.set(maxTimestamp); keyCache = new TokenKeys(Collections.unmodifiableMap(map), currentUsedKeyHash); logger.debug(() -> format("refreshed keys current: %s, keys: %s", currentUsedKeyHash, keyCache.cache.keySet())); } private SecureString generateTokenKey() { byte[] keyBytes = new byte[KEY_BYTES]; byte[] encode = new byte[0]; char[] ref = new char[0]; try { secureRandom.nextBytes(keyBytes); encode = Base64.getUrlEncoder().withoutPadding().encode(keyBytes); ref = new char[encode.length]; int len = UnicodeUtil.UTF8toUTF16(encode, 0, encode.length, ref); return new SecureString(Arrays.copyOfRange(ref, 0, len)); } finally { Arrays.fill(keyBytes, (byte) 0x00); Arrays.fill(encode, (byte) 0x00); Arrays.fill(ref, (char) 0x00); } } @SuppressForbidden(reason = "legacy usage of unbatched task") // TODO add support for batching here private void submitUnbatchedTask(@SuppressWarnings("SameParameterValue") String source, ClusterStateUpdateTask task) { clusterService.submitUnbatchedStateUpdateTask(source, task); } private void initialize(ClusterService clusterService) { clusterService.addListener(event -> { ClusterState state = event.state(); if (state.getBlocks().hasGlobalBlock(STATE_NOT_RECOVERED_BLOCK)) { return; } if (state.nodes().isLocalNodeElectedMaster()) { if (XPackPlugin.isReadyForXPackCustomMetadata(state)) { installTokenMetadata(state); } else { logger.debug( "cannot add token metadata to cluster as the following nodes might not understand the metadata: {}", () -> XPackPlugin.nodesNotReadyForXPackCustomMetadata(state) ); } } TokenMetadata custom = event.state().custom(TokenMetadata.TYPE); if (custom != null && custom.equals(getTokenMetadata()) == false) { logger.info("refresh keys"); try { refreshMetadata(custom); } catch (Exception e) { logger.warn("refreshing metadata failed", e); } logger.info("refreshed keys"); } }); } // to prevent too many cluster state update tasks to be queued for doing the same update private final AtomicBoolean installTokenMetadataInProgress = new AtomicBoolean(false); private void installTokenMetadata(ClusterState state) { if (state.custom(TokenMetadata.TYPE) == null) { if (installTokenMetadataInProgress.compareAndSet(false, true)) { submitUnbatchedTask("install-token-metadata", new ClusterStateUpdateTask(Priority.URGENT) { @Override public ClusterState execute(ClusterState currentState) { XPackPlugin.checkReadyForXPackCustomMetadata(currentState); if (currentState.custom(TokenMetadata.TYPE) == null) { return ClusterState.builder(currentState).putCustom(TokenMetadata.TYPE, getTokenMetadata()).build(); } else { return currentState; } } @Override public void onFailure(Exception e) { installTokenMetadataInProgress.set(false); logger.error("unable to install token metadata", e); } @Override public void clusterStateProcessed(ClusterState oldState, ClusterState newState) { installTokenMetadataInProgress.set(false); } }); } } } /** * Package private for testing */ void clearActiveKeyCache() { this.keyCache.activeKeyCache.keyCache.invalidateAll(); } static final class KeyAndCache implements Closeable { private final KeyAndTimestamp keyAndTimestamp; private final Cache keyCache; private final BytesKey salt; private final BytesKey keyHash; private KeyAndCache(KeyAndTimestamp keyAndTimestamp, BytesKey salt) { this.keyAndTimestamp = keyAndTimestamp; keyCache = CacheBuilder.builder() .setExpireAfterAccess(TimeValue.timeValueMinutes(60L)) .setMaximumWeight(500L) .build(); try { SecretKey secretKey = computeSecretKey(keyAndTimestamp.getKey().getChars(), salt.bytes, TOKEN_SERVICE_KEY_ITERATIONS); keyCache.put(salt, secretKey); } catch (Exception e) { throw new IllegalStateException(e); } this.salt = salt; this.keyHash = calculateKeyHash(keyAndTimestamp.getKey()); } private SecretKey getKey(BytesKey salt) { return keyCache.get(salt); } public SecretKey getOrComputeKey(BytesKey decodedSalt) throws ExecutionException { return keyCache.computeIfAbsent(decodedSalt, (salt) -> { try (SecureString closeableChars = keyAndTimestamp.getKey().clone()) { return computeSecretKey(closeableChars.getChars(), salt.bytes, TOKEN_SERVICE_KEY_ITERATIONS); } }); } @Override public void close() { keyAndTimestamp.getKey().close(); } BytesKey getKeyHash() { return keyHash; } private static BytesKey calculateKeyHash(SecureString key) { MessageDigest messageDigest = sha256(); BytesRefBuilder b = new BytesRefBuilder(); try { b.copyChars(key); BytesRef bytesRef = b.toBytesRef(); try { messageDigest.update(bytesRef.bytes, bytesRef.offset, bytesRef.length); return new BytesKey(Arrays.copyOfRange(messageDigest.digest(), 0, 8)); } finally { Arrays.fill(bytesRef.bytes, (byte) 0x00); } } finally { Arrays.fill(b.bytes(), (byte) 0x00); } } BytesKey getSalt() { return salt; } } private static final class TokenKeys { final Map cache; final BytesKey currentTokenKeyHash; final KeyAndCache activeKeyCache; private TokenKeys(Map cache, BytesKey currentTokenKeyHash) { this.cache = cache; this.currentTokenKeyHash = currentTokenKeyHash; this.activeKeyCache = cache.get(currentTokenKeyHash); } KeyAndCache get(BytesKey passphraseHash) { return cache.get(passphraseHash); } } /** * Contains metadata associated with the refresh token that is used for validity checks, but does not contain the proper token string. */ static final class RefreshTokenStatus { private final boolean invalidated; private final String associatedUser; private final String associatedRealm; @Nullable private final Authentication associatedAuthentication; private final boolean refreshed; @Nullable private final Instant refreshInstant; @Nullable private final String supersedingTokens; @Nullable private final String iv; @Nullable private final String salt; private TransportVersion version; // pkg-private for testing RefreshTokenStatus( boolean invalidated, Authentication associatedAuthentication, boolean refreshed, Instant refreshInstant, String supersedingTokens, String iv, String salt ) { assert associatedAuthentication.getEffectiveSubject().getTransportVersion().onOrAfter(VERSION_CLIENT_AUTH_FOR_REFRESH); this.invalidated = invalidated; // not used, filled-in for consistency's sake this.associatedUser = associatedAuthentication.getEffectiveSubject().getUser().principal(); this.associatedRealm = associatedAuthentication.getAuthenticatingSubject().getRealm().getName(); this.associatedAuthentication = associatedAuthentication; this.refreshed = refreshed; this.refreshInstant = refreshInstant; this.supersedingTokens = supersedingTokens; this.iv = iv; this.salt = salt; } @Deprecated RefreshTokenStatus( boolean invalidated, String associatedUser, String associatedRealm, boolean refreshed, Instant refreshInstant, String supersedingTokens, String iv, String salt ) { this.invalidated = invalidated; this.associatedUser = associatedUser; this.associatedRealm = associatedRealm; this.associatedAuthentication = null; this.refreshed = refreshed; this.refreshInstant = refreshInstant; this.supersedingTokens = supersedingTokens; this.iv = iv; this.salt = salt; } boolean isInvalidated() { return invalidated; } String getAssociatedUser() { return associatedUser; } String getAssociatedRealm() { return associatedRealm; } Authentication getAssociatedAuthentication() { return associatedAuthentication; } boolean isRefreshed() { return refreshed; } @Nullable Instant getRefreshInstant() { return refreshInstant; } @Nullable String getSupersedingTokens() { return supersedingTokens; } @Nullable String getIv() { return iv; } @Nullable String getSalt() { return salt; } TransportVersion getTransportVersion() { return version; } void setTransportVersion(TransportVersion version) { this.version = version; } static RefreshTokenStatus fromSourceMap(Map refreshTokenSource) { final Boolean invalidated = (Boolean) refreshTokenSource.get("invalidated"); if (invalidated == null) { throw new IllegalStateException("token document is missing the \"invalidated\" field"); } @SuppressWarnings("unchecked") final Map clientInfo = (Map) refreshTokenSource.get("client"); if (clientInfo == null) { throw new IllegalStateException("token document is missing the \"client\" field"); } final Boolean refreshed = (Boolean) refreshTokenSource.get("refreshed"); if (refreshed == null) { throw new IllegalStateException("token document is missing the \"refreshed\" field"); } final Long refreshEpochMilli = (Long) refreshTokenSource.get("refresh_time"); final Instant refreshInstant = refreshEpochMilli == null ? null : Instant.ofEpochMilli(refreshEpochMilli); final String supersedingTokens = (String) refreshTokenSource.get("superseding.encrypted_tokens"); final String iv = (String) refreshTokenSource.get("superseding.encryption_iv"); final String salt = (String) refreshTokenSource.get("superseding.encryption_salt"); if (clientInfo.containsKey("authentication")) { // newer refresh token that contains the full authentication of the client that created it final Authentication associatedAuthentication; try { associatedAuthentication = AuthenticationContextSerializer.decode((String) clientInfo.get("authentication")); } catch (IOException e) { throw new IllegalStateException("invalid client authentication for refresh", e); } if (clientInfo.containsKey("user") || clientInfo.containsKey("realm")) { throw new IllegalStateException("user and/or associated realm must not be present when associated authentication is"); } return new RefreshTokenStatus( invalidated, associatedAuthentication, refreshed, refreshInstant, supersedingTokens, iv, salt ); } else { final String associatedUser; final String associatedRealm; // the previous way that only recorded selected fields of the creator client's authentication if (false == clientInfo.containsKey("user")) { throw new IllegalStateException( "token document must contain the \"client.user\" field if the associated " + "\"authentication\" field does not exist" ); } associatedUser = (String) clientInfo.get("user"); if (false == clientInfo.containsKey("realm")) { throw new IllegalStateException( "token document must contain the \"client.realm\" field if the associated " + "\"authentication\" field does not exist" ); } associatedRealm = (String) clientInfo.get("realm"); return new RefreshTokenStatus( invalidated, associatedUser, associatedRealm, refreshed, refreshInstant, supersedingTokens, iv, salt ); } } } record Doc(String id, Map sourceAsMap, long seqNo, long primaryTerm) { Doc(SearchHit searchHit) { this(searchHit.getId(), searchHit.getSourceAsMap(), searchHit.getSeqNo(), searchHit.getPrimaryTerm()); } Doc(GetResponse getResponse) { this(getResponse.getId(), getResponse.getSource(), getResponse.getSeqNo(), getResponse.getPrimaryTerm()); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy