| 1 | |
|
| 2 | |
|
| 3 | |
|
| 4 | |
|
| 5 | |
|
| 6 | |
|
| 7 | |
|
| 8 | |
|
| 9 | |
|
| 10 | |
|
| 11 | |
|
| 12 | |
|
| 13 | |
|
| 14 | |
|
| 15 | |
|
| 16 | |
package org.kuali.rice.krad.uif.util; |
| 17 | |
|
| 18 | |
import org.springframework.expression.spel.InternalParseException; |
| 19 | |
import org.springframework.expression.spel.SpelMessage; |
| 20 | |
import org.springframework.expression.spel.SpelParseException; |
| 21 | |
import org.springframework.util.Assert; |
| 22 | |
|
| 23 | |
import java.util.ArrayList; |
| 24 | |
import java.util.Arrays; |
| 25 | |
import java.util.List; |
| 26 | |
|
| 27 | |
|
| 28 | |
|
| 29 | |
|
| 30 | |
public class Tokenizer { |
| 31 | |
String expressionString; |
| 32 | |
char[] toProcess; |
| 33 | |
int pos; |
| 34 | |
int max; |
| 35 | 0 | List<Token> tokens = new ArrayList<Token>(); |
| 36 | |
|
| 37 | 0 | protected Tokenizer(String inputdata) { |
| 38 | 0 | for (int ch = '0'; ch <= '9'; ch++) { |
| 39 | 0 | flags[ch] |= IS_DIGIT | IS_HEXDIGIT; |
| 40 | |
} |
| 41 | 0 | for (int ch = 'A'; ch <= 'F'; ch++) { |
| 42 | 0 | flags[ch] |= IS_HEXDIGIT; |
| 43 | |
} |
| 44 | 0 | for (int ch = 'a'; ch <= 'f'; ch++) { |
| 45 | 0 | flags[ch] |= IS_HEXDIGIT; |
| 46 | |
} |
| 47 | 0 | for (int ch = 'A'; ch <= 'Z'; ch++) { |
| 48 | 0 | flags[ch] |= IS_ALPHA; |
| 49 | |
} |
| 50 | 0 | for (int ch = 'a'; ch <= 'z'; ch++) { |
| 51 | 0 | flags[ch] |= IS_ALPHA; |
| 52 | |
} |
| 53 | |
|
| 54 | 0 | this.expressionString = inputdata; |
| 55 | 0 | this.toProcess = (inputdata + "\0").toCharArray(); |
| 56 | 0 | this.max = toProcess.length; |
| 57 | 0 | this.pos = 0; |
| 58 | 0 | process(); |
| 59 | 0 | } |
| 60 | |
|
| 61 | |
public void process() { |
| 62 | 0 | while (pos < max) { |
| 63 | 0 | char ch = toProcess[pos]; |
| 64 | 0 | if (isAlphabetic(ch)) { |
| 65 | 0 | lexIdentifier(); |
| 66 | |
} else { |
| 67 | 0 | switch (ch) { |
| 68 | |
case '+': |
| 69 | 0 | pushCharToken(TokenKind.PLUS); |
| 70 | 0 | break; |
| 71 | |
case '_': |
| 72 | 0 | lexIdentifier(); |
| 73 | 0 | break; |
| 74 | |
case '-': |
| 75 | 0 | pushCharToken(TokenKind.MINUS); |
| 76 | 0 | break; |
| 77 | |
case ':': |
| 78 | 0 | pushCharToken(TokenKind.COLON); |
| 79 | 0 | break; |
| 80 | |
case '.': |
| 81 | 0 | pushCharToken(TokenKind.DOT); |
| 82 | 0 | break; |
| 83 | |
case ',': |
| 84 | 0 | pushCharToken(TokenKind.COMMA); |
| 85 | 0 | break; |
| 86 | |
case '*': |
| 87 | 0 | pushCharToken(TokenKind.STAR); |
| 88 | 0 | break; |
| 89 | |
case '/': |
| 90 | 0 | pushCharToken(TokenKind.DIV); |
| 91 | 0 | break; |
| 92 | |
case '%': |
| 93 | 0 | pushCharToken(TokenKind.MOD); |
| 94 | 0 | break; |
| 95 | |
case '(': |
| 96 | 0 | pushCharToken(TokenKind.LPAREN); |
| 97 | 0 | break; |
| 98 | |
case ')': |
| 99 | 0 | pushCharToken(TokenKind.RPAREN); |
| 100 | 0 | break; |
| 101 | |
case '[': |
| 102 | 0 | pushCharToken(TokenKind.LSQUARE); |
| 103 | 0 | break; |
| 104 | |
case '#': |
| 105 | 0 | pushCharToken(TokenKind.HASH); |
| 106 | 0 | break; |
| 107 | |
case ']': |
| 108 | 0 | pushCharToken(TokenKind.RSQUARE); |
| 109 | 0 | break; |
| 110 | |
case '{': |
| 111 | 0 | pushCharToken(TokenKind.LCURLY); |
| 112 | 0 | break; |
| 113 | |
case '}': |
| 114 | 0 | pushCharToken(TokenKind.RCURLY); |
| 115 | 0 | break; |
| 116 | |
case '@': |
| 117 | 0 | pushCharToken(TokenKind.BEAN_REF); |
| 118 | 0 | break; |
| 119 | |
case '^': |
| 120 | 0 | if (isTwoCharToken(TokenKind.SELECT_FIRST)) { |
| 121 | 0 | pushPairToken(TokenKind.SELECT_FIRST); |
| 122 | |
} else { |
| 123 | 0 | pushCharToken(TokenKind.POWER); |
| 124 | |
} |
| 125 | 0 | break; |
| 126 | |
case '!': |
| 127 | 0 | if (isTwoCharToken(TokenKind.NE)) { |
| 128 | 0 | pushPairToken(TokenKind.NE); |
| 129 | 0 | } else if (isTwoCharToken(TokenKind.PROJECT)) { |
| 130 | 0 | pushPairToken(TokenKind.PROJECT); |
| 131 | |
} else { |
| 132 | 0 | pushCharToken(TokenKind.NOT); |
| 133 | |
} |
| 134 | 0 | break; |
| 135 | |
case '=': |
| 136 | 0 | if (isTwoCharToken(TokenKind.EQ)) { |
| 137 | 0 | pushPairToken(TokenKind.EQ); |
| 138 | |
} else { |
| 139 | 0 | pushCharToken(TokenKind.ASSIGN); |
| 140 | |
} |
| 141 | 0 | break; |
| 142 | |
case '?': |
| 143 | 0 | if (isTwoCharToken(TokenKind.SELECT)) { |
| 144 | 0 | pushPairToken(TokenKind.SELECT); |
| 145 | 0 | } else if (isTwoCharToken(TokenKind.ELVIS)) { |
| 146 | 0 | pushPairToken(TokenKind.ELVIS); |
| 147 | 0 | } else if (isTwoCharToken(TokenKind.SAFE_NAVI)) { |
| 148 | 0 | pushPairToken(TokenKind.SAFE_NAVI); |
| 149 | |
} else { |
| 150 | 0 | pushCharToken(TokenKind.QMARK); |
| 151 | |
} |
| 152 | 0 | break; |
| 153 | |
case '$': |
| 154 | 0 | if (isTwoCharToken(TokenKind.SELECT_LAST)) { |
| 155 | 0 | pushPairToken(TokenKind.SELECT_LAST); |
| 156 | |
} else { |
| 157 | 0 | lexIdentifier(); |
| 158 | |
} |
| 159 | 0 | break; |
| 160 | |
case '>': |
| 161 | 0 | if (isTwoCharToken(TokenKind.GE)) { |
| 162 | 0 | pushPairToken(TokenKind.GE); |
| 163 | |
} else { |
| 164 | 0 | pushCharToken(TokenKind.GT); |
| 165 | |
} |
| 166 | 0 | break; |
| 167 | |
case '<': |
| 168 | 0 | if (isTwoCharToken(TokenKind.LE)) { |
| 169 | 0 | pushPairToken(TokenKind.LE); |
| 170 | |
} else { |
| 171 | 0 | pushCharToken(TokenKind.LT); |
| 172 | |
} |
| 173 | 0 | break; |
| 174 | |
case '0': |
| 175 | |
case '1': |
| 176 | |
case '2': |
| 177 | |
case '3': |
| 178 | |
case '4': |
| 179 | |
case '5': |
| 180 | |
case '6': |
| 181 | |
case '7': |
| 182 | |
case '8': |
| 183 | |
case '9': |
| 184 | 0 | lexNumericLiteral(ch == '0'); |
| 185 | 0 | break; |
| 186 | |
case ' ': |
| 187 | |
case '\t': |
| 188 | |
case '\r': |
| 189 | |
case '\n': |
| 190 | |
|
| 191 | 0 | pos++; |
| 192 | 0 | break; |
| 193 | |
case '\'': |
| 194 | 0 | lexQuotedStringLiteral(); |
| 195 | 0 | break; |
| 196 | |
case '"': |
| 197 | 0 | lexDoubleQuotedStringLiteral(); |
| 198 | 0 | break; |
| 199 | |
case 0: |
| 200 | |
|
| 201 | 0 | pos++; |
| 202 | 0 | break; |
| 203 | |
default: |
| 204 | 0 | throw new IllegalStateException("Cannot handle (" |
| 205 | |
+ Integer.valueOf(ch) |
| 206 | |
+ ") '" |
| 207 | |
+ ch |
| 208 | |
+ "', in expression: " |
| 209 | |
+ expressionString); |
| 210 | |
} |
| 211 | |
} |
| 212 | 0 | } |
| 213 | 0 | } |
| 214 | |
|
| 215 | |
public List<Token> getTokens() { |
| 216 | 0 | return tokens; |
| 217 | |
} |
| 218 | |
|
| 219 | |
|
| 220 | |
private void lexQuotedStringLiteral() { |
| 221 | 0 | int start = pos; |
| 222 | 0 | boolean terminated = false; |
| 223 | 0 | while (!terminated) { |
| 224 | 0 | pos++; |
| 225 | 0 | char ch = toProcess[pos]; |
| 226 | 0 | if (ch == '\'') { |
| 227 | |
|
| 228 | 0 | if (toProcess[pos + 1] == '\'') { |
| 229 | 0 | pos++; |
| 230 | |
} else { |
| 231 | 0 | terminated = true; |
| 232 | |
} |
| 233 | |
} |
| 234 | 0 | if (ch == 0) { |
| 235 | 0 | throw new InternalParseException(new SpelParseException(expressionString, start, |
| 236 | |
SpelMessage.NON_TERMINATING_QUOTED_STRING)); |
| 237 | |
} |
| 238 | 0 | } |
| 239 | 0 | pos++; |
| 240 | 0 | tokens.add(new Token(TokenKind.LITERAL_STRING, subarray(start, pos), start, pos)); |
| 241 | 0 | } |
| 242 | |
|
| 243 | |
|
| 244 | |
private void lexDoubleQuotedStringLiteral() { |
| 245 | 0 | int start = pos; |
| 246 | 0 | boolean terminated = false; |
| 247 | 0 | while (!terminated) { |
| 248 | 0 | pos++; |
| 249 | 0 | char ch = toProcess[pos]; |
| 250 | 0 | if (ch == '"') { |
| 251 | 0 | terminated = true; |
| 252 | |
} |
| 253 | 0 | if (ch == 0) { |
| 254 | 0 | throw new InternalParseException(new SpelParseException(expressionString, start, |
| 255 | |
SpelMessage.NON_TERMINATING_DOUBLE_QUOTED_STRING)); |
| 256 | |
} |
| 257 | 0 | } |
| 258 | 0 | pos++; |
| 259 | 0 | tokens.add(new Token(TokenKind.LITERAL_STRING, subarray(start, pos), start, pos)); |
| 260 | 0 | } |
| 261 | |
|
| 262 | |
|
| 263 | |
|
| 264 | |
|
| 265 | |
|
| 266 | |
|
| 267 | |
|
| 268 | |
|
| 269 | |
|
| 270 | |
|
| 271 | |
|
| 272 | |
|
| 273 | |
|
| 274 | |
|
| 275 | |
|
| 276 | |
private void lexNumericLiteral(boolean firstCharIsZero) { |
| 277 | 0 | boolean isReal = false; |
| 278 | 0 | int start = pos; |
| 279 | 0 | char ch = toProcess[pos + 1]; |
| 280 | 0 | boolean isHex = ch == 'x' || ch == 'X'; |
| 281 | |
|
| 282 | |
|
| 283 | 0 | if (firstCharIsZero && isHex) { |
| 284 | 0 | pos = pos + 1; |
| 285 | |
do { |
| 286 | 0 | pos++; |
| 287 | 0 | } while (isHexadecimalDigit(toProcess[pos])); |
| 288 | 0 | if (isChar('L', 'l')) { |
| 289 | 0 | pushHexIntToken(subarray(start + 2, pos), true, start, pos); |
| 290 | 0 | pos++; |
| 291 | |
} else { |
| 292 | 0 | pushHexIntToken(subarray(start + 2, pos), false, start, pos); |
| 293 | |
} |
| 294 | 0 | return; |
| 295 | |
} |
| 296 | |
|
| 297 | |
|
| 298 | |
|
| 299 | |
|
| 300 | |
do { |
| 301 | 0 | pos++; |
| 302 | 0 | } while (isDigit(toProcess[pos])); |
| 303 | |
|
| 304 | |
|
| 305 | 0 | ch = toProcess[pos]; |
| 306 | 0 | if (ch == '.') { |
| 307 | 0 | isReal = true; |
| 308 | |
|
| 309 | |
do { |
| 310 | 0 | pos++; |
| 311 | 0 | } while (isDigit(toProcess[pos])); |
| 312 | |
} |
| 313 | |
|
| 314 | 0 | int endOfNumber = pos; |
| 315 | |
|
| 316 | |
|
| 317 | |
|
| 318 | |
|
| 319 | 0 | if (isChar('L', 'l')) { |
| 320 | 0 | if (isReal) { |
| 321 | 0 | throw new InternalParseException(new SpelParseException(expressionString, start, |
| 322 | |
SpelMessage.REAL_CANNOT_BE_LONG)); |
| 323 | |
} |
| 324 | 0 | pushIntToken(subarray(start, endOfNumber), true, start, endOfNumber); |
| 325 | 0 | pos++; |
| 326 | 0 | } else if (isExponentChar(toProcess[pos])) { |
| 327 | 0 | isReal = true; |
| 328 | 0 | pos++; |
| 329 | 0 | char possibleSign = toProcess[pos]; |
| 330 | 0 | if (isSign(possibleSign)) { |
| 331 | 0 | pos++; |
| 332 | |
} |
| 333 | |
|
| 334 | |
|
| 335 | |
do { |
| 336 | 0 | pos++; |
| 337 | 0 | } while (isDigit(toProcess[pos])); |
| 338 | 0 | boolean isFloat = false; |
| 339 | 0 | if (isFloatSuffix(toProcess[pos])) { |
| 340 | 0 | isFloat = true; |
| 341 | 0 | endOfNumber = ++pos; |
| 342 | 0 | } else if (isDoubleSuffix(toProcess[pos])) { |
| 343 | 0 | endOfNumber = ++pos; |
| 344 | |
} |
| 345 | 0 | pushRealToken(subarray(start, pos), isFloat, start, pos); |
| 346 | 0 | } else { |
| 347 | 0 | ch = toProcess[pos]; |
| 348 | 0 | boolean isFloat = false; |
| 349 | 0 | if (isFloatSuffix(ch)) { |
| 350 | 0 | isReal = true; |
| 351 | 0 | isFloat = true; |
| 352 | 0 | endOfNumber = ++pos; |
| 353 | 0 | } else if (isDoubleSuffix(ch)) { |
| 354 | 0 | isReal = true; |
| 355 | 0 | endOfNumber = ++pos; |
| 356 | |
} |
| 357 | 0 | if (isReal) { |
| 358 | 0 | pushRealToken(subarray(start, endOfNumber), isFloat, start, endOfNumber); |
| 359 | |
} else { |
| 360 | 0 | pushIntToken(subarray(start, endOfNumber), false, start, endOfNumber); |
| 361 | |
} |
| 362 | |
} |
| 363 | 0 | } |
| 364 | |
|
| 365 | |
|
| 366 | 0 | private String[] alternativeOperatorNames = {"DIV", "EQ", "GE", "GT", "LE", "LT", "MOD", "NE", "NOT"}; |
| 367 | |
|
| 368 | |
private void lexIdentifier() { |
| 369 | 0 | int start = pos; |
| 370 | |
do { |
| 371 | 0 | pos++; |
| 372 | 0 | } while (isIdentifier(toProcess[pos])); |
| 373 | 0 | char[] subarray = subarray(start, pos); |
| 374 | |
|
| 375 | |
|
| 376 | 0 | if ((pos - start) == 2 || (pos - start) == 3) { |
| 377 | 0 | String asString = new String(subarray).toUpperCase(); |
| 378 | 0 | int idx = Arrays.binarySearch(alternativeOperatorNames, asString); |
| 379 | 0 | if (idx >= 0) { |
| 380 | 0 | pushOneCharOrTwoCharToken(TokenKind.valueOf(asString), start); |
| 381 | 0 | return; |
| 382 | |
} |
| 383 | |
} |
| 384 | 0 | tokens.add(new Token(TokenKind.IDENTIFIER, subarray, start, pos)); |
| 385 | 0 | } |
| 386 | |
|
| 387 | |
private void pushIntToken(char[] data, boolean isLong, int start, int end) { |
| 388 | 0 | if (isLong) { |
| 389 | 0 | tokens.add(new Token(TokenKind.LITERAL_LONG, data, start, end)); |
| 390 | |
} else { |
| 391 | 0 | tokens.add(new Token(TokenKind.LITERAL_INT, data, start, end)); |
| 392 | |
} |
| 393 | 0 | } |
| 394 | |
|
| 395 | |
private void pushHexIntToken(char[] data, boolean isLong, int start, int end) { |
| 396 | 0 | if (data.length == 0) { |
| 397 | 0 | if (isLong) { |
| 398 | 0 | throw new InternalParseException(new SpelParseException(expressionString, start, SpelMessage.NOT_A_LONG, |
| 399 | |
expressionString.substring(start, end + 1))); |
| 400 | |
} else { |
| 401 | 0 | throw new InternalParseException(new SpelParseException(expressionString, start, |
| 402 | |
SpelMessage.NOT_AN_INTEGER, expressionString.substring(start, end))); |
| 403 | |
} |
| 404 | |
} |
| 405 | 0 | if (isLong) { |
| 406 | 0 | tokens.add(new Token(TokenKind.LITERAL_HEXLONG, data, start, end)); |
| 407 | |
} else { |
| 408 | 0 | tokens.add(new Token(TokenKind.LITERAL_HEXINT, data, start, end)); |
| 409 | |
} |
| 410 | 0 | } |
| 411 | |
|
| 412 | |
private void pushRealToken(char[] data, boolean isFloat, int start, int end) { |
| 413 | 0 | if (isFloat) { |
| 414 | 0 | tokens.add(new Token(TokenKind.LITERAL_REAL_FLOAT, data, start, end)); |
| 415 | |
} else { |
| 416 | 0 | tokens.add(new Token(TokenKind.LITERAL_REAL, data, start, end)); |
| 417 | |
} |
| 418 | 0 | } |
| 419 | |
|
| 420 | |
private char[] subarray(int start, int end) { |
| 421 | 0 | char[] result = new char[end - start]; |
| 422 | 0 | System.arraycopy(toProcess, start, result, 0, end - start); |
| 423 | 0 | return result; |
| 424 | |
} |
| 425 | |
|
| 426 | |
|
| 427 | |
|
| 428 | |
|
| 429 | |
private boolean isTwoCharToken(TokenKind kind) { |
| 430 | 0 | Assert.isTrue(kind.tokenChars.length == 2); |
| 431 | 0 | Assert.isTrue(toProcess[pos] == kind.tokenChars[0]); |
| 432 | 0 | return toProcess[pos + 1] == kind.tokenChars[1]; |
| 433 | |
} |
| 434 | |
|
| 435 | |
|
| 436 | |
|
| 437 | |
|
| 438 | |
private void pushCharToken(TokenKind kind) { |
| 439 | 0 | tokens.add(new Token(kind, pos, pos + 1)); |
| 440 | 0 | pos++; |
| 441 | 0 | } |
| 442 | |
|
| 443 | |
|
| 444 | |
|
| 445 | |
|
| 446 | |
private void pushPairToken(TokenKind kind) { |
| 447 | 0 | tokens.add(new Token(kind, pos, pos + 2)); |
| 448 | 0 | pos += 2; |
| 449 | 0 | } |
| 450 | |
|
| 451 | |
private void pushOneCharOrTwoCharToken(TokenKind kind, int pos) { |
| 452 | 0 | tokens.add(new Token(kind, pos, pos + kind.getLength())); |
| 453 | 0 | } |
| 454 | |
|
| 455 | |
|
| 456 | |
private boolean isIdentifier(char ch) { |
| 457 | 0 | return isAlphabetic(ch) || isDigit(ch) || ch == '_' || ch == '$'; |
| 458 | |
} |
| 459 | |
|
| 460 | |
private boolean isChar(char a, char b) { |
| 461 | 0 | char ch = toProcess[pos]; |
| 462 | 0 | return ch == a || ch == b; |
| 463 | |
} |
| 464 | |
|
| 465 | |
private boolean isExponentChar(char ch) { |
| 466 | 0 | return ch == 'e' || ch == 'E'; |
| 467 | |
} |
| 468 | |
|
| 469 | |
private boolean isFloatSuffix(char ch) { |
| 470 | 0 | return ch == 'f' || ch == 'F'; |
| 471 | |
} |
| 472 | |
|
| 473 | |
private boolean isDoubleSuffix(char ch) { |
| 474 | 0 | return ch == 'd' || ch == 'D'; |
| 475 | |
} |
| 476 | |
|
| 477 | |
private boolean isSign(char ch) { |
| 478 | 0 | return ch == '+' || ch == '-'; |
| 479 | |
} |
| 480 | |
|
| 481 | |
private boolean isDigit(char ch) { |
| 482 | 0 | if (ch > 255) { |
| 483 | 0 | return false; |
| 484 | |
} |
| 485 | 0 | return (flags[ch] & IS_DIGIT) != 0; |
| 486 | |
} |
| 487 | |
|
| 488 | |
private boolean isAlphabetic(char ch) { |
| 489 | 0 | if (ch > 255) { |
| 490 | 0 | return false; |
| 491 | |
} |
| 492 | 0 | return (flags[ch] & IS_ALPHA) != 0; |
| 493 | |
} |
| 494 | |
|
| 495 | |
private boolean isHexadecimalDigit(char ch) { |
| 496 | 0 | if (ch > 255) { |
| 497 | 0 | return false; |
| 498 | |
} |
| 499 | 0 | return (flags[ch] & IS_HEXDIGIT) != 0; |
| 500 | |
} |
| 501 | |
|
| 502 | 0 | private final byte flags[] = new byte[256]; |
| 503 | |
private static final byte IS_DIGIT = 0x01; |
| 504 | |
private static final byte IS_HEXDIGIT = 0x02; |
| 505 | |
private static final byte IS_ALPHA = 0x04; |
| 506 | |
|
| 507 | |
public class Token { |
| 508 | |
TokenKind kind; |
| 509 | |
String data; |
| 510 | |
int startpos; |
| 511 | |
int endpos; |
| 512 | |
|
| 513 | |
|
| 514 | |
|
| 515 | |
|
| 516 | |
|
| 517 | |
|
| 518 | |
|
| 519 | 0 | public Token(TokenKind tokenKind, int startpos, int endpos) { |
| 520 | 0 | this.kind = tokenKind; |
| 521 | 0 | this.startpos = startpos; |
| 522 | 0 | this.endpos = endpos; |
| 523 | 0 | } |
| 524 | |
|
| 525 | |
Token(TokenKind tokenKind, char[] tokenData, int pos, int endpos) { |
| 526 | 0 | this(tokenKind, pos, endpos); |
| 527 | 0 | this.data = new String(tokenData); |
| 528 | 0 | } |
| 529 | |
|
| 530 | |
public TokenKind getKind() { |
| 531 | 0 | return kind; |
| 532 | |
} |
| 533 | |
|
| 534 | |
public String toString() { |
| 535 | 0 | StringBuilder s = new StringBuilder(); |
| 536 | 0 | s.append("[").append(kind.toString()); |
| 537 | 0 | if (kind.hasPayload()) { |
| 538 | 0 | s.append(":").append(data); |
| 539 | |
} |
| 540 | 0 | s.append("]"); |
| 541 | 0 | s.append("(").append(startpos).append(",").append(endpos).append(")"); |
| 542 | 0 | return s.toString(); |
| 543 | |
} |
| 544 | |
|
| 545 | |
public boolean isIdentifier() { |
| 546 | 0 | return kind == TokenKind.IDENTIFIER; |
| 547 | |
} |
| 548 | |
|
| 549 | |
public boolean isNumericRelationalOperator() { |
| 550 | 0 | return kind == TokenKind.GT |
| 551 | |
|| kind == TokenKind.GE |
| 552 | |
|| kind == TokenKind.LT |
| 553 | |
|| kind == TokenKind.LE |
| 554 | |
|| kind == TokenKind.EQ |
| 555 | |
|| kind == TokenKind.NE; |
| 556 | |
} |
| 557 | |
|
| 558 | |
public String stringValue() { |
| 559 | 0 | return data; |
| 560 | |
} |
| 561 | |
|
| 562 | |
public Token asInstanceOfToken() { |
| 563 | 0 | return new Token(TokenKind.INSTANCEOF, startpos, endpos); |
| 564 | |
} |
| 565 | |
|
| 566 | |
public Token asMatchesToken() { |
| 567 | 0 | return new Token(TokenKind.MATCHES, startpos, endpos); |
| 568 | |
} |
| 569 | |
|
| 570 | |
public Token asBetweenToken() { |
| 571 | 0 | return new Token(TokenKind.BETWEEN, startpos, endpos); |
| 572 | |
} |
| 573 | |
} |
| 574 | |
|
| 575 | 0 | public enum TokenKind { |
| 576 | |
|
| 577 | 0 | LITERAL_INT, LITERAL_LONG, LITERAL_HEXINT, LITERAL_HEXLONG, LITERAL_STRING, LITERAL_REAL, LITERAL_REAL_FLOAT, |
| 578 | 0 | LPAREN("("), RPAREN(")"), COMMA(","), IDENTIFIER, |
| 579 | 0 | COLON(":"), HASH("#"), RSQUARE("]"), LSQUARE("["), |
| 580 | 0 | LCURLY("{"), RCURLY("}"), |
| 581 | 0 | DOT("."), PLUS("+"), STAR("*"), DIV("/"), NOT("!"), MINUS("-"), SELECT_FIRST("^["), SELECT_LAST("$["), QMARK( |
| 582 | 0 | "?"), PROJECT("!["), |
| 583 | 0 | GE(">="), GT(">"), LE("<="), LT("<"), EQ("=="), NE("!="), ASSIGN("="), INSTANCEOF("instanceof"), MATCHES( |
| 584 | 0 | "matches"), BETWEEN("between"), |
| 585 | 0 | SELECT("?["), MOD("%"), POWER("^"), |
| 586 | 0 | ELVIS("?:"), SAFE_NAVI("?."), BEAN_REF("@"); |
| 587 | |
|
| 588 | |
char[] tokenChars; |
| 589 | |
private boolean hasPayload; |
| 590 | |
|
| 591 | 0 | private TokenKind(String tokenString) { |
| 592 | 0 | tokenChars = tokenString.toCharArray(); |
| 593 | 0 | hasPayload = tokenChars.length == 0; |
| 594 | 0 | } |
| 595 | |
|
| 596 | |
private TokenKind() { |
| 597 | 0 | this(""); |
| 598 | 0 | } |
| 599 | |
|
| 600 | |
public String toString() { |
| 601 | 0 | return this.name() + (tokenChars.length != 0 ? "(" + new String(tokenChars) + ")" : ""); |
| 602 | |
} |
| 603 | |
|
| 604 | |
public boolean hasPayload() { |
| 605 | 0 | return hasPayload; |
| 606 | |
} |
| 607 | |
|
| 608 | |
public int getLength() { |
| 609 | 0 | return tokenChars.length; |
| 610 | |
} |
| 611 | |
} |
| 612 | |
} |