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

com.microsoft.azure.iot.service.transport.http.HttpConnection Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) Microsoft. All rights reserved.
 * Licensed under the MIT license. See LICENSE file in the project root for full license information.
 */

package com.microsoft.azure.iot.service.transport.http;

import javax.net.ssl.HttpsURLConnection;
import java.io.IOException;
import java.io.InputStream;
import java.net.ProtocolException;
import java.net.URL;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Map;

/**
 * A wrapper for the Java SE class HttpsURLConnection. Used to avoid
 * compatibility issues when testing with the mocking framework JMockit, as well
 * as to avoid some undocumented side effects when using HttpsURLConnection.
 * 

* The underlying HttpsURLConnection is transparently managed by Java. To reuse * connections, for each time connect() is called, the input streams (input * stream or error stream, if input stream is not accessible) must be completely * read. Otherwise, the data remains in the stream and the connection will not * be reusable. */ public class HttpConnection { /** The underlying HTTPS connection. */ protected final HttpsURLConnection connection; /** * The body. HttpsURLConnection silently calls connect() when the output * stream is written to. We buffer the body and defer writing to the output * stream until connect() is called. */ protected byte[] body; /** * Constructor. Opens a connection to the given URL. * * @param url The URL for the HTTPS connection. * @param method The HTTPS method (i.e. GET). * * @throws IOException This exception is thrown if the connection was unable to be opened. */ public HttpConnection(URL url, HttpMethod method) throws IOException { // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_004: [If the URI given does not use the HTTPS protocol, the constructor shall throw an IllegalArgumentException.] String protocol = url.getProtocol(); if (!protocol.equalsIgnoreCase("HTTPS")) { String errMsg = String.format("Expected URL that uses iotHubServiceClientProtocol " + "HTTPS but received one that uses " + "iotHubServiceClientProtocol '%s'.\n", protocol); throw new IllegalArgumentException(errMsg); } // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_001: [The constructor shall open a connection to the given URL.] // Coses_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_002: [The constructor shall throw an IOException if the connection was unable to be opened.] this.connection = (HttpsURLConnection) url.openConnection(); // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_003: [The constructor shall set the HTTPS method to the given method.] this.connection.setRequestMethod(method.name()); this.body = new byte[0]; } /** * Sends the request to the URL given in the constructor. * * @throws IOException This exception thrown if the connection could not be established, * or the server responded with a bad status code. */ public void connect() throws IOException { // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_006: [The function shall stream the request body, if present, through the connection.] if (this.body.length > 0) { this.connection.setDoOutput(true); this.connection.getOutputStream().write(this.body); } // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_005: [The function shall send a request to the URL given in the constructor.] // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_007: [The function shall throw an IOException if the connection could not be established, or the server responded with a bad status code.] this.connection.connect(); } /** * Sets the request method (i.e. POST). * * @param method The request method. * * @throws IllegalArgumentException This exception thrown if the request currently * has a non-empty body and the new method is not a POST or a PUT. This is because Java's * HttpsURLConnection silently converts the HTTPS method to POST or PUT if a * body is written to the request. */ public void setRequestMethod(HttpMethod method) { // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_009: [The function shall throw an IllegalArgumentException if the request currently has a non-empty body and the new method is not a POST or a PUT.] if (method != HttpMethod.POST && method != HttpMethod.PUT) { if (this.body.length > 0) { throw new IllegalArgumentException( "Cannot change the request method from POST " + "or PUT when the request body is non-empty."); } } // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_008: [The function shall set the request method.] try { this.connection.setRequestMethod(method.name()); } catch (ProtocolException e) { // should never happen, since the method names are hard-coded. } } /** * Sets the request header field to the given value. * * @param field The header field name. * @param value The header field value. */ public void setRequestHeader(String field, String value) { // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_010: [The function shall set the given request header field.] this.connection.setRequestProperty(field, value); } /** * Sets the read timeout in milliseconds. The read timeout is the number of * milliseconds after the server receives a request and before the server * sends data back. * * @param timeout The read timeout. */ public void setReadTimeoutMillis(int timeout) { // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_011: [The function shall set the read timeout to the given value.] this.connection.setReadTimeout(timeout); } /** * Saves the body to be sent with the request. * * @param body The request body. * * @throws IllegalArgumentException if the request does not currently use * method POST or PUT and the body is non-empty. This is because Java's * HttpsURLConnection silently converts the HTTPS method to POST or PUT if a * body is written to the request. */ public void writeOutput(byte[] body) { // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_013: [The function shall throw an IllegalArgumentException if the request does not currently use method POST or PUT and the body is non-empty.] HttpMethod method = HttpMethod.valueOf( this.connection.getRequestMethod()); if (method != HttpMethod.POST && method != HttpMethod.PUT) { if (body.length > 0) { throw new IllegalArgumentException( "Cannot write a body to a request that " + "is not a POST or a PUT request."); } } else { // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_012: [The function shall save the body to be sent with the request.] this.body = Arrays.copyOf(body, body.length); } } /** * Reads from the input stream (response stream) and returns the response. * * @return The response body. * * @throws IOException This exception thrown if the input stream could not be * accessed, for example if the server could not be reached. */ public byte[] readInput() throws IOException { // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_014: [The function shall read from the input stream (response stream) and return the response.] // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_015: [The function shall throw an IOException if the input stream could not be accessed.] InputStream inputStream = this.connection.getInputStream(); byte[] input = readInputStream(inputStream); // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_016: [The function shall close the input stream after it has been completely read.] inputStream.close(); return input; } /** * Reads from the error stream and returns the error reason. * * @return The error reason. * * @throws IOException This exception thrown if the input stream could not be * accessed, for example if the server could not be reached. */ public byte[] readError() throws IOException { // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_017: [The function shall read from the error stream and return the response.] // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_018: [The function shall throw an IOException if the error stream could not be accessed.] InputStream errorStream = this.connection.getErrorStream(); byte[] error = new byte[0]; // if there is no error reason, getErrorStream() returns null. if (errorStream != null) { error = readInputStream(errorStream); // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_019: [The function shall close the error stream after it has been completely read.] errorStream.close(); } return error; } /** * Returns the response status code. * * @return The response status code. * * @throws IOException This exception thrown if no response was received. */ public int getResponseStatus() throws IOException { // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_020: [The function shall return the response status code.] // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_021: [The function shall throw an IOException if no response was received.] return this.connection.getResponseCode(); } /** * Returns the response headers as a Map, where the key is the header field * name and the values are the values associated with the header field * name. * * @return the response headers. * * @throws IOException This exception thrown if no response was received. */ public Map> getResponseHeaders() throws IOException { // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_022: [The function shall return a mapping of header field names to the values associated with the header field name.] // Codes_SRS_SERVICE_SDK_JAVA_HTTPCONNECTION_12_023: [The function shall throw an IOException if no response was received.] return this.connection.getHeaderFields(); } /** * Reads the input stream until the stream is empty. * * @param stream The input stream. * * @return The content of the input stream. * * @throws IOException This exception thrown if the input stream could not be read from. */ protected static byte[] readInputStream(InputStream stream) throws IOException { ArrayList byteBuffer = new ArrayList<>(); int nextByte = -1; // read(byte[]) reads the byte into the buffer and returns the number // of bytes read, or -1 if the end of the stream has been reached. while ((nextByte = stream.read()) > -1) { byteBuffer.add((byte) nextByte); } int bufferSize = byteBuffer.size(); byte[] byteArray = new byte[bufferSize]; for (int i = 0; i < bufferSize; ++i) { byteArray[i] = byteBuffer.get(i); } return byteArray; } protected HttpConnection() { this.connection = null; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy