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

com.vmware.pscoe.iac.artifact.rest.auth.VraSsoAuth Maven / Gradle / Ivy

There is a newer version: 3.1.1
Show newest version
/*
 * #%L
 * artifact-manager
 * %%
 * Copyright (C) 2023 VMware
 * %%
 * Build Tools for VMware Aria
 * Copyright 2023 VMware, Inc.
 * 
 * This product is licensed to you under the BSD-2 license (the "License"). You may not use this product except in compliance with the BSD-2 License.  
 * 
 * This product may include a number of subcomponents with separate copyright notices and license terms. Your use of these subcomponents is subject to the terms and conditions of the subcomponent's license, as noted in the LICENSE file.
 * #L%
 */
package com.vmware.pscoe.iac.artifact.rest.auth;

import java.net.URI;
import java.util.HashMap;
import java.util.Map;

import com.jayway.jsonpath.Configuration;
import com.jayway.jsonpath.Option;
import net.minidev.json.JSONArray;
import org.springframework.http.HttpEntity;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpMethod;
import org.springframework.http.MediaType;
import org.springframework.http.ResponseEntity;
import org.springframework.util.LinkedMultiValueMap;
import org.springframework.util.MultiValueMap;
import org.springframework.util.StringUtils;
import org.springframework.web.client.RestTemplate;
import org.springframework.web.util.UriComponentsBuilder;

import com.google.gson.Gson;
import com.jayway.jsonpath.DocumentContext;
import com.jayway.jsonpath.JsonPath;
import com.vmware.pscoe.iac.artifact.configuration.ConfigurationVro;
import com.vmware.pscoe.iac.artifact.configuration.ConfigurationVro.AuthProvider;
import com.vmware.pscoe.iac.artifact.rest.model.sso.SsoEndpointDto;

public class VraSsoAuth {
	/** CAFE_CLI_OWNER. */
	private static final String CAFE_CLI_OWNER = "CAFE_CLI_OWNER";
	/** Authorication endpoint for vRA 8.x. */
	private static final String AUTHORIZATION_SERVICE_URL_VRA_CLOUD = "/csp/gateway/am/api/auth/api-tokens/authorize";
	/** Login endpoint for vRA 8.x . */
	private static final String AUTHORIZATION_SERVICE_URL_VRA_8 = "/csp/gateway/am/api/login";
	/** Authentication endpoint for vRA 7.x . */
	private static final String AUTHORIZATION_SERVICE_URL_VRA_7 = "/SAAS/t/%s/auth/oauthtoken";
	/** Single Sign On endpont for vRA 7.x . */
	private static final String COMPONENT_REGISTRY_URL_VRA_7 = "/component-registry/endpoints/types/sso";
	/** Endpont used to get the version. */
	private static final String VERSION_URL = "vco/api/about";
	/** Protocol scheme. */
	private static final String DEFAULT_HTTP_SCHEME = "https";
	/** CSP Authentication token name. */
	private static final String TOKEN_NAME = "cspAuthToken";
	/** Bearer token type. */
	private static final String DEFAULT_TOKEN_TYPE = "Bearer";
	/** Version prefix for vRA versions 8.x. */
	private static final String VRA_8_VERSION_PREFIX = "8.";
	/** Http Entity. */
	private static final String HTTP_ENTITY_TYPE = "httpEntity";
	/** Token URI. */
	private static final String TOKEN_URL_TYPE = "tokenUri";

	/** vRO Config. */
	private ConfigurationVro vroConfig;
	/** REST Tempalate to be used for communicating with the REST API of the. */
	private RestTemplate restTemplate;
	/** Aria Automation version. */
	private String serverVersion;

	/**
	 * Create new instance of the class that is using the orivided configuraiton and
	 * REST Template.
	 * 
	 * @param config       Configuration on how to connect to Aria Automation.
	 * @param restTemplate REST Temaplate to be used to issue REST API calls.
	 */
	public VraSsoAuth(ConfigurationVro config, RestTemplate restTemplate) {
		this.vroConfig = config;
		this.restTemplate = restTemplate;
		this.serverVersion = this.getVersion();
	}

	/**
	 * Accuire authentication token from Aria Automation.
	 * 
	 * @return The token returned by Aria Automation.
	 */
	public SsoToken acquireToken() {
		return acquireToken(null);
	}

	/**
	 * Accuire authentcation token from Aria Automation using the specified client
	 * id.
	 * 
	 * @param clientId The client id.
	 * @return The token obtained from Aria Automation.
	 */
	public SsoToken acquireToken(String clientId) {
		final Map tokenUriAndHttpEntity;
		final boolean isTokenAuth = this.serverVersion.endsWith("-saas-enabled")
				|| !StringUtils.isEmpty(this.vroConfig.getRefreshToken());

		if (isTokenAuth) {
			tokenUriAndHttpEntity = getTokenUriAndHttpEntityVraCloud();
		} else {
			tokenUriAndHttpEntity = getTokenUriAndHttpEntityVra8();
		}

		final URI tokenUri = (URI) tokenUriAndHttpEntity.get(TOKEN_URL_TYPE);
		final HttpEntity httpEntity = (HttpEntity) tokenUriAndHttpEntity.get(HTTP_ENTITY_TYPE);
		try {
			final ResponseEntity response = this.restTemplate.exchange(tokenUri, HttpMethod.POST, httpEntity,
					String.class);

			final DocumentContext responseBody = JsonPath.parse(response.getBody());

			final String tokenValue = isTokenAuth
					? responseBody.read("$.access_token")
					: responseBody.read("$." + TOKEN_NAME);

			return new SsoToken(tokenValue, AuthProvider.VRA);
		} catch (Exception e) {
			throw new RuntimeException(String.format(
					"Unable to acquire token for VRO SSO authentication: %s. Request was %s %s. Request body not shown due to sensitive data.",
					e.getMessage(), HttpMethod.POST, tokenUri));
		}
	}

	/**
	 * Obtain and return an authentication URL type and entity type from Aria
	 * Automation 8.x.
	 * This will only work with Aria Automation verison 8.x of Aria Automation.
	 * 
	 * @return A map that is containing the Token URL type (TOKEN_URL_TYPE) and the
	 *         entity type (HTTP_ENTITY_TYPE).
	 */
	public Map getTokenUriAndHttpEntityVra8() {
		final URI tokenUri = UriComponentsBuilder.newInstance()
				.scheme(DEFAULT_HTTP_SCHEME)
				.host(this.vroConfig.getAuthHost())
				.port(this.vroConfig.getAuthPort())
				.path(AUTHORIZATION_SERVICE_URL_VRA_8)
				.queryParam("tenant", this.vroConfig.getTenant())
				.queryParam(TOKEN_NAME, "").build().toUri();

		// Prepare request payload data
		final Map payload = new HashMap<>();
		payload.put("username", this.vroConfig.getUsername());
		payload.put("password", this.vroConfig.getPassword());
		payload.put("domain", this.vroConfig.getDomain());

		Gson gsonObject = new Gson();
		String stringPayload = gsonObject.toJson(payload);

		final HttpHeaders headers = new HttpHeaders();
		headers.setContentType(MediaType.APPLICATION_JSON_UTF8);

		HttpEntity entity = new HttpEntity<>(stringPayload, headers);

		final Map tokenUriAndHttpEntity = new HashMap<>();
		tokenUriAndHttpEntity.put(TOKEN_URL_TYPE, tokenUri);
		tokenUriAndHttpEntity.put(HTTP_ENTITY_TYPE, entity);

		return tokenUriAndHttpEntity;
	}

	/**
	 * Obtain and return an authentication URL type and entity type from Aria
	 * Automation Cloud.
	 * This will only work with Aria Automation Cloud.
	 * 
	 * @return A map that is containing the Token URL type (TOKEN_URL_TYPE) and the
	 *         entity type (HTTP_ENTITY_TYPE).
	 */
	public Map getTokenUriAndHttpEntityVraCloud() {
		final URI tokenUri = UriComponentsBuilder.newInstance()
				.scheme(DEFAULT_HTTP_SCHEME)
				.host(this.vroConfig.getAuthHost())
				.port(this.vroConfig.getAuthPort())
				.path(AUTHORIZATION_SERVICE_URL_VRA_CLOUD).build().toUri();

		final HttpHeaders headers = new HttpHeaders();
		headers.setContentType(MediaType.APPLICATION_FORM_URLENCODED);

		final MultiValueMap map = new LinkedMultiValueMap<>();
		map.add("api_token", this.vroConfig.getRefreshToken());

		HttpEntity> entity = new HttpEntity<>(map, headers);

		final Map tokenUriAndHttpEntity = new HashMap<>();
		tokenUriAndHttpEntity.put(TOKEN_URL_TYPE, tokenUri);
		tokenUriAndHttpEntity.put(HTTP_ENTITY_TYPE, entity);

		return tokenUriAndHttpEntity;
	}

	/**
	 * Retrieve SSO Client Id.
	 * 
	 * @return The client id.
	 */
	public String retrieveClientId() {
		final URI tokenUri = UriComponentsBuilder.newInstance()
				.scheme(DEFAULT_HTTP_SCHEME)
				.host(this.vroConfig.getAuthHost())
				.port(this.vroConfig.getAuthPort())
				.path(COMPONENT_REGISTRY_URL_VRA_7).build().toUri();

		final SsoEndpointDto ssoEndpoint = this.restTemplate.getForObject(tokenUri, SsoEndpointDto.class);

		return ssoEndpoint.getEndPointAttributes().stream().filter(attr -> CAFE_CLI_OWNER.equals(attr.getKey()))
				.findFirst()
				.orElseThrow(() -> new RuntimeException("Could not find the SSO Client ID")).getValue();
	}

	/**
	 * Retrive vRA version.
	 * 
	 * @return the strig representation of the version.
	 */
	private String getVersion() {
		// vRA Cloud doesn't have vRO services and it's auth host
		// (console.cloud.vmware.com)
		// is different than the Extensibility Proxy's vRO address
		// (ext-proxy01.corp.local),
		// hence the /vco/api/about is not available on the 'authHost' address, but on
		// the 'host' address
		final URI versionUri = UriComponentsBuilder.newInstance()
				.scheme(DEFAULT_HTTP_SCHEME)
				.host(this.vroConfig.getHost())
				.port(this.vroConfig.getPort())
				.path(VERSION_URL).build().toUri();

		ResponseEntity response = restTemplate.exchange(versionUri, HttpMethod.GET, getDefaultHttpEntity(),
				String.class);

		Configuration conf = Configuration.defaultConfiguration();
		conf.addOptions(Option.DEFAULT_PATH_LEAF_TO_NULL);
		conf.addOptions(Option.SUPPRESS_EXCEPTIONS);
		DocumentContext jsonDoc = JsonPath.using(conf).parse(response.getBody());
		JSONArray saasCheck = jsonDoc.read("$.features[?(@.name == 'saas-enabled')].name");

		String version = jsonDoc.read("$.version");
		if (version != null && !saasCheck.isEmpty()) {
			version += "-saas-enabled";
		}

		return version;
	}

	private HttpEntity getDefaultHttpEntity() {
		HttpHeaders headers = new HttpHeaders();

		headers.setContentType(MediaType.APPLICATION_JSON_UTF8);

		return new HttpEntity<>(headers);
	}

	/**
	 * Single Signe On token class.
	 */
	public static class SsoToken {
		/** The token value. */
		private String value;
		/** The token type, such as "Bearer". */
		private String tokenType;
		/** Authentication provider. */
		private AuthProvider type;

		/**
		 * Create a new Single Sign On token based on the given token string and
		 * Authentication Provider.
		 * 
		 * @param token Toek string.
		 * @param type  Authentication provider.
		 */
		public SsoToken(String token, AuthProvider type) {
			this.value = token;
			this.type = type;
			this.tokenType = DEFAULT_TOKEN_TYPE;
		}

		/**
		 * Return the token string.
		 * 
		 * @return the token string.
		 */
		public String getValue() {
			return value;
		}

		/**
		 * Return true if expired and false otherwise.
		 * 
		 * @return true if expired and false otherwise.
		 */
		public boolean isExpired() {
			return Boolean.TRUE;
		}

		/**
		 * Return the authentication provider type.
		 * 
		 * @return the authentication provider type.
		 */
		public AuthProvider getType() {
			return type;
		}

		/**
		 * The toekn type such as Bearer.
		 * 
		 * @return toekn type such as Bearer.
		 */
		public String getTokenType() {
			return tokenType;
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy