Value.java revision 56ed4167b942ec265f9cee70ac4d71d10b3835ce
1/*
2 * Copyright (C) 2010 Google Inc.
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package com.google.clearsilver.jsilver.values;
18
19import com.google.clearsilver.jsilver.autoescape.EscapeMode;
20import com.google.clearsilver.jsilver.data.DataContext;
21
22import java.util.HashMap;
23import java.util.Map;
24
25/**
26 * Dynamic typing system used by JSilver interpreter. A value (e.g. "2") can act as a string,
27 * integer and boolean. Values can be literal or references to variables held elsewhere (e.g. in
28 * external data structures such as HDF).
29 */
30public abstract class Value {
31
32  private static final Map<EscapeMode, Value> EMPTY_PART_ESCAPED;
33  private static final Map<EscapeMode, Value> EMPTY_UNESCAPED;
34  private static final Map<EscapeMode, Value> ZERO_PART_ESCAPED;
35  private static final Map<EscapeMode, Value> ZERO_UNESCAPED;
36  private static final Map<EscapeMode, Value> ONE_PART_ESCAPED;
37  private static final Map<EscapeMode, Value> ONE_UNESCAPED;
38
39  static {
40    // Currently a Value's EscapeMode is either ESCAPE_NONE (no escaping) or
41    // ESCAPE_IS_CONSTANT (is a constant or has some escaping applied).
42    // This may change in the future if we implement stricter auto escaping.
43    // See EscapeMode.combineModes.
44    EMPTY_PART_ESCAPED = new HashMap<EscapeMode, Value>(2);
45    EMPTY_PART_ESCAPED.put(EscapeMode.ESCAPE_NONE,
46        new StringValue("", EscapeMode.ESCAPE_NONE, true));
47    EMPTY_PART_ESCAPED.put(EscapeMode.ESCAPE_IS_CONSTANT, new StringValue("",
48        EscapeMode.ESCAPE_IS_CONSTANT, true));
49
50    EMPTY_UNESCAPED = new HashMap<EscapeMode, Value>(2);
51    EMPTY_UNESCAPED.put(EscapeMode.ESCAPE_NONE, new StringValue("", EscapeMode.ESCAPE_NONE, false));
52    EMPTY_UNESCAPED.put(EscapeMode.ESCAPE_IS_CONSTANT, new StringValue("",
53        EscapeMode.ESCAPE_IS_CONSTANT, false));
54
55    ZERO_PART_ESCAPED = new HashMap<EscapeMode, Value>(2);
56    ZERO_PART_ESCAPED.put(EscapeMode.ESCAPE_NONE, new NumberValue(0, EscapeMode.ESCAPE_NONE, true));
57    ZERO_PART_ESCAPED.put(EscapeMode.ESCAPE_IS_CONSTANT, new NumberValue(0,
58        EscapeMode.ESCAPE_IS_CONSTANT, true));
59
60    ZERO_UNESCAPED = new HashMap<EscapeMode, Value>(2);
61    ZERO_UNESCAPED.put(EscapeMode.ESCAPE_NONE, new NumberValue(0, EscapeMode.ESCAPE_NONE, false));
62    ZERO_UNESCAPED.put(EscapeMode.ESCAPE_IS_CONSTANT, new NumberValue(0,
63        EscapeMode.ESCAPE_IS_CONSTANT, false));
64
65    ONE_PART_ESCAPED = new HashMap<EscapeMode, Value>(2);
66    ONE_PART_ESCAPED.put(EscapeMode.ESCAPE_NONE, new NumberValue(1, EscapeMode.ESCAPE_NONE, true));
67    ONE_PART_ESCAPED.put(EscapeMode.ESCAPE_IS_CONSTANT, new NumberValue(1,
68        EscapeMode.ESCAPE_IS_CONSTANT, true));
69
70    ONE_UNESCAPED = new HashMap<EscapeMode, Value>(2);
71    ONE_UNESCAPED.put(EscapeMode.ESCAPE_NONE, new NumberValue(1, EscapeMode.ESCAPE_NONE, false));
72    ONE_UNESCAPED.put(EscapeMode.ESCAPE_IS_CONSTANT, new NumberValue(1,
73        EscapeMode.ESCAPE_IS_CONSTANT, false));
74  }
75
76  /**
77   * True if either the {@code Value} was escaped, or it was created from a combination of escaped
78   * and unescaped values.
79   */
80  private final boolean partiallyEscaped;
81  private final EscapeMode escapeMode;
82
83  public Value(EscapeMode escapeMode, boolean partiallyEscaped) {
84    this.escapeMode = escapeMode;
85    this.partiallyEscaped = partiallyEscaped;
86  }
87
88  /**
89   * Fetch value as boolean. All non empty strings and numbers != 0 are treated as true.
90   */
91  public abstract boolean asBoolean();
92
93  /**
94   * Fetch value as string.
95   */
96  public abstract String asString();
97
98  /**
99   * Fetch value as number. If number is not parseable, 0 is returned (this is consistent with
100   * ClearSilver).
101   */
102  public abstract int asNumber();
103
104  /**
105   * Whether this value exists. Literals always return true, but variable references will return
106   * false if the value behind it is null.
107   */
108  public abstract boolean exists();
109
110  public abstract boolean isEmpty();
111
112  /**
113   * Create a literal value using an int.
114   */
115  public static Value literalValue(int value, EscapeMode mode, boolean partiallyEscaped) {
116    return getIntValue(mode, partiallyEscaped, value);
117  }
118
119  /**
120   * Create a literal value using a String.
121   */
122  public static Value literalValue(String value, EscapeMode mode, boolean partiallyEscaped) {
123    if (value.isEmpty()) {
124      Value v = (partiallyEscaped ? EMPTY_PART_ESCAPED : EMPTY_UNESCAPED).get(mode);
125      if (v != null) {
126        return v;
127      }
128    }
129
130    return new StringValue(value, mode, partiallyEscaped);
131  }
132
133  /**
134   * Create a literal value using a boolean.
135   */
136  public static Value literalValue(boolean value, EscapeMode mode, boolean partiallyEscaped) {
137    return getIntValue(mode, partiallyEscaped, value ? 1 : 0);
138  }
139
140  private static Value getIntValue(EscapeMode mode, boolean partiallyEscaped, int num) {
141    Value v = null;
142    if (num == 0) {
143      v = (partiallyEscaped ? ZERO_PART_ESCAPED : ZERO_UNESCAPED).get(mode);
144    } else if (num == 1) {
145      v = (partiallyEscaped ? ONE_PART_ESCAPED : ONE_UNESCAPED).get(mode);
146    }
147
148    if (v != null) {
149      return v;
150    }
151
152    return new NumberValue(num, mode, partiallyEscaped);
153  }
154
155  /**
156   * Create a literal value using an int with a {@code escapeMode} of {@code
157   * EscapeMode.ESCAPE_IS_CONSTANT} and {@code partiallyEscaped} based on the {@code
158   * partiallyEscaped} values of the inputs.
159   *
160   * @param value integer value of the literal
161   * @param inputs Values that were used to compute the integer value.
162   */
163  public static Value literalConstant(int value, Value... inputs) {
164    boolean isPartiallyEscaped = false;
165    for (Value input : inputs) {
166      if (input.isPartiallyEscaped()) {
167        isPartiallyEscaped = true;
168        break;
169      }
170    }
171    return literalValue(value, EscapeMode.ESCAPE_IS_CONSTANT, isPartiallyEscaped);
172  }
173
174  /**
175   * Create a literal value using a string with a {@code escapeMode} of {@code
176   * EscapeMode.ESCAPE_IS_CONSTANT} and {@code partiallyEscaped} based on the {@code
177   * partiallyEscaped} values of the inputs.
178   *
179   * @param value String value of the literal
180   * @param inputs Values that were used to compute the string value.
181   */
182  public static Value literalConstant(String value, Value... inputs) {
183    boolean isPartiallyEscaped = false;
184    for (Value input : inputs) {
185      if (input.isPartiallyEscaped()) {
186        isPartiallyEscaped = true;
187        break;
188      }
189    }
190    return literalValue(value, EscapeMode.ESCAPE_IS_CONSTANT, isPartiallyEscaped);
191  }
192
193  /**
194   * Create a literal value using a boolean with a {@code escapeMode} of {@code
195   * EscapeMode.ESCAPE_IS_CONSTANT} and {@code partiallyEscaped} based on the {@code
196   * partiallyEscaped} values of the inputs.
197   *
198   * @param value boolean value of the literal
199   * @param inputs Values that were used to compute the boolean value.
200   */
201  public static Value literalConstant(boolean value, Value... inputs) {
202    boolean isPartiallyEscaped = false;
203    for (Value input : inputs) {
204      if (input.isPartiallyEscaped()) {
205        isPartiallyEscaped = true;
206        break;
207      }
208    }
209    return literalValue(value, EscapeMode.ESCAPE_IS_CONSTANT, isPartiallyEscaped);
210  }
211
212  /**
213   * Create a value linked to a variable name.
214   *
215   * @param name The pathname of the variable relative to the given {@link DataContext}
216   * @param dataContext The DataContext defining the scope and Data objects to use when
217   *        dereferencing the name.
218   * @return A Value object that allows access to the variable name, the variable Data object (if it
219   *         exists) and to the value of the variable.
220   */
221  public static Value variableValue(String name, DataContext dataContext) {
222    return new VariableValue(name, dataContext);
223  }
224
225  @Override
226  public boolean equals(Object other) {
227    if (other == null || !(other instanceof Value)) {
228      return false;
229    }
230    Value otherValue = (Value) other;
231    // This behaves the same way as ClearSilver.
232    return exists() == otherValue.exists()
233        && (asString().equals(otherValue.asString()) || (isEmpty() && otherValue.isEmpty()));
234  }
235
236  @Override
237  public int hashCode() {
238    return toString().hashCode();
239  }
240
241  @Override
242  public String toString() {
243    return asString();
244  }
245
246  public boolean isPartiallyEscaped() {
247    return partiallyEscaped;
248  }
249
250  /**
251   * Indicates the escaping that was applied to the expression represented by this value.
252   *
253   * <p>
254   * May be checked by the JSilver code before applying autoescaping. It differs from {@code
255   * isEscaped}, which is true iff any part of the variable expression contains an escaping
256   * function, even if the entire expression has not been escaped. Both methods are required,
257   * {@code isEscaped} to determine whether &lt;?cs escape &gt; commands should be applied, and
258   * {@code getEscapeMode} for autoescaping. This is done to maintain compatibility with
259   * ClearSilver's behaviour.
260   *
261   * @return {@code EscapeMode.ESCAPE_IS_CONSTANT} if the value represents a constant string
262   *         literal. Or the appropriate {@link EscapeMode} if the value is the output of an
263   *         escaping function.
264   *
265   * @see EscapeMode
266   */
267  public EscapeMode getEscapeMode() {
268    return escapeMode;
269  }
270
271}
272