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

com.google.api.client.http.javanet.NetHttpResponse Maven / Gradle / Ivy

Go to download

Google HTTP Client Library for Java. Functionality that works on all supported Java platforms, including Java 5 (or higher) desktop (SE) and web (EE), Android, and Google App Engine.

The newest version!
/*
 * Copyright (c) 2010 Google Inc.
 *
 * 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 com.google.api.client.http.javanet;

import com.google.api.client.http.LowLevelHttpResponse;
import java.io.FilterInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;

final class NetHttpResponse extends LowLevelHttpResponse {

  private final HttpURLConnection connection;
  private final int responseCode;
  private final String responseMessage;
  private final ArrayList headerNames = new ArrayList();
  private final ArrayList headerValues = new ArrayList();

  NetHttpResponse(HttpURLConnection connection) throws IOException {
    this.connection = connection;
    int responseCode = connection.getResponseCode();
    this.responseCode = responseCode == -1 ? 0 : responseCode;
    responseMessage = connection.getResponseMessage();
    List headerNames = this.headerNames;
    List headerValues = this.headerValues;
    for (Map.Entry> entry : connection.getHeaderFields().entrySet()) {
      String key = entry.getKey();
      if (key != null) {
        for (String value : entry.getValue()) {
          if (value != null) {
            headerNames.add(key);
            headerValues.add(value);
          }
        }
      }
    }
  }

  @Override
  public int getStatusCode() {
    return responseCode;
  }

  /**
   * {@inheritDoc}
   *
   * 

Returns {@link HttpURLConnection#getInputStream} when it doesn't throw {@link IOException}, * otherwise it returns {@link HttpURLConnection#getErrorStream}. * *

Upgrade warning: in prior version 1.16 {@link #getContent()} returned {@link * HttpURLConnection#getInputStream} only when the status code was successful. Starting with * version 1.17 it returns {@link HttpURLConnection#getInputStream} when it doesn't throw {@link * IOException}, otherwise it returns {@link HttpURLConnection#getErrorStream} * *

Upgrade warning: in versions prior to 1.20 {@link #getContent()} returned {@link * HttpURLConnection#getInputStream()} or {@link HttpURLConnection#getErrorStream()}, both of * which silently returned -1 for read() calls when the connection got closed in the middle of * receiving a response. This is highly likely a bug from JDK's {@link HttpURLConnection}. Since * version 1.20, the bytes read off the wire will be checked and an {@link IOException} will be * thrown if the response is not fully delivered when the connection is closed by server for * whatever reason, e.g., server restarts. Note though that this is a best-effort check: when the * response is chunk encoded, we have to rely on the underlying HTTP library to behave correctly. */ @Override public InputStream getContent() throws IOException { InputStream in = null; try { in = connection.getInputStream(); } catch (IOException ioe) { in = connection.getErrorStream(); } return in == null ? null : new SizeValidatingInputStream(in); } @Override public String getContentEncoding() { return connection.getContentEncoding(); } @Override public long getContentLength() { String string = connection.getHeaderField("Content-Length"); return string == null ? -1 : Long.parseLong(string); } @Override public String getContentType() { return connection.getHeaderField("Content-Type"); } @Override public String getReasonPhrase() { return responseMessage; } @Override public String getStatusLine() { String result = connection.getHeaderField(0); return result != null && result.startsWith("HTTP/1.") ? result : null; } @Override public int getHeaderCount() { return headerNames.size(); } @Override public String getHeaderName(int index) { return headerNames.get(index); } @Override public String getHeaderValue(int index) { return headerValues.get(index); } /** * Closes the connection to the HTTP server. * * @since 1.4 */ @Override public void disconnect() { connection.disconnect(); } /** * A wrapper arround the base {@link InputStream} that validates EOF returned by the read calls. * * @since 1.20 */ private final class SizeValidatingInputStream extends FilterInputStream { private long bytesRead = 0; public SizeValidatingInputStream(InputStream in) { super(in); } /** * java.io.InputStream#read(byte[], int, int) swallows IOException thrown from read() so we have * to override it. * * @see * "http://grepcode.com/file/repository.grepcode.com/java/root/jdk/openjdk/8-b132/java/io/InputStream.java#185" */ @Override public int read(byte[] b, int off, int len) throws IOException { int n = in.read(b, off, len); if (n == -1) { throwIfFalseEOF(); } else { bytesRead += n; } return n; } @Override public int read() throws IOException { int n = in.read(); if (n == -1) { throwIfFalseEOF(); } else { bytesRead++; } return n; } @Override public long skip(long len) throws IOException { long n = in.skip(len); bytesRead += n; return n; } // Throws an IOException if gets an EOF in the middle of a response. private void throwIfFalseEOF() throws IOException { long contentLength = getContentLength(); if (contentLength == -1) { // If a Content-Length header is missing, there's nothing we can do. return; } // According to RFC2616, message-body is prohibited in responses to certain requests, e.g., // HEAD. Nevertheless an entity-header (possibly with non-zero Content-Length) may be present. // Thus we exclude the case where bytesRead == 0. // // See http://www.w3.org/Protocols/rfc2616/rfc2616-sec4.html#sec4.4 for details. if (bytesRead != 0 && bytesRead < contentLength) { throw new IOException( "Connection closed prematurely: bytesRead = " + bytesRead + ", Content-Length = " + contentLength); } } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy