Mercurial Hosting > luan
comparison src/org/eclipse/jetty/util/StringUtil.java @ 802:3428c60d7cfc
replace jetty jars with source
author | Franklin Schmidt <fschmidt@gmail.com> |
---|---|
date | Wed, 07 Sep 2016 21:15:48 -0600 |
parents | |
children | 8e9db0bbf4f9 |
comparison
equal
deleted
inserted
replaced
801:6a21393191c1 | 802:3428c60d7cfc |
---|---|
1 // | |
2 // ======================================================================== | |
3 // Copyright (c) 1995-2014 Mort Bay Consulting Pty. Ltd. | |
4 // ------------------------------------------------------------------------ | |
5 // All rights reserved. This program and the accompanying materials | |
6 // are made available under the terms of the Eclipse Public License v1.0 | |
7 // and Apache License v2.0 which accompanies this distribution. | |
8 // | |
9 // The Eclipse Public License is available at | |
10 // http://www.eclipse.org/legal/epl-v10.html | |
11 // | |
12 // The Apache License v2.0 is available at | |
13 // http://www.opensource.org/licenses/apache2.0.php | |
14 // | |
15 // You may elect to redistribute this code under either of these licenses. | |
16 // ======================================================================== | |
17 // | |
18 | |
19 package org.eclipse.jetty.util; | |
20 | |
21 import java.io.UnsupportedEncodingException; | |
22 import java.nio.charset.Charset; | |
23 | |
24 import org.eclipse.jetty.util.log.Log; | |
25 import org.eclipse.jetty.util.log.Logger; | |
26 | |
27 /** Fast String Utilities. | |
28 * | |
29 * These string utilities provide both conveniance methods and | |
30 * performance improvements over most standard library versions. The | |
31 * main aim of the optimizations is to avoid object creation unless | |
32 * absolutely required. | |
33 * | |
34 * | |
35 */ | |
36 public class StringUtil | |
37 { | |
38 private static final Logger LOG = Log.getLogger(StringUtil.class); | |
39 | |
40 public static final String ALL_INTERFACES="0.0.0.0"; | |
41 public static final String CRLF="\015\012"; | |
42 public static final String __LINE_SEPARATOR= | |
43 System.getProperty("line.separator","\n"); | |
44 | |
45 public static final String __ISO_8859_1="ISO-8859-1"; | |
46 public final static String __UTF8="UTF-8"; | |
47 public final static String __UTF8Alt="UTF8"; | |
48 public final static String __UTF16="UTF-16"; | |
49 | |
50 public final static Charset __UTF8_CHARSET; | |
51 public final static Charset __ISO_8859_1_CHARSET; | |
52 | |
53 static | |
54 { | |
55 __UTF8_CHARSET=Charset.forName(__UTF8); | |
56 __ISO_8859_1_CHARSET=Charset.forName(__ISO_8859_1); | |
57 } | |
58 | |
59 private static char[] lowercases = { | |
60 '\000','\001','\002','\003','\004','\005','\006','\007', | |
61 '\010','\011','\012','\013','\014','\015','\016','\017', | |
62 '\020','\021','\022','\023','\024','\025','\026','\027', | |
63 '\030','\031','\032','\033','\034','\035','\036','\037', | |
64 '\040','\041','\042','\043','\044','\045','\046','\047', | |
65 '\050','\051','\052','\053','\054','\055','\056','\057', | |
66 '\060','\061','\062','\063','\064','\065','\066','\067', | |
67 '\070','\071','\072','\073','\074','\075','\076','\077', | |
68 '\100','\141','\142','\143','\144','\145','\146','\147', | |
69 '\150','\151','\152','\153','\154','\155','\156','\157', | |
70 '\160','\161','\162','\163','\164','\165','\166','\167', | |
71 '\170','\171','\172','\133','\134','\135','\136','\137', | |
72 '\140','\141','\142','\143','\144','\145','\146','\147', | |
73 '\150','\151','\152','\153','\154','\155','\156','\157', | |
74 '\160','\161','\162','\163','\164','\165','\166','\167', | |
75 '\170','\171','\172','\173','\174','\175','\176','\177' }; | |
76 | |
77 /* ------------------------------------------------------------ */ | |
78 /** | |
79 * fast lower case conversion. Only works on ascii (not unicode) | |
80 * @param s the string to convert | |
81 * @return a lower case version of s | |
82 */ | |
83 public static String asciiToLowerCase(String s) | |
84 { | |
85 char[] c = null; | |
86 int i=s.length(); | |
87 | |
88 // look for first conversion | |
89 while (i-->0) | |
90 { | |
91 char c1=s.charAt(i); | |
92 if (c1<=127) | |
93 { | |
94 char c2=lowercases[c1]; | |
95 if (c1!=c2) | |
96 { | |
97 c=s.toCharArray(); | |
98 c[i]=c2; | |
99 break; | |
100 } | |
101 } | |
102 } | |
103 | |
104 while (i-->0) | |
105 { | |
106 if(c[i]<=127) | |
107 c[i] = lowercases[c[i]]; | |
108 } | |
109 | |
110 return c==null?s:new String(c); | |
111 } | |
112 | |
113 | |
114 /* ------------------------------------------------------------ */ | |
115 public static boolean startsWithIgnoreCase(String s,String w) | |
116 { | |
117 if (w==null) | |
118 return true; | |
119 | |
120 if (s==null || s.length()<w.length()) | |
121 return false; | |
122 | |
123 for (int i=0;i<w.length();i++) | |
124 { | |
125 char c1=s.charAt(i); | |
126 char c2=w.charAt(i); | |
127 if (c1!=c2) | |
128 { | |
129 if (c1<=127) | |
130 c1=lowercases[c1]; | |
131 if (c2<=127) | |
132 c2=lowercases[c2]; | |
133 if (c1!=c2) | |
134 return false; | |
135 } | |
136 } | |
137 return true; | |
138 } | |
139 | |
140 /* ------------------------------------------------------------ */ | |
141 public static boolean endsWithIgnoreCase(String s,String w) | |
142 { | |
143 if (w==null) | |
144 return true; | |
145 | |
146 if (s==null) | |
147 return false; | |
148 | |
149 int sl=s.length(); | |
150 int wl=w.length(); | |
151 | |
152 if (sl<wl) | |
153 return false; | |
154 | |
155 for (int i=wl;i-->0;) | |
156 { | |
157 char c1=s.charAt(--sl); | |
158 char c2=w.charAt(i); | |
159 if (c1!=c2) | |
160 { | |
161 if (c1<=127) | |
162 c1=lowercases[c1]; | |
163 if (c2<=127) | |
164 c2=lowercases[c2]; | |
165 if (c1!=c2) | |
166 return false; | |
167 } | |
168 } | |
169 return true; | |
170 } | |
171 | |
172 /* ------------------------------------------------------------ */ | |
173 /** | |
174 * returns the next index of a character from the chars string | |
175 */ | |
176 public static int indexFrom(String s,String chars) | |
177 { | |
178 for (int i=0;i<s.length();i++) | |
179 if (chars.indexOf(s.charAt(i))>=0) | |
180 return i; | |
181 return -1; | |
182 } | |
183 | |
184 /* ------------------------------------------------------------ */ | |
185 /** | |
186 * replace substrings within string. | |
187 */ | |
188 public static String replace(String s, String sub, String with) | |
189 { | |
190 int c=0; | |
191 int i=s.indexOf(sub,c); | |
192 if (i == -1) | |
193 return s; | |
194 | |
195 StringBuilder buf = new StringBuilder(s.length()+with.length()); | |
196 | |
197 do | |
198 { | |
199 buf.append(s.substring(c,i)); | |
200 buf.append(with); | |
201 c=i+sub.length(); | |
202 } while ((i=s.indexOf(sub,c))!=-1); | |
203 | |
204 if (c<s.length()) | |
205 buf.append(s.substring(c,s.length())); | |
206 | |
207 return buf.toString(); | |
208 | |
209 } | |
210 | |
211 | |
212 /* ------------------------------------------------------------ */ | |
213 /** Remove single or double quotes. | |
214 */ | |
215 public static String unquote(String s) | |
216 { | |
217 return QuotedStringTokenizer.unquote(s); | |
218 } | |
219 | |
220 | |
221 /* ------------------------------------------------------------ */ | |
222 /** Append substring to StringBuilder | |
223 * @param buf StringBuilder to append to | |
224 * @param s String to append from | |
225 * @param offset The offset of the substring | |
226 * @param length The length of the substring | |
227 */ | |
228 public static void append(StringBuilder buf, | |
229 String s, | |
230 int offset, | |
231 int length) | |
232 { | |
233 synchronized(buf) | |
234 { | |
235 int end=offset+length; | |
236 for (int i=offset; i<end;i++) | |
237 { | |
238 if (i>=s.length()) | |
239 break; | |
240 buf.append(s.charAt(i)); | |
241 } | |
242 } | |
243 } | |
244 | |
245 | |
246 /* ------------------------------------------------------------ */ | |
247 /** | |
248 * append hex digit | |
249 * | |
250 */ | |
251 public static void append(StringBuilder buf,byte b,int base) | |
252 { | |
253 int bi=0xff&b; | |
254 int c='0'+(bi/base)%base; | |
255 if (c>'9') | |
256 c= 'a'+(c-'0'-10); | |
257 buf.append((char)c); | |
258 c='0'+bi%base; | |
259 if (c>'9') | |
260 c= 'a'+(c-'0'-10); | |
261 buf.append((char)c); | |
262 } | |
263 | |
264 /* ------------------------------------------------------------ */ | |
265 public static void append2digits(StringBuffer buf,int i) | |
266 { | |
267 if (i<100) | |
268 { | |
269 buf.append((char)(i/10+'0')); | |
270 buf.append((char)(i%10+'0')); | |
271 } | |
272 } | |
273 | |
274 /* ------------------------------------------------------------ */ | |
275 public static void append2digits(StringBuilder buf,int i) | |
276 { | |
277 if (i<100) | |
278 { | |
279 buf.append((char)(i/10+'0')); | |
280 buf.append((char)(i%10+'0')); | |
281 } | |
282 } | |
283 | |
284 /* ------------------------------------------------------------ */ | |
285 /** Return a non null string. | |
286 * @param s String | |
287 * @return The string passed in or empty string if it is null. | |
288 */ | |
289 public static String nonNull(String s) | |
290 { | |
291 if (s==null) | |
292 return ""; | |
293 return s; | |
294 } | |
295 | |
296 /* ------------------------------------------------------------ */ | |
297 public static boolean equals(String s,char[] buf, int offset, int length) | |
298 { | |
299 if (s.length()!=length) | |
300 return false; | |
301 for (int i=0;i<length;i++) | |
302 if (buf[offset+i]!=s.charAt(i)) | |
303 return false; | |
304 return true; | |
305 } | |
306 | |
307 /* ------------------------------------------------------------ */ | |
308 public static String toUTF8String(byte[] b,int offset,int length) | |
309 { | |
310 try | |
311 { | |
312 return new String(b,offset,length,__UTF8); | |
313 } | |
314 catch (UnsupportedEncodingException e) | |
315 { | |
316 throw new IllegalArgumentException(e); | |
317 } | |
318 } | |
319 | |
320 /* ------------------------------------------------------------ */ | |
321 public static String toString(byte[] b,int offset,int length,String charset) | |
322 { | |
323 try | |
324 { | |
325 return new String(b,offset,length,charset); | |
326 } | |
327 catch (UnsupportedEncodingException e) | |
328 { | |
329 throw new IllegalArgumentException(e); | |
330 } | |
331 } | |
332 | |
333 | |
334 /* ------------------------------------------------------------ */ | |
335 public static boolean isUTF8(String charset) | |
336 { | |
337 return __UTF8.equalsIgnoreCase(charset)||__UTF8Alt.equalsIgnoreCase(charset); | |
338 } | |
339 | |
340 | |
341 /* ------------------------------------------------------------ */ | |
342 public static String printable(String name) | |
343 { | |
344 if (name==null) | |
345 return null; | |
346 StringBuilder buf = new StringBuilder(name.length()); | |
347 for (int i=0;i<name.length();i++) | |
348 { | |
349 char c=name.charAt(i); | |
350 if (!Character.isISOControl(c)) | |
351 buf.append(c); | |
352 } | |
353 return buf.toString(); | |
354 } | |
355 | |
356 /* ------------------------------------------------------------ */ | |
357 public static String printable(byte[] b) | |
358 { | |
359 StringBuilder buf = new StringBuilder(); | |
360 for (int i=0;i<b.length;i++) | |
361 { | |
362 char c=(char)b[i]; | |
363 if (Character.isWhitespace(c)|| c>' ' && c<0x7f) | |
364 buf.append(c); | |
365 else | |
366 { | |
367 buf.append("0x"); | |
368 TypeUtil.toHex(b[i],buf); | |
369 } | |
370 } | |
371 return buf.toString(); | |
372 } | |
373 | |
374 public static byte[] getBytes(String s) | |
375 { | |
376 try | |
377 { | |
378 return s.getBytes(__ISO_8859_1); | |
379 } | |
380 catch(Exception e) | |
381 { | |
382 LOG.warn(e); | |
383 return s.getBytes(); | |
384 } | |
385 } | |
386 | |
387 public static byte[] getBytes(String s,String charset) | |
388 { | |
389 try | |
390 { | |
391 return s.getBytes(charset); | |
392 } | |
393 catch(Exception e) | |
394 { | |
395 LOG.warn(e); | |
396 return s.getBytes(); | |
397 } | |
398 } | |
399 | |
400 | |
401 | |
402 /** | |
403 * Converts a binary SID to a string SID | |
404 * | |
405 * http://en.wikipedia.org/wiki/Security_Identifier | |
406 * | |
407 * S-1-IdentifierAuthority-SubAuthority1-SubAuthority2-...-SubAuthorityn | |
408 */ | |
409 public static String sidBytesToString(byte[] sidBytes) | |
410 { | |
411 StringBuilder sidString = new StringBuilder(); | |
412 | |
413 // Identify this as a SID | |
414 sidString.append("S-"); | |
415 | |
416 // Add SID revision level (expect 1 but may change someday) | |
417 sidString.append(Byte.toString(sidBytes[0])).append('-'); | |
418 | |
419 StringBuilder tmpBuilder = new StringBuilder(); | |
420 | |
421 // crunch the six bytes of issuing authority value | |
422 for (int i = 2; i <= 7; ++i) | |
423 { | |
424 tmpBuilder.append(Integer.toHexString(sidBytes[i] & 0xFF)); | |
425 } | |
426 | |
427 sidString.append(Long.parseLong(tmpBuilder.toString(), 16)); // '-' is in the subauth loop | |
428 | |
429 // the number of subAuthorities we need to attach | |
430 int subAuthorityCount = sidBytes[1]; | |
431 | |
432 // attach each of the subAuthorities | |
433 for (int i = 0; i < subAuthorityCount; ++i) | |
434 { | |
435 int offset = i * 4; | |
436 tmpBuilder.setLength(0); | |
437 // these need to be zero padded hex and little endian | |
438 tmpBuilder.append(String.format("%02X%02X%02X%02X", | |
439 (sidBytes[11 + offset] & 0xFF), | |
440 (sidBytes[10 + offset] & 0xFF), | |
441 (sidBytes[9 + offset] & 0xFF), | |
442 (sidBytes[8 + offset] & 0xFF))); | |
443 sidString.append('-').append(Long.parseLong(tmpBuilder.toString(), 16)); | |
444 } | |
445 | |
446 return sidString.toString(); | |
447 } | |
448 | |
449 /** | |
450 * Converts a string SID to a binary SID | |
451 * | |
452 * http://en.wikipedia.org/wiki/Security_Identifier | |
453 * | |
454 * S-1-IdentifierAuthority-SubAuthority1-SubAuthority2-...-SubAuthorityn | |
455 */ | |
456 public static byte[] sidStringToBytes( String sidString ) | |
457 { | |
458 String[] sidTokens = sidString.split("-"); | |
459 | |
460 int subAuthorityCount = sidTokens.length - 3; // S-Rev-IdAuth- | |
461 | |
462 int byteCount = 0; | |
463 byte[] sidBytes = new byte[1 + 1 + 6 + (4 * subAuthorityCount)]; | |
464 | |
465 // the revision byte | |
466 sidBytes[byteCount++] = (byte)Integer.parseInt(sidTokens[1]); | |
467 | |
468 // the # of sub authorities byte | |
469 sidBytes[byteCount++] = (byte)subAuthorityCount; | |
470 | |
471 // the certAuthority | |
472 String hexStr = Long.toHexString(Long.parseLong(sidTokens[2])); | |
473 | |
474 while( hexStr.length() < 12) // pad to 12 characters | |
475 { | |
476 hexStr = "0" + hexStr; | |
477 } | |
478 | |
479 // place the certAuthority 6 bytes | |
480 for ( int i = 0 ; i < hexStr.length(); i = i + 2) | |
481 { | |
482 sidBytes[byteCount++] = (byte)Integer.parseInt(hexStr.substring(i, i + 2),16); | |
483 } | |
484 | |
485 | |
486 for ( int i = 3; i < sidTokens.length ; ++i) | |
487 { | |
488 hexStr = Long.toHexString(Long.parseLong(sidTokens[i])); | |
489 | |
490 while( hexStr.length() < 8) // pad to 8 characters | |
491 { | |
492 hexStr = "0" + hexStr; | |
493 } | |
494 | |
495 // place the inverted sub authorities, 4 bytes each | |
496 for ( int j = hexStr.length(); j > 0; j = j - 2) | |
497 { | |
498 sidBytes[byteCount++] = (byte)Integer.parseInt(hexStr.substring(j-2, j),16); | |
499 } | |
500 } | |
501 | |
502 return sidBytes; | |
503 } | |
504 } |