forked from kangjianwei/LearningJDK
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Console.java
656 lines (586 loc) · 26.3 KB
/
Console.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
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
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
/*
* Copyright (c) 2005, 2013, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package java.io;
import java.util.*;
import java.nio.charset.Charset;
import jdk.internal.misc.JavaIOAccess;
import jdk.internal.misc.JavaLangAccess;
import jdk.internal.misc.SharedSecrets;
import sun.nio.cs.StreamDecoder;
import sun.nio.cs.StreamEncoder;
/**
* Methods to access the character-based console device, if any, associated
* with the current Java virtual machine.
*
* <p> Whether a virtual machine has a console is dependent upon the
* underlying platform and also upon the manner in which the virtual
* machine is invoked. If the virtual machine is started from an
* interactive command line without redirecting the standard input and
* output streams then its console will exist and will typically be
* connected to the keyboard and display from which the virtual machine
* was launched. If the virtual machine is started automatically, for
* example by a background job scheduler, then it will typically not
* have a console.
* <p>
* If this virtual machine has a console then it is represented by a
* unique instance of this class which can be obtained by invoking the
* {@link java.lang.System#console()} method. If no console device is
* available then an invocation of that method will return {@code null}.
* <p>
* Read and write operations are synchronized to guarantee the atomic
* completion of critical operations; therefore invoking methods
* {@link #readLine()}, {@link #readPassword()}, {@link #format format()},
* {@link #printf printf()} as well as the read, format and write operations
* on the objects returned by {@link #reader()} and {@link #writer()} may
* block in multithreaded scenarios.
* <p>
* Invoking {@code close()} on the objects returned by the {@link #reader()}
* and the {@link #writer()} will not close the underlying stream of those
* objects.
* <p>
* The console-read methods return {@code null} when the end of the
* console input stream is reached, for example by typing control-D on
* Unix or control-Z on Windows. Subsequent read operations will succeed
* if additional characters are later entered on the console's input
* device.
* <p>
* Unless otherwise specified, passing a {@code null} argument to any method
* in this class will cause a {@link NullPointerException} to be thrown.
* <p>
* <b>Security note:</b>
* If an application needs to read a password or other secure data, it should
* use {@link #readPassword()} or {@link #readPassword(String, Object...)} and
* manually zero the returned character array after processing to minimize the
* lifetime of sensitive data in memory.
*
* <blockquote><pre>{@code
* Console cons;
* char[] passwd;
* if ((cons = System.console()) != null &&
* (passwd = cons.readPassword("[%s]", "Password:")) != null) {
* ...
* java.util.Arrays.fill(passwd, ' ');
* }
* }</pre></blockquote>
*
* @author Xueming Shen
* @since 1.6
*/
// 终端对象,往往关联着标准输入流和标准输出流。在非tty环境下(比如IDE中),无法获取终端对象。
public final class Console implements Flushable {
private Object readLock;
private Object writeLock;
private static Console cons;
private Reader reader; // 当前终端关联的标准输入流
private Writer out; // 当前终端关联的标准输出流
private PrintWriter pw;
private Formatter formatter;
private Charset cs;
private char[] rcb;
private boolean restoreEcho;
private boolean shutdownHookInstalled; // 指示属于Console类的钩子是否已注册
// Set up JavaIOAccess in SharedSecrets
static {
SharedSecrets.setJavaIOAccess(new JavaIOAccess() {
public Console console() {
// 如果处于终端环境下
if(istty()) {
if(cons == null) {
cons = new Console();
}
return cons;
}
return null;
}
public Charset charset() {
// This method is called in sun.security.util.Password, cons already exists when this method is called
return cons.cs;
}
});
}
// 构造中终端对象,需要从System类中获取
private Console() {
readLock = new Object();
writeLock = new Object();
// 当前系统编码名称
String csname = encoding();
if(csname != null) {
try {
cs = Charset.forName(csname);
} catch(Exception x) {
}
}
if(cs == null) {
// 返回Java虚拟机的默认字符集
cs = Charset.defaultCharset();
}
// 直接使用文件描述符初始化输出流,FileDescriptor.out指示标准输出流,被封装为System.out
FileOutputStream fos = new FileOutputStream(FileDescriptor.out);
out = StreamEncoder.forOutputStreamWriter(fos, writeLock, cs);
pw = new PrintWriter(out, true) {
public void close() {
}
};
formatter = new Formatter(out);
// 直接使用文件描述符初始化输入流,FileDescriptor.in指示标准输入流,被封装为System.in
FileInputStream fis = new FileInputStream(FileDescriptor.in);
reader = new LineReader(StreamDecoder.forInputStreamReader(fis, readLock, cs));
rcb = new char[1024];
}
/**
* Retrieves the unique {@link java.io.PrintWriter PrintWriter} object
* associated with this console.
*
* @return The printwriter associated with this console
*/
// 返回定向到标准输出流的Writer
public PrintWriter writer() {
return pw;
}
/**
* Retrieves the unique {@link java.io.Reader Reader} object associated
* with this console.
* <p>
* This method is intended to be used by sophisticated applications, for
* example, a {@link java.util.Scanner} object which utilizes the rich
* parsing/scanning functionality provided by the {@code Scanner}:
* <blockquote><pre>
* Console con = System.console();
* if (con != null) {
* Scanner sc = new Scanner(con.reader());
* ...
* }
* </pre></blockquote>
* <p>
* For simple applications requiring only line-oriented reading, use
* {@link #readLine}.
* <p>
* The bulk read operations {@link java.io.Reader#read(char[]) read(char[]) },
* {@link java.io.Reader#read(char[], int, int) read(char[], int, int) } and
* {@link java.io.Reader#read(java.nio.CharBuffer) read(java.nio.CharBuffer)}
* on the returned object will not read in characters beyond the line
* bound for each invocation, even if the destination buffer has space for
* more characters. The {@code Reader}'s {@code read} methods may block if a
* line bound has not been entered or reached on the console's input device.
* A line bound is considered to be any one of a line feed ({@code '\n'}),
* a carriage return ({@code '\r'}), a carriage return followed immediately
* by a linefeed, or an end of stream.
*
* @return The reader associated with this console
*/
// 返回定向到标准输入流的Reader
public Reader reader() {
return reader;
}
/**
* A convenience method to write a formatted string to this console's
* output stream using the specified format string and arguments.
*
* <p> An invocation of this method of the form
* {@code con.printf(format, args)} behaves in exactly the same way
* as the invocation of
* <pre>con.format(format, args)</pre>.
*
* @param format A format string as described in <a
* href="../util/Formatter.html#syntax">Format string syntax</a>.
* @param args Arguments referenced by the format specifiers in the format
* string. If there are more arguments than format specifiers, the
* extra arguments are ignored. The number of arguments is
* variable and may be zero. The maximum number of arguments is
* limited by the maximum dimension of a Java array as defined by
* <cite>The Java™ Virtual Machine Specification</cite>.
* The behaviour on a
* {@code null} argument depends on the <a
* href="../util/Formatter.html#syntax">conversion</a>.
*
* @return This console
*
* @throws IllegalFormatException If a format string contains an illegal syntax, a format
* specifier that is incompatible with the given arguments,
* insufficient arguments given the format string, or other
* illegal conditions. For specification of all possible
* formatting errors, see the <a
* href="../util/Formatter.html#detail">Details</a> section of the
* formatter class specification.
*/
// 按指定的格式打印字符串
public Console printf(String format, Object... args) {
return format(format, args);
}
/**
* Writes a formatted string to this console's output stream using
* the specified format string and arguments.
*
* @param fmt A format string as described in <a
* href="../util/Formatter.html#syntax">Format string syntax</a>
* @param args Arguments referenced by the format specifiers in the format
* string. If there are more arguments than format specifiers, the
* extra arguments are ignored. The number of arguments is
* variable and may be zero. The maximum number of arguments is
* limited by the maximum dimension of a Java array as defined by
* <cite>The Java™ Virtual Machine Specification</cite>.
* The behaviour on a
* {@code null} argument depends on the <a
* href="../util/Formatter.html#syntax">conversion</a>.
*
* @return This console
*
* @throws IllegalFormatException If a format string contains an illegal syntax, a format
* specifier that is incompatible with the given arguments,
* insufficient arguments given the format string, or other
* illegal conditions. For specification of all possible
* formatting errors, see the <a
* href="../util/Formatter.html#detail">Details</a> section
* of the formatter class specification.
*/
// 按指定的格式打印字符串
public Console format(String fmt, Object... args) {
formatter.format(fmt, args).flush();
return this;
}
/**
* Reads a single line of text from the console.
*
* @return A string containing the line read from the console, not
* including any line-termination characters, or {@code null}
* if an end of stream has been reached.
*
* @throws IOError If an I/O error occurs.
*/
// 读取单行数据
public String readLine() {
return readLine("");
}
/**
* Provides a formatted prompt, then reads a single line of text from the console.
*
* @param fmt A format string as described in <a
* href="../util/Formatter.html#syntax">Format string syntax</a>.
* @param args Arguments referenced by the format specifiers in the format
* string. If there are more arguments than format specifiers, the
* extra arguments are ignored. The maximum number of arguments is
* limited by the maximum dimension of a Java array as defined by
* <cite>The Java™ Virtual Machine Specification</cite>.
*
* @return A string containing the line read from the console, not
* including any line-termination characters, or {@code null}
* if an end of stream has been reached.
*
* @throws IllegalFormatException If a format string contains an illegal syntax, a format
* specifier that is incompatible with the given arguments,
* insufficient arguments given the format string, or other
* illegal conditions. For specification of all possible
* formatting errors, see the <a
* href="../util/Formatter.html#detail">Details</a> section
* of the formatter class specification.
* @throws IOError If an I/O error occurs.
*/
// 读取单行数据,允许先打印指定格式的输入提示语句
public String readLine(String fmt, Object... args) {
String line = null;
synchronized(writeLock) {
synchronized(readLock) {
if(fmt.length() != 0) {
pw.format(fmt, args);
}
try {
char[] ca = readline(false);
if(ca != null) {
line = new String(ca);
}
} catch(IOException x) {
throw new IOError(x);
}
}
}
return line;
}
/**
* Reads a password or passphrase from the console with echoing disabled
*
* @return A character array containing the password or passphrase read
* from the console, not including any line-termination characters,
* or {@code null} if an end of stream has been reached.
*
* @throws IOError If an I/O error occurs.
*/
// 读取密码信息
public char[] readPassword() {
return readPassword("");
}
/**
* Provides a formatted prompt, then reads a password or passphrase from
* the console with echoing disabled.
*
* @param fmt A format string as described in <a
* href="../util/Formatter.html#syntax">Format string syntax</a>
* for the prompt text.
* @param args Arguments referenced by the format specifiers in the format
* string. If there are more arguments than format specifiers, the
* extra arguments are ignored. The maximum number of arguments is
* limited by the maximum dimension of a Java array as defined by
* <cite>The Java™ Virtual Machine Specification</cite>.
*
* @return A character array containing the password or passphrase read
* from the console, not including any line-termination characters,
* or {@code null} if an end of stream has been reached.
*
* @throws IllegalFormatException If a format string contains an illegal syntax, a format
* specifier that is incompatible with the given arguments,
* insufficient arguments given the format string, or other
* illegal conditions. For specification of all possible
* formatting errors, see the <a
* href="../util/Formatter.html#detail">Details</a>
* section of the formatter class specification.
* @throws IOError If an I/O error occurs.
*/
// 读取指定格式的密码信息,允许先打印指定格式的输入提示语句
public char[] readPassword(String fmt, Object... args) {
char[] passwd = null;
synchronized(writeLock) {
synchronized(readLock) {
// 注册一个在虚拟机关闭时执行的钩子:开启控制台回显
installShutdownHook();
try {
restoreEcho = echo(false);
} catch(IOException x) {
throw new IOError(x);
}
IOError ioe = null;
try {
if(fmt.length() != 0) {
pw.format(fmt, args);
}
passwd = readline(true);
} catch(IOException x) {
ioe = new IOError(x);
} finally {
try {
if(restoreEcho) {
restoreEcho = echo(true);
}
} catch(IOException x) {
if(ioe == null) {
ioe = new IOError(x);
} else {
ioe.addSuppressed(x);
}
}
if(ioe != null) {
throw ioe;
}
}
pw.println();
}
}
return passwd;
}
/**
* Flushes the console and forces any buffered output to be written immediately .
*/
// 刷新控制台
public void flush() {
pw.flush();
}
private static native String encoding();
/**
* Sets the console echo status to {@code on} and returns the previous console on/off status.
*
* @param on the echo status to set to. {@code true} for echo on and {@code false} for echo off
*
* @return true if the previous console echo status is on
*/
// 是否开启控制台回显,在输入密码时通常需要关闭回显
private static native boolean echo(boolean on) throws IOException;
// 是否在终端环境下
private static native boolean istty();
// 注册一个在虚拟机关闭时执行的钩子:开启控制台回显
private void installShutdownHook() {
// 已注册
if(shutdownHookInstalled) {
return;
}
try {
/* Add a shutdown hook to restore console's echo state should it be necessary */
JavaLangAccess javaLangAccess = SharedSecrets.getJavaLangAccess();
// 将指定的钩子注册到0号插槽中,以便在虚拟机关闭时处理这些钩子
javaLangAccess.registerShutdownHook(0 /* shutdown hook invocation order */, false /* only register if shutdown is not in progress */, new Runnable() {
public void run() {
try {
if(restoreEcho) {
echo(true); // 开启控制台回显
}
} catch(IOException x) {
}
}
});
} catch(IllegalStateException e) {
// shutdown is already in progress and readPassword is first used by a shutdown hook
}
// 标记属于Console类的钩子已注册
shutdownHookInstalled = true;
}
private char[] readline(boolean zeroOut) throws IOException {
// 尝试从字符输入流中读取length个char,并将其填充到rcb中。返回实际填充的字符数量
int len = reader.read(rcb, 0, rcb.length);
if(len<0) {
return null; //EOL
}
// 如果以'\r'结尾,直接移除
if(rcb[len - 1] == '\r') {
len--; //remove CR at end;
// 如果以'\n'结尾,尝试移除"\r\n"
} else if(rcb[len - 1] == '\n') {
len--; //remove LF at end;
if(len>0 && rcb[len - 1] == '\r') {
len--; //remove the CR, if there is one
}
}
char[] b = new char[len];
if(len>0) {
System.arraycopy(rcb, 0, b, 0, len);
// 如果需要清空缓冲区
if(zeroOut) {
Arrays.fill(rcb, 0, len, ' ');
}
}
return b;
}
private char[] grow() {
assert Thread.holdsLock(readLock);
char[] t = new char[rcb.length * 2];
System.arraycopy(rcb, 0, t, 0, rcb.length);
rcb = t;
return rcb;
}
class LineReader extends Reader {
boolean leftoverLF;
private Reader in;
private char[] cb;
private int nChars, nextChar;
LineReader(Reader in) {
this.in = in;
cb = new char[1024];
nextChar = nChars = 0;
leftoverLF = false;
}
public void close() {
}
public boolean ready() throws IOException {
//in.ready synchronizes on readLock already
return in.ready();
}
public int read(char[] cbuf, int offset, int length) throws IOException {
int off = offset;
int end = offset + length;
if(offset<0 || offset>cbuf.length || length<0 || end<0 || end>cbuf.length) {
throw new IndexOutOfBoundsException();
}
synchronized(readLock) {
boolean eof = false;
char c = 0;
for(; ; ) {
if(nextChar >= nChars) { //fill
int n = 0;
do {
n = in.read(cb, 0, cb.length);
} while(n == 0);
if(n>0) {
nChars = n;
nextChar = 0;
if(n<cb.length && cb[n - 1] != '\n' && cb[n - 1] != '\r') {
/*
* we're in canonical mode so each "fill" should
* come back with an eol. if there no lf or nl at
* the end of returned bytes we reached an eof.
*/
eof = true;
}
} else { /*EOF*/
if(off - offset == 0)
return -1;
return off - offset;
}
}
if(leftoverLF && cbuf == rcb && cb[nextChar] == '\n') {
/*
* if invoked by our readline, skip the leftover, otherwise
* return the LF.
*/
nextChar++;
}
leftoverLF = false;
while(nextChar<nChars) {
c = cbuf[off++] = cb[nextChar];
cb[nextChar++] = 0;
if(c == '\n') {
return off - offset;
} else if(c == '\r') {
if(off == end) {
/* no space left even the next is LF, so return
* whatever we have if the invoker is not our
* readLine()
*/
if(cbuf == rcb) {
cbuf = grow();
end = cbuf.length;
} else {
leftoverLF = true;
return off - offset;
}
}
if(nextChar == nChars && in.ready()) {
/*
* we have a CR and we reached the end of
* the read in buffer, fill to make sure we
* don't miss a LF, if there is one, it's possible
* that it got cut off during last round reading
* simply because the read in buffer was full.
*/
nChars = in.read(cb, 0, cb.length);
nextChar = 0;
}
if(nextChar<nChars && cb[nextChar] == '\n') {
cbuf[off++] = '\n';
nextChar++;
}
return off - offset;
} else if(off == end) {
if(cbuf == rcb) {
cbuf = grow();
end = cbuf.length;
} else {
return off - offset;
}
}
}
if(eof) {
return off - offset;
}
}
}
}
}
}