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

retrofit2.ParameterHandler Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2015 Square, 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 retrofit2;

import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.lang.reflect.Array;
import java.lang.reflect.Method;
import java.net.URLEncoder;
import java.util.Iterator;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import javax.annotation.Nullable;
import okhttp3.MultipartBody;
import okhttp3.RequestBody;

abstract class ParameterHandler {
  abstract void apply(RequestBuilder builder, @Nullable T value) throws IOException;

  final ParameterHandler> iterable() {
    return new ParameterHandler>() {
      @Override
      void apply(RequestBuilder builder, @Nullable Iterable values) throws IOException {
        if (values == null) return; // Skip null values.

        for (T value : values) {
          ParameterHandler.this.apply(builder, value);
        }
      }
    };
  }

  final ParameterHandler array() {
    return new ParameterHandler() {
      @Override
      void apply(RequestBuilder builder, @Nullable Object values) throws IOException {
        if (values == null) return; // Skip null values.

        for (int i = 0, size = Array.getLength(values); i < size; i++) {
          //noinspection unchecked
          ParameterHandler.this.apply(builder, (T) Array.get(values, i));
        }
      }
    };
  }

  static final class RelativeUrl extends ParameterHandler {
    private final Method method;
    private final int p;

    RelativeUrl(Method method, int p) {
      this.method = method;
      this.p = p;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable Object value) {
      if (value == null) {
        throw Utils.parameterError(method, p, "@Url parameter is null.");
      }
      builder.setRelativeUrl(value);
    }
  }

  static final class Header extends ParameterHandler {
    private final String name;
    private final Converter valueConverter;

    Header(String name, Converter valueConverter) {
      this.name = Objects.requireNonNull(name, "name == null");
      this.valueConverter = valueConverter;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable T value) throws IOException {
      if (value == null) return; // Skip null values.

      String headerValue = valueConverter.convert(value);
      if (headerValue == null) return; // Skip converted but null values.

      builder.addHeader(name, headerValue);
    }
  }

  static final class Path extends ParameterHandler {
    private final Method method;
    private final int p;
    private final String name;
    private final Converter valueConverter;
    private final boolean encoded;

    Path(Method method, int p, String name, Converter valueConverter, boolean encoded) {
      this.method = method;
      this.p = p;
      this.name = Objects.requireNonNull(name, "name == null");
      this.valueConverter = valueConverter;
      this.encoded = encoded;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable T value) throws IOException {
      if (value == null) {
        throw Utils.parameterError(
            method, p, "Path parameter \"" + name + "\" value must not be null.");
      }
      builder.addPathParam(name, valueConverter.convert(value), encoded);
    }
  }

  static final class Query extends ParameterHandler {
    private final String name;
    private final Converter valueConverter;
    private final boolean encoded;

    Query(String name, Converter valueConverter, boolean encoded) {
      this.name = Objects.requireNonNull(name, "name == null");
      this.valueConverter = valueConverter;
      this.encoded = encoded;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable T value) throws IOException {
      if (value == null) return; // Skip null values.

      String queryValue = valueConverter.convert(value);
      if (queryValue == null) return; // Skip converted but null values

      builder.addQueryParam(name, queryValue, encoded);
    }
  }

  static final class QueryName extends ParameterHandler {
    private final Converter nameConverter;
    private final boolean encoded;

    QueryName(Converter nameConverter, boolean encoded) {
      this.nameConverter = nameConverter;
      this.encoded = encoded;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable T value) throws IOException {
      if (value == null) return; // Skip null values.
      builder.addQueryParam(nameConverter.convert(value), null, encoded);
    }
  }

  static final class QueryMap extends ParameterHandler> {
    private final Method method;
    private final int p;
    private final Converter valueConverter;
    private final boolean encoded;

