com.github.davidmoten.microsoft.authentication.ClientCredentialsAccessTokenProvider Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of odata-client-microsoft-client-builder Show documentation
Show all versions of odata-client-microsoft-client-builder Show documentation
Builds client for Microsoft Graph 1.0 and Beta, includes MsGraph authentication support
package com.github.davidmoten.microsoft.authentication;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.net.InetSocketAddress;
import java.net.Proxy;
import java.net.URL;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.Base64;
import java.util.Date;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.TimeUnit;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import javax.net.ssl.HttpsURLConnection;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.github.davidmoten.guavamini.Preconditions;
import com.github.davidmoten.guavamini.annotations.VisibleForTesting;
import com.github.davidmoten.odata.client.ClientException;
import com.github.davidmoten.odata.client.internal.Util;
public final class ClientCredentialsAccessTokenProvider implements AccessTokenProvider {
private static final Logger log = LoggerFactory
.getLogger(ClientCredentialsAccessTokenProvider.class);
private static final int OK = 200;
private static final String POST = "POST";
private static final String APPLICATION_JSON = "application/json";
private static final String REQUEST_HEADER = "Accept";
private static final String GRANT_TYPE_CLIENT_CREDENTIALS = "client_credentials";
private static final String OAUTH2_TOKEN_URL_SUFFIX = "/oauth2/token";
private static final String PARAMETER_SCOPE = "scope";
private static final String PARAMETER_CLIENT_SECRET = "client_secret";
private static final String PARAMETER_GRANT_TYPE = "grant_type";
private static final String PARAMETER_CLIENT_ID = "client_id";
private static final String PARAMETER_RESOURCE = "resource";
private final String tenantName;
private final String clientId;
private final String clientSecret;
private final long refreshBeforeExpiryMs;
private final long connectTimeoutMs;
private final long readTimeoutMs;
private final List scopes;
private final String resource;
private final Optional proxyHost;
private final Optional proxyPort;
private final Optional proxyUsername;
private final Optional proxyPassword;
private final String authenticationEndpoint;
private long expiryTime;
private String accessToken;
private ClientCredentialsAccessTokenProvider(String tenantName, String clientId,
String clientSecret, long refreshBeforeExpiryMs, long connectTimeoutMs,
long readTimeoutMs, String authenticationEndpoint, String resource, List scopes,
Optional proxyHost, //
Optional proxyPort, //
Optional proxyUsername, Optional proxyPassword) {
Preconditions.checkNotNull(tenantName);
Preconditions.checkNotNull(clientId);
Preconditions.checkNotNull(clientSecret);
Preconditions.checkArgument(refreshBeforeExpiryMs >= 0,
"refreshBeforeExpiryMs must be >=0");
Preconditions.checkArgument(connectTimeoutMs >= 0, "connectTimeoutMs must be >=0");
Preconditions.checkArgument(readTimeoutMs >= 0, "readTimeoutMs must be >=0");
Preconditions.checkNotNull(authenticationEndpoint);
Preconditions.checkNotNull(proxyHost);
Preconditions.checkNotNull(proxyPort);
Preconditions.checkNotNull(proxyUsername);
Preconditions.checkNotNull(proxyPassword);
Preconditions.checkArgument(!proxyHost.isPresent() || proxyPort.isPresent(),
"if proxyHost specified then so must proxyPort be specified");
Preconditions.checkNotNull(resource);
Preconditions.checkNotNull(scopes);
this.tenantName = tenantName;
this.clientId = clientId;
this.clientSecret = clientSecret;
this.refreshBeforeExpiryMs = refreshBeforeExpiryMs;
this.connectTimeoutMs = connectTimeoutMs;
this.readTimeoutMs = readTimeoutMs;
this.authenticationEndpoint = authenticationEndpoint;
this.scopes = scopes;
this.resource = resource;
this.proxyHost = proxyHost;
this.proxyPort = proxyPort;
this.proxyUsername = proxyUsername;
this.proxyPassword = proxyPassword;
}
public static Builder tenantName(String tenantName) {
return new Builder(tenantName);
}
@Override
public synchronized String get() {
long now = System.currentTimeMillis();
if (accessToken != null && now < expiryTime - refreshBeforeExpiryMs) {
return accessToken;
} else {
return refreshAccessToken();
}
}
private String refreshAccessToken() {
// post some parameters in json format to the access token url
// and record returned expiry information so they we know when we
// need to refresh the token
try {
log.debug("refreshing access token");
URL url = new URL(authenticationEndpoint + tenantName + OAUTH2_TOKEN_URL_SUFFIX);
final HttpsURLConnection con;
if (proxyHost.isPresent()) {
InetSocketAddress proxyInet = new InetSocketAddress(proxyHost.get(),
proxyPort.get());
Proxy proxy = new Proxy(Proxy.Type.HTTP, proxyInet);
con = (HttpsURLConnection) url.openConnection(proxy);
if (proxyUsername.isPresent()) {
String usernameAndPassword = proxyUsername.get() + ":" + proxyPassword.get();
String authString = "Basic " + Base64.getEncoder()
.encodeToString(usernameAndPassword.getBytes(StandardCharsets.UTF_8));
con.setRequestProperty("Proxy-Authorization", authString);
}
// TODO support NTLM?
} else {
con = (HttpsURLConnection) url.openConnection();
}
con.setConnectTimeout((int) connectTimeoutMs);
con.setReadTimeout((int) readTimeoutMs);
con.setRequestMethod(POST);
con.setRequestProperty(REQUEST_HEADER, APPLICATION_JSON);
StringBuilder params = new StringBuilder();
add(params, PARAMETER_RESOURCE, resource);
add(params, PARAMETER_CLIENT_ID, clientId);
add(params, PARAMETER_GRANT_TYPE, GRANT_TYPE_CLIENT_CREDENTIALS);
add(params, PARAMETER_CLIENT_SECRET, clientSecret);
for (String scope : scopes) {
add(params, PARAMETER_SCOPE, scope);
}
con.setDoOutput(true);
try (DataOutputStream dos = new DataOutputStream(con.getOutputStream())) {
dos.writeBytes(params.toString());
}
int responseCode = con.getResponseCode();
String json = Util.readString(con.getInputStream(), StandardCharsets.UTF_8);
if (responseCode != OK) {
throw new ClientException(responseCode, json);
} else {
ObjectMapper om = new ObjectMapper();
JsonNode o = om.readTree(json);
// update the cached values
expiryTime = o.get("expires_on").asLong() * 1000;
accessToken = o.get("access_token").asText();
log.debug("refreshed access token, expires on " + new Date(expiryTime));
return accessToken;
}
} catch (IOException e) {
// reset stuff
expiryTime = 0;
accessToken = null;
Optional code = extractStatusCode(e.getMessage());
if (code.isPresent()) {
throw new ClientException(code.get(), e);
} else {
throw new ClientException(e);
}
}
}
private static final Pattern RESPONSE_CODE_PATTERN = Pattern.compile("^Server returned HTTP response code: (\\d+) for.*");
@VisibleForTesting
static Optional extractStatusCode(String message) {
if (message == null) {
return Optional.empty();
}
Matcher m = RESPONSE_CODE_PATTERN.matcher(message);
if (m.find()) {
String code = m.group(1);
return Optional.of(Integer.parseInt(code));
} else {
return Optional.empty();
}
}
private static void add(StringBuilder params, String key, String value) {
if (params.length() > 0) {
params.append("&");
}
params.append(key);
params.append("=");
try {
params.append(URLEncoder.encode(value, "UTF-8"));
} catch (UnsupportedEncodingException e) {
throw new RuntimeException(e);
}
}
public static final class Builder {
final String tenantName;
List scopes = new ArrayList<>();
String resource;
String clientId;
String clientSecret;
// default to refresh access token on every call of get()
long refreshBeforeExpiryMs = Long.MAX_VALUE;
long connectTimeoutMs = TimeUnit.SECONDS.toMillis(30);
long readTimeoutMs = TimeUnit.SECONDS.toMillis(30);
String endpoint = AuthenticationEndpoint.GLOBAL.url();
Optional proxyHost = Optional.empty();
Optional proxyPort = Optional.empty();
Optional proxyUsername = Optional.empty();
Optional proxyPassword = Optional.empty();
Builder(String tenantName) {
this.tenantName = tenantName;
}
public Builder2 resource(String resource) {
this.resource = resource;
return new Builder2(this);
}
}
public static final class Builder2 {
private final Builder b;
Builder2(Builder b) {
this.b = b;
}
public Builder3 scope(List scopes) {
Preconditions.checkNotNull(scopes);
b.scopes.addAll(scopes);
return new Builder3(b);
}
}
public static final class Builder3 {
private final Builder b;
public Builder3(Builder b) {
this.b = b;
}
public Builder4 clientId(String clientId) {
b.clientId = clientId;
return new Builder4(b);
}
}
public static final class Builder4 {
private final Builder b;
Builder4(Builder b) {
this.b = b;
}
public Builder5 clientSecret(String clientSecret) {
b.clientSecret = clientSecret;
return new Builder5(b);
}
}
public static final class Builder5 {
private final Builder b;
Builder5(Builder b) {
this.b = b;
}
/**
* The access token is returned from AD with an expiry time. If you call
* {@code get()} within {@code duration} of the expiry time then a refresh of
* the access token will be performed. If this value is not set then the access
* token is refreshed on every call of {@code get()}.
*
* @param duration duration before expiry time after which point a refresh will
* be run (on next authentication attempt)
* @param unit time unit for the duration
* @return builder
*/
public Builder5 refreshBeforeExpiry(long duration, TimeUnit unit) {
b.refreshBeforeExpiryMs = unit.toMillis(duration);
return this;
}
public Builder5 connectTimeoutMs(long duration, TimeUnit unit) {
b.connectTimeoutMs = unit.toMillis(duration);
return this;
}
public Builder5 readTimeoutMs(long duration, TimeUnit unit) {
b.readTimeoutMs = unit.toMillis(duration);
return this;
}
public Builder5 scope(List scopes) {
b.scopes.addAll(scopes);
return this;
}
/**
* Default value is {@link AuthenticationEndpoint#GLOBAL}.
*
* @param endpoint microsoft authentication endpoint
* @return this
*/
public Builder5 authenticationEndpoint(AuthenticationEndpoint endpoint) {
b.endpoint = endpoint.url();
return this;
}
public Builder5 authenticationEndpoint(String endpoint) {
b.endpoint = endpoint;
return this;
}
public Builder5 proxyHost(String proxyHost) {
Preconditions.checkNotNull(proxyHost);
return proxyHost(Optional.of(proxyHost));
}
public Builder5 proxyPort(int proxyPort) {
return proxyPort(Optional.of(proxyPort));
}
public Builder5 proxyUsername(String username) {
Preconditions.checkNotNull(username);
return proxyUsername (Optional.of(username));
}
public Builder5 proxyPassword(String password) {
Preconditions.checkNotNull(password);
return proxyPassword(Optional.of(password));
}
public Builder5 proxyHost(Optional proxyHost) {
Preconditions.checkNotNull(proxyHost);
b.proxyHost = proxyHost;
return this;
}
public Builder5 proxyPort(Optional proxyPort) {
Preconditions.checkNotNull(proxyPort);
b.proxyPort = proxyPort;
return this;
}
public Builder5 proxyUsername(Optional username) {
Preconditions.checkNotNull(username);
b.proxyUsername = username;
return this;
}
public Builder5 proxyPassword(Optional password) {
Preconditions.checkNotNull(password);
b.proxyPassword = password;
return this;
}
public ClientCredentialsAccessTokenProvider build() {
return new ClientCredentialsAccessTokenProvider(b.tenantName, b.clientId,
b.clientSecret, b.refreshBeforeExpiryMs, b.connectTimeoutMs, b.readTimeoutMs,
b.endpoint, b.resource, b.scopes, b.proxyHost, b.proxyPort, b.proxyUsername,
b.proxyPassword);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy