1 package org.apache.maven.surefire.util.internal;
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22 import java.nio.ByteBuffer;
23 import java.nio.CharBuffer;
24 import java.nio.charset.CharacterCodingException;
25 import java.nio.charset.Charset;
26 import java.util.StringTokenizer;
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53 public final class StringUtils
54 {
55 public static final String NL = System.getProperty( "line.separator" );
56
57 private static final byte[] HEX_CHARS = {
58 '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F' };
59
60 private static final Charset DEFAULT_CHARSET = Charset.defaultCharset();
61
62
63
64
65
66 public static final Charset US_ASCII = Charset.forName( "US-ASCII" );
67
68
69 public static final Charset ISO_8859_1 = Charset.forName( "ISO-8859-1" );
70
71 public static final Charset UTF_8 = Charset.forName( "UTF-8" );
72
73 private StringUtils()
74 {
75 throw new IllegalStateException( "no instantiable constructor" );
76 }
77
78 public static String[] split( String text, String separator )
79 {
80 final StringTokenizer tok;
81 if ( separator == null )
82 {
83
84
85 tok = new StringTokenizer( text );
86 }
87 else
88 {
89 tok = new StringTokenizer( text, separator );
90 }
91
92 String[] list = new String[tok.countTokens()];
93 for ( int i = 0; tok.hasMoreTokens(); i++ )
94 {
95 list[i] = tok.nextToken();
96 }
97 return list;
98 }
99
100
101
102
103
104
105
106
107
108 public static boolean isBlank( String str )
109 {
110 return str == null || str.trim().isEmpty();
111 }
112
113
114
115
116
117
118
119
120
121 public static boolean isNotBlank( String str )
122 {
123 return !isBlank( str );
124 }
125
126
127
128
129
130
131
132
133
134
135 @SuppressWarnings( "checkstyle:magicnumber" )
136 public static void escapeToPrintable( StringBuilder target, CharSequence str )
137 {
138 if ( target == null )
139 {
140 throw new IllegalArgumentException( "The target buffer must not be null" );
141 }
142 if ( str == null )
143 {
144 return;
145 }
146
147 for ( int i = 0; i < str.length(); i++ )
148 {
149 char c = str.charAt( i );
150
151
152 if ( c < 32 || c > 126 || c == '\\' || c == ',' )
153 {
154 target.append( '\\' );
155 target.append( (char) HEX_CHARS[( 0xF000 & c ) >> 12] );
156 target.append( (char) HEX_CHARS[( 0x0F00 & c ) >> 8] );
157 target.append( (char) HEX_CHARS[( 0x00F0 & c ) >> 4] );
158 target.append( (char) HEX_CHARS[( 0x000F & c )] );
159 }
160 else
161 {
162 target.append( c );
163 }
164 }
165 }
166
167
168
169
170
171
172
173 public static void unescapeString( StringBuilder target, CharSequence str )
174 {
175 if ( target == null )
176 {
177 throw new IllegalArgumentException( "The target buffer must not be null" );
178 }
179 if ( str == null )
180 {
181 return;
182 }
183
184 for ( int i = 0; i < str.length(); i++ )
185 {
186 char ch = str.charAt( i );
187
188 if ( ch == '\\' )
189 {
190 target.append( (char) (
191 digit( str.charAt( ++i ) ) << 12
192 | digit( str.charAt( ++i ) ) << 8
193 | digit( str.charAt( ++i ) ) << 4
194 | digit( str.charAt( ++i ) )
195 ) );
196 }
197 else
198 {
199 target.append( ch );
200 }
201 }
202 }
203
204 private static int digit( char ch )
205 {
206 if ( ch >= 'a' )
207 {
208 return 10 + ch - 'a';
209 }
210 else if ( ch >= 'A' )
211 {
212 return 10 + ch - 'A';
213 }
214 else
215 {
216 return ch - '0';
217 }
218 }
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236 @SuppressWarnings( "checkstyle:magicnumber" )
237 public static int escapeBytesToPrintable( byte[] out, int outoff, byte[] input, int off, int len )
238 {
239 if ( out == null )
240 {
241 throw new IllegalArgumentException( "The output array must not be null" );
242 }
243 if ( input == null || input.length == 0 )
244 {
245 return 0;
246 }
247 int outputPos = outoff;
248 int end = off + len;
249 for ( int i = off; i < end; i++ )
250 {
251 byte b = input[i];
252
253
254 if ( b < 32 || b > 126 || b == '\\' || b == ',' )
255 {
256 int upper = ( 0xF0 & b ) >> 4;
257 int lower = ( 0x0F & b );
258 out[outputPos++] = '\\';
259 out[outputPos++] = HEX_CHARS[upper];
260 out[outputPos++] = HEX_CHARS[lower];
261 }
262 else
263 {
264 out[outputPos++] = b;
265 }
266 }
267
268 return outputPos - outoff;
269 }
270
271
272
273
274
275
276
277
278 public static ByteBuffer unescapeBytes( String str, String charsetName )
279 {
280 int outPos = 0;
281
282 if ( str == null )
283 {
284 return ByteBuffer.wrap( new byte[0] );
285 }
286
287 byte[] out = new byte[str.length()];
288 for ( int i = 0; i < str.length(); i++ )
289 {
290 char ch = str.charAt( i );
291
292 if ( ch == '\\' )
293 {
294 int upper = digit( str.charAt( ++i ) );
295 int lower = digit( str.charAt( ++i ) );
296 out[outPos++] = (byte) ( upper << 4 | lower );
297 }
298 else
299 {
300 out[outPos++] = (byte) ch;
301 }
302 }
303
304 Charset sourceCharset = Charset.forName( charsetName );
305 if ( !DEFAULT_CHARSET.equals( sourceCharset ) )
306 {
307 CharBuffer decodedFromSourceCharset;
308 try
309 {
310 decodedFromSourceCharset = sourceCharset.newDecoder().decode( ByteBuffer.wrap( out, 0, outPos ) );
311 return DEFAULT_CHARSET.encode( decodedFromSourceCharset );
312 }
313 catch ( CharacterCodingException e )
314 {
315
316 }
317 }
318
319 return ByteBuffer.wrap( out, 0, outPos );
320 }
321
322 public static byte[] encodeStringForForkCommunication( String string )
323 {
324 return string.getBytes( ISO_8859_1 );
325 }
326
327
328
329
330
331
332
333
334 public static boolean startsWith( StringBuffer buffer, String pattern )
335 {
336 if ( buffer.length() < pattern.length() )
337 {
338 return false;
339 }
340 else
341 {
342 for ( int i = 0, len = pattern.length(); i < len; i++ )
343 {
344 if ( buffer.charAt( i ) != pattern.charAt( i ) )
345 {
346 return false;
347 }
348 }
349 return true;
350 }
351 }
352 }