    QueryMap(Method method, int p, Converter valueConverter, boolean encoded) {
      this.method = method;
      this.p = p;
      this.valueConverter = valueConverter;
      this.encoded = encoded;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable Map value) throws IOException {
      if (value == null) {
        throw Utils.parameterError(method, p, "Query map was null");
      }

      for (Map.Entry entry : value.entrySet()) {
        String entryKey = entry.getKey();
        if (entryKey == null) {
          throw Utils.parameterError(method, p, "Query map contained null key.");
        }
        T entryValue = entry.getValue();
        if (entryValue == null) {
          throw Utils.parameterError(
              method, p, "Query map contained null value for key '" + entryKey + "'.");
        }

        String convertedEntryValue = valueConverter.convert(entryValue);
        if (convertedEntryValue == null) {
          throw Utils.parameterError(
              method,
              p,
              "Query map value '"
                  + entryValue
                  + "' converted to null by "
                  + valueConverter.getClass().getName()
                  + " for key '"
                  + entryKey
                  + "'.");
        }

        builder.addQueryParam(entryKey, convertedEntryValue, encoded);
      }
    }
  }

  static final class HeaderMap extends ParameterHandler> {
    private final Method method;
    private final int p;
    private final Converter valueConverter;

    HeaderMap(Method method, int p, Converter valueConverter) {
      this.method = method;
      this.p = p;
      this.valueConverter = valueConverter;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable Map value) throws IOException {
      if (value == null) {
        throw Utils.parameterError(method, p, "Header map was null.");
      }

      for (Map.Entry entry : value.entrySet()) {
        String headerName = entry.getKey();
        if (headerName == null) {
          throw Utils.parameterError(method, p, "Header map contained null key.");
        }
        T headerValue = entry.getValue();
        if (headerValue == null) {
          throw Utils.parameterError(
              method, p, "Header map contained null value for key '" + headerName + "'.");
        }
        builder.addHeader(headerName, valueConverter.convert(headerValue));
      }
    }
  }

  static final class Headers extends ParameterHandler {
    private final Method method;
    private final int p;

    Headers(Method method, int p) {
      this.method = method;
      this.p = p;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable okhttp3.Headers headers) {
      if (headers == null) {
        throw Utils.parameterError(method, p, "Headers parameter must not be null.");
      }
      builder.addHeaders(headers);
    }
  }

  static final class Field extends ParameterHandler {
    private final String name;
    private final Converter valueConverter;
    private final boolean encoded;

    Field(String name, Converter valueConverter, boolean encoded) {
      this.name = Objects.requireNonNull(name, "name == null");
      this.valueConverter = valueConverter;
      this.encoded = encoded;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable T value) throws IOException {
      if (value == null) return; // Skip null values.

      String fieldValue = valueConverter.convert(value);
      if (fieldValue == null) return; // Skip null converted values

      builder.addFormField(name, fieldValue, encoded);
    }
  }

  static final class FieldMap extends ParameterHandler> {
    private final Method method;
    private final int p;
    private final Converter valueConverter;
    private final boolean encoded;

    FieldMap(Method method, int p, Converter valueConverter, boolean encoded) {
      this.method = method;
      this.p = p;
      this.valueConverter = valueConverter;
      this.encoded = encoded;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable Map value) throws IOException {
      if (value == null) {
        throw Utils.parameterError(method, p, "Field map was null.");
      }

      for (Map.Entry entry : value.entrySet()) {
        String entryKey = entry.getKey();
        if (entryKey == null) {
          throw Utils.parameterError(method, p, "Field map contained null key.");
        }
        T entryValue = entry.getValue();
        if (entryValue == null) {
          throw Utils.parameterError(
              method, p, "Field map contained null value for key '" + entryKey + "'.");
        }

        String fieldEntry = valueConverter.convert(entryValue);
        if (fieldEntry == null) {
          throw Utils.parameterError(
              method,
              p,
              "Field map value '"
                  + entryValue
                  + "' converted to null by "
                  + valueConverter.getClass().getName()
                  + " for key '"
                  + entryKey
                  + "'.");
        }

        builder.addFormField(entryKey, fieldEntry, encoded);
      }
    }
  }

  static final class Part extends ParameterHandler {
    private final Method method;
    private final int p;
    private final okhttp3.Headers headers;
    private final Converter converter;

    Part(Method method, int p, okhttp3.Headers headers, Converter converter) {
      this.method = method;
      this.p = p;
      this.headers = headers;
      this.converter = converter;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable T value) {
      if (value == null) return; // Skip null values.

      RequestBody body;
      try {
        body = converter.convert(value);
      } catch (IOException e) {
        throw Utils.parameterError(method, p, "Unable to convert " + value + " to RequestBody", e);
      }
      builder.addPart(headers, body);
    }
  }

  static final class RawPart extends ParameterHandler {
    static final RawPart INSTANCE = new RawPart();

    private RawPart() {}

    @Override
    void apply(RequestBuilder builder, @Nullable MultipartBody.Part value) {
      if (value != null) { // Skip null values.
        builder.addPart(value);
      }
    }
  }

  static final class PartMap extends ParameterHandler> {
    private final Method method;
    private final int p;
    private final Converter valueConverter;
    private final String transferEncoding;

    PartMap(
        Method method, int p, Converter valueConverter, String transferEncoding) {
      this.method = method;
      this.p = p;
      this.valueConverter = valueConverter;
      this.transferEncoding = transferEncoding;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable Map value) throws IOException {
      if (value == null) {
        throw Utils.parameterError(method, p, "Part map was null.");
      }

      for (Map.Entry entry : value.entrySet()) {
        String entryKey = entry.getKey();
        if (entryKey == null) {
          throw Utils.parameterError(method, p, "Part map contained null key.");
        }
        T entryValue = entry.getValue();
        if (entryValue == null) {
          throw Utils.parameterError(
              method, p, "Part map contained null value for key '" + entryKey + "'.");
        }

        okhttp3.Headers headers =
            okhttp3.Headers.of(
                "Content-Disposition",
                "form-data; name=\"" + entryKey + "\"",
                "Content-Transfer-Encoding",
                transferEncoding);

        builder.addPart(headers, valueConverter.convert(entryValue));
      }
    }
  }

  static final class Body extends ParameterHandler {
    private final Method method;
    private final int p;
    private final Converter converter;

    Body(Method method, int p, Converter converter) {
      this.method = method;
      this.p = p;
      this.converter = converter;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable T value) {
      if (value == null) {
        throw Utils.parameterError(method, p, "Body parameter value must not be null.");
      }
      RequestBody body;
      try {
        body = converter.convert(value);
      } catch (IOException e) {
        throw Utils.parameterError(method, e, p, "Unable to convert " + value + " to RequestBody");
      }
      builder.setBody(body);
    }
  }

  static final class Tag extends ParameterHandler {
    final Class cls;

    Tag(Class cls) {
      this.cls = cls;
    }

    @Override
    void apply(RequestBuilder builder, @Nullable T value) {
      builder.addTag(cls, value);
    }
  }

  static final class ParamQuery extends ParameterHandler {
    public final String key;
    private final String name;
    public final String value;
    private final Converter valueConverter;
    private final boolean encoded;

    ParamQuery(String query, Converter valueConverter, boolean encoded) {
      String[] split = query.split("=");
      if (split.length != 2) {
        throw new IllegalArgumentException("@ParamQuerys Configuration errors,at " + query);
      }
      this.name = Objects.requireNonNull(split[0], "name == null");
      this.value = Objects.requireNonNull(split[1], "query value null");
      Set set = RequestFactory.Builder.parsePathParameters(this.value);
      if (set.size() > 1) {
        throw new IllegalArgumentException(
            "@ParamQuerys Configuration errors,at "
                + name
                + ", You can only "
                + "have a maximum of one parameter");
      } else if (set.size() == 1) {
        Iterator iterator = set.iterator();
        key = iterator.next();
      } else {
        key = null;
      }
      this.valueConverter = valueConverter;
      this.encoded = encoded;
    }

    @Override
    void apply(RequestBuilder builder, T value) throws IOException {
      String rValue;
      String resultValue;
      rValue = valueConverter.convert(value);
      if (null != rValue && !"".equals(rValue)) {
        if (key != null) {
          resultValue = this.value.replace("{" + key + "}", rValue);
        } else {
          resultValue = rValue;
        }
        builder.addQueryParam(name, resultValue, encoded);
      }
    }
  }

  static final class ParamHeader extends ParameterHandler {
    private static final String UTF_8 = "utf-8";
    public final String key;
    private final String name;
    public final String value;
    private final Converter valueConverter;

    ParamHeader(String header, Converter valueConverter) {
      int index = header.indexOf(":");
      this.name = Objects.requireNonNull(header.substring(0, index), "name == null");
      this.value = Objects.requireNonNull(header.substring(index + 1), "query value null");
      Set set = RequestFactory.Builder.parseHeaderParameters(this.value);
      if (set.size() > 1) {
        throw new IllegalArgumentException(
            "@ParamHeader Configuration errors,at "
                + name
                + ", You can only have"
                + " a maximum of one parameter");
      } else if (set.size() == 1) {
        Iterator iterator = set.iterator();
        key = iterator.next();
      } else {
        key = null;
      }
      this.valueConverter = valueConverter;
    }

    @Override
    void apply(RequestBuilder builder, T value) throws IOException {
      String rValue;
      String resultValue = this.value;
      if (key != null) {
        rValue = extactValid(valueConverter.convert(value));
        if (rValue != null && !"".equals(rValue)) {
          resultValue = this.value.replace("{" + key + "}", rValue);
        }
      }
      if (resultValue != null && !"".equals(resultValue)) {
        builder.addHeader(name, resultValue.trim());
      }
    }

    String extactValid(String value) {
      if (value == null || "".equals(value)) {
        return null;
      }

      String result = null;
      StringBuffer buffer = new StringBuffer("");
      if (value.indexOf(";") != -1) { // 组合Value
        String[] strs = value.split(";");
        if (null != strs && strs.length > 0) {
          for (String str : strs) {
            if (str.indexOf("=") != -1 && str.lastIndexOf("=") != str.length() - 1) {
              String key = str.substring(0, str.indexOf("="));
              String val = str.substring(str.indexOf("=") + 1);
              if (key != null && !"".equals(key) && val != null && !"".equals(val)) {
                buffer
                    .append(encode(key, UTF_8))
                    .append("=")
                    .append(encode(val, UTF_8))
                    .append(";");
              }
            }
          }
          if (buffer.length() > 0 && buffer.toString().endsWith(";")) {
            result = buffer.deleteCharAt(buffer.length() - 1).toString();
          }
        }
        return result;
      } else { // 单个Value
        return encode(value, UTF_8);
      }
    }

    String encode(String content, String charset) {
      if (content == null || "".equals(content.trim())) {
        return null;
      }
      try {
        return URLEncoder.encode(content, charset != null ? charset : UTF_8);
      } catch (UnsupportedEncodingException ex) {
        return null;
      }
    }
  }

  public static class ParamUrl extends ParameterHandler {

    public final String url;

    public final String key;

    ParamUrl(String url) {
      this.url = url;
      Set set = RequestFactory.Builder.parsePathParameters(this.url);
      if (set.size() > 1) {
        throw new IllegalArgumentException(
            "@ParamQuerys Configuration errors,at "
                + url
                + ", You can only have"
                + " a maximum of one parameter");
      } else if (set.size() == 1) {
        Iterator iterator = set.iterator();
        key = iterator.next();
      } else {
        key = null;
      }
    }

    @Override
    void apply(RequestBuilder builder, T value) throws IOException {
      String resultValue = this.url;
      if (url != null) {
        resultValue = this.url.replace("{" + key + "}", value == null ? "" : value.toString());
      }
      builder.setServiceUrl(resultValue);
    }
  }
}