org.apache.pulsar.client.impl.auth.oauth2.AuthenticationOAuth2 Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.pulsar.client.impl.auth.oauth2;
import java.io.IOException;
import java.time.Clock;
import java.time.Instant;
import java.util.Map;
import lombok.Data;
import lombok.extern.slf4j.Slf4j;
import org.apache.pulsar.shade.org.apache.commons.lang.NotImplementedException;
import org.apache.pulsar.shade.org.apache.commons.lang.StringUtils;
import org.apache.pulsar.client.api.Authentication;
import org.apache.pulsar.client.api.AuthenticationDataProvider;
import org.apache.pulsar.client.api.EncodedAuthenticationParameterSupport;
import org.apache.pulsar.client.api.PulsarClientException;
import org.apache.pulsar.client.impl.AuthenticationUtil;
import org.apache.pulsar.client.impl.auth.oauth2.protocol.TokenResult;
/**
* Pulsar client authentication provider based on OAuth 2.0.
*/
@Slf4j
public class AuthenticationOAuth2 implements Authentication, EncodedAuthenticationParameterSupport {
public static final String CONFIG_PARAM_TYPE = "type";
public static final String TYPE_CLIENT_CREDENTIALS = "client_credentials";
public static final String AUTH_METHOD_NAME = "token";
public static final double EXPIRY_ADJUSTMENT = 0.9;
private static final long serialVersionUID = 1L;
final Clock clock;
Flow flow;
transient CachedToken cachedToken;
public AuthenticationOAuth2() {
this.clock = Clock.systemDefaultZone();
}
AuthenticationOAuth2(Flow flow, Clock clock) {
this.flow = flow;
this.clock = clock;
}
@Override
public String getAuthMethodName() {
return AUTH_METHOD_NAME;
}
@Override
public void configure(String encodedAuthParamString) {
if (StringUtils.isBlank(encodedAuthParamString)) {
throw new IllegalArgumentException("No authentication parameters were provided");
}
Map params;
try {
params = AuthenticationUtil.configureFromJsonString(encodedAuthParamString);
} catch (IOException e) {
throw new IllegalArgumentException("Malformed authentication parameters", e);
}
String type = params.getOrDefault(CONFIG_PARAM_TYPE, TYPE_CLIENT_CREDENTIALS);
switch(type) {
case TYPE_CLIENT_CREDENTIALS:
this.flow = ClientCredentialsFlow.fromParameters(params);
break;
default:
throw new IllegalArgumentException("Unsupported authentication type: " + type);
}
}
@Override
@Deprecated
public void configure(Map authParams) {
throw new NotImplementedException("Deprecated; use EncodedAuthenticationParameterSupport");
}
@Override
public void start() throws PulsarClientException {
flow.initialize();
}
@Override
public synchronized AuthenticationDataProvider getAuthData() throws PulsarClientException {
if (this.cachedToken == null || this.cachedToken.isExpired()) {
TokenResult tr = this.flow.authenticate();
this.cachedToken = new CachedToken(tr);
}
return this.cachedToken.getAuthData();
}
@Override
public void close() throws IOException {
try {
flow.close();
} catch (Exception e) {
throw new IOException(e);
}
}
@Data
class CachedToken {
private final TokenResult latest;
private final Instant expiresAt;
private final AuthenticationDataOAuth2 authData;
public CachedToken(TokenResult latest) {
this.latest = latest;
int adjustedExpiresIn = (int) (latest.getExpiresIn() * EXPIRY_ADJUSTMENT);
this.expiresAt = AuthenticationOAuth2.this.clock.instant().plusSeconds(adjustedExpiresIn);
this.authData = new AuthenticationDataOAuth2(latest.getAccessToken());
}
public boolean isExpired() {
return AuthenticationOAuth2.this.clock.instant().isAfter(this.expiresAt);
}
}
}