View Javadoc
1   package org.kuali.common.httplib.api.model;
2   
3   import static com.google.common.base.Optional.absent;
4   import static com.google.common.collect.Lists.newArrayList;
5   import static org.apache.commons.lang3.StringUtils.trimToNull;
6   import static org.kuali.common.jute.base.Optionals.fromTrimToNull;
7   import static org.kuali.common.jute.base.Precondition.checkNotBlank;
8   import static org.kuali.common.jute.base.Precondition.checkNotNull;
9   
10  import java.util.List;
11  
12  import com.fasterxml.jackson.annotation.JsonSetter;
13  import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
14  import com.google.common.base.Optional;
15  import com.google.common.collect.ImmutableList;
16  
17  /**
18   * {@code name} will not be null, the empty string (""), or pure whitespace.
19   *
20   * {@code value} will not be null, and (if present) will not be the empty string (""), or pure whitespace.
21   *
22   * An HTTP header with no value sounds like it is technically allowed, although the spec does not directly address this:
23   *
24   * <pre>
25   * http://www.w3.org/Protocols/rfc2616/rfc2616-sec4.html#sec4.2
26   *
27   * The field-content does not include any leading or trailing LWS: linear white space occurring before the first non-whitespace character of the field-value or after the last
28   * non-whitespace character of the field-value. Such leading or trailing LWS MAY be removed without changing the semantics of the field value. Any LWS that occurs between
29   * field-content MAY be replaced with a single SP before interpreting the field value or forwarding the message downstream.
30   * </pre>
31   *
32   * This class translates Apache's {@code org.apache.http.Header.getValue()} into an immutable Guava {@code Optional<String>} where {@code Optional.absent()} is used to represent
33   * header values that are the empty string ("") or pure whitespace.
34   */
35  @JsonDeserialize(builder = Header.Builder.class)
36  public final class Header {
37  
38      private final String name;
39      private final Optional<String> value;
40      private final ImmutableList<HeaderElement> elements;
41  
42      private Header(Builder builder) {
43          this.name = builder.name;
44          this.value = builder.value;
45          this.elements = ImmutableList.copyOf(builder.elements);
46      }
47  
48      /**
49       * Leading and trailing whitespace is trimmed from both {@code name} and {@code value} before copying values.
50       *
51       * There may be whitespace in the middle of the {@code name} or {@code value}, but the first and last characters are guaranteed to be non-whitespace text.
52       */
53      public static ImmutableList<Header> copyOf(Iterable<? extends org.apache.http.Header> mutable) {
54          List<Header> list = newArrayList();
55          for (org.apache.http.Header element : mutable) {
56              Header immutable = copyOf(element);
57              list.add(immutable);
58          }
59          return ImmutableList.copyOf(list);
60      }
61  
62      /**
63       * Leading and trailing whitespace is trimmed from both {@code name} and {@code value} before copying values.
64       *
65       * There may be whitespace in the middle of the {@code name} or {@code value}, but the first and last characters are guaranteed to be non-whitespace text.
66       */
67      public static Header copyOf(org.apache.http.Header mutable) {
68          Builder builder = builder();
69          builder.withName(trimToNull(mutable.getName()));
70          builder.withValue(fromTrimToNull(mutable.getValue()));
71          return builder.build();
72      }
73  
74      public static Builder builder() {
75          return new Builder();
76      }
77  
78      public static class Builder implements org.apache.commons.lang3.builder.Builder<Header> {
79  
80          private String name;
81          private Optional<String> value = absent();
82          private List<HeaderElement> elements = newArrayList();
83  
84          public Builder withName(String name) {
85              this.name = name;
86              return this;
87          }
88  
89          @JsonSetter
90          public Builder withValue(Optional<String> value) {
91              this.value = value;
92              return this;
93          }
94  
95          public Builder withValue(String value) {
96              return withValue(Optional.of(value));
97          }
98  
99          public Builder withElements(List<HeaderElement> elements) {
100             this.elements = elements;
101             return this;
102         }
103 
104         @Override
105         public Header build() {
106             return validate(new Header(this));
107         }
108 
109         private static Header validate(Header instance) {
110             checkNotBlank(instance.name, "name");
111             checkNotBlank(instance.value, "value");
112             checkNotNull(instance.elements, "elements");
113             return instance;
114         }
115     }
116 
117     public String getName() {
118         return name;
119     }
120 
121     public Optional<String> getValue() {
122         return value;
123     }
124 
125     public List<HeaderElement> getElements() {
126         return elements;
127     }
128 
129 }