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

org.frankframework.filesystem.MsalClientAdapter Maven / Gradle / Ivy

The newest version!
/*
   Copyright 2022 WeAreFrank!

   Licensed 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.frankframework.filesystem;

import java.io.IOException;
import java.net.URI;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import com.microsoft.aad.msal4j.HttpRequest;
import com.microsoft.aad.msal4j.IHttpClient;
import com.microsoft.aad.msal4j.IHttpResponse;

import org.apache.commons.lang3.NotImplementedException;
import org.apache.http.Header;
import org.apache.http.HttpEntity;
import org.apache.http.client.methods.HttpEntityEnclosingRequestBase;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.methods.HttpRequestBase;
import org.apache.logging.log4j.Logger;
import org.frankframework.core.PipeLineSession;
import org.frankframework.core.SenderException;
import org.frankframework.doc.Protected;
import org.frankframework.http.HttpMessageEntity;
import org.frankframework.http.HttpResponseHandler;
import org.frankframework.http.HttpSenderBase;
import org.frankframework.parameters.Parameter;
import org.frankframework.parameters.ParameterValueList;
import org.frankframework.stream.Message;
import org.frankframework.util.EnumUtils;
import org.frankframework.util.LogUtil;

/**
 * This class ensures that Microsoft Authentication Library (MSAL) requests are sent through the configured proxy and the correct SSLSocketFactory.
 * @see ExchangeFileSystem
 *
 */
@Protected
public class MsalClientAdapter extends HttpSenderBase implements IHttpClient {
	private static final String METHOD_SESSION_KEY = "HTTP_METHOD";
	private static final String REQUEST_HEADERS_SESSION_KEY = "HTTP_REQUEST_HEADERS";
	private static final String RESPONSE_HEADERS_SESSION_KEY = "HTTP_RESPONSE_HEADERS";
	private static final String URL_SESSION_KEY = "URL";
	private static final String STATUS_CODE_SESSION_KEY = "HTTP_STATUSCODE";

	public MsalClientAdapter() {
		setName("MSAL Autentication Sender");

		Parameter urlParameter = new Parameter();
		urlParameter.setName("url");
		urlParameter.setSessionKey(URL_SESSION_KEY);
		addParameter(urlParameter);

		setResultStatusCodeSessionKey(STATUS_CODE_SESSION_KEY);
	}

	@Override
	public IHttpResponse send(HttpRequest httpRequest) throws Exception {
		PipeLineSession session = prepareSession(httpRequest);
		Message request = new Message(httpRequest.body());

		try {
			Message response = sendMessageOrThrow(request, session);
			return new MsalResponse(response, session);
		} catch (Exception e) {
			log.error("An exception occurred whilst connecting with MSAL HTTPS call to [{}]", httpRequest.url().toString(), e);
			throw e;
		} finally {
			session.close();
		}
	}

	private PipeLineSession prepareSession(HttpRequest httpRequest) {
		PipeLineSession session = new PipeLineSession();
		session.put(URL_SESSION_KEY, httpRequest.url().toString());
		log.debug("Put request URL [{}] in session under key [{}]", ()->httpRequest.url().toString(), ()->URL_SESSION_KEY);

		session.put(METHOD_SESSION_KEY, httpRequest.httpMethod().name());
		log.debug("Put http method [{}] in session under key [{}]", ()->httpRequest.httpMethod().name(), ()->METHOD_SESSION_KEY);

		session.put(REQUEST_HEADERS_SESSION_KEY, httpRequest.headers());
		if(log.isDebugEnabled()) log.debug("Put http headers [{}] in session under key [{}]", httpRequest::headers, ()->REQUEST_HEADERS_SESSION_KEY);

		return session;
	}

	@Override
	protected HttpRequestBase getMethod(URI uri, Message message, ParameterValueList parameters, PipeLineSession session) throws SenderException {
		HttpMethod httpMethod = EnumUtils.parse(HttpMethod.class, (String) session.get(METHOD_SESSION_KEY));
		Map headers = (Map) session.get(REQUEST_HEADERS_SESSION_KEY);

		if(uri == null) {
			throw new SenderException("No URI to connect to!");
		}

		switch (httpMethod) {
		case GET:
			HttpGet getMethod = new HttpGet(uri.toString());

			return appendHeaders(headers, getMethod);
		case POST:
			HttpEntityEnclosingRequestBase method = new HttpPost(uri.toString());
			HttpEntity entity = new HttpMessageEntity(message); //No need to set the content-type, MSAL sets it to application/soap+xml later on

			method.setEntity(entity);
			return appendHeaders(headers, method);
		default:
				throw new NotImplementedException("method ["+httpMethod+"] has not been implemented");
		}
	}

	@Override
	protected Message extractResult(HttpResponseHandler responseHandler, PipeLineSession session) {
		// Parse headers to comma separated string
		Header[] responseHeaders = responseHandler.getAllHeaders();
		String headers = "";
		for(Header header : responseHeaders) {
			session.put(header.getName(), header.getValue());
			if(headers.length() > 0) {
				headers += ",";
			}
			headers += header.getName();
		}
		session.put(RESPONSE_HEADERS_SESSION_KEY, headers);

		return responseHandler.getResponseMessage();
	}

	private HttpRequestBase appendHeaders(Map headers, HttpRequestBase request) {
		for(Map.Entry header : headers.entrySet()) {
			String name = header.getKey();
			String value = header.getValue();
			request.setHeader(name, value);
			log.debug("Appending header [{}] [{}]", name, value);
		}
		return request;
	}

	private class MsalResponse implements IHttpResponse {
		protected Logger log = LogUtil.getLogger(this);

		private final int statusCode;
		private final Map> headers = new HashMap<>();
		private String body = "";

		public MsalResponse(Message response, PipeLineSession session) {
			this.statusCode = Integer.parseInt((String) session.get(STATUS_CODE_SESSION_KEY));
			if(log.isDebugEnabled())
				log.debug("Parsing status code [{}]", statusCode);

			String[] headersAsCsv = ((String) session.get(RESPONSE_HEADERS_SESSION_KEY)).split(",");
			for (String headerName : headersAsCsv) {
				List values = new ArrayList<>();
				String headerValue = (String) session.get(headerName);
				values.add(headerValue);

				if(log.isDebugEnabled())
					log.debug("Parsing header [{}] [{}]", headerName, headerValue);
				this.headers.put(headerName, values);
			}

			try {
				log.debug("Parsing body [{}]", response::toString);
				this.body = response.asString();
			} catch (IOException e) {
				log.error("An exception occurred whilst parsing the response body of MSAL authentication call.", e);
			}
		}

		@Override
		public int statusCode() {
			return statusCode;
		}

		@Override
		public Map> headers() {
			return headers;
		}

		@Override
		public String body() {
			return body;
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy