forked from arduino/arduino-cli
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathterminal.go
108 lines (95 loc) · 3.3 KB
/
terminal.go
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
// This file is part of arduino-cli.
//
// Copyright 2022 ARDUINO SA (http://www.arduino.cc/)
//
// This software is released under the GNU General Public License version 3,
// which covers the main part of arduino-cli.
// The terms of this license can be found at:
// https://www.gnu.org/licenses/gpl-3.0.en.html
//
// You can be released from the requirements of the above licenses by purchasing
// a commercial license. Buying such a license is mandatory if you want to
// modify or otherwise use the software for commercial activities involving the
// Arduino software without disclosing the source code of your own applications.
// To purchase a commercial license, send an email to [email protected].
package feedback
import (
"bufio"
"errors"
"fmt"
"io"
"os"
"golang.org/x/term"
)
// InteractiveStreams returns the underlying io.Reader and io.Writer to directly stream to
// stdin and stdout. It errors if the selected output format is not Text or the terminal is
// not interactive.
func InteractiveStreams() (io.Reader, io.Writer, error) {
if !formatSelected {
panic("output format not yet selected")
}
if format != Text {
return nil, nil, errors.New(tr("interactive terminal not supported for the '%s' output format", format))
}
return os.Stdin, stdOut, nil
}
var oldStateStdin *term.State
// SetRawModeStdin sets the stdin stream in RAW mode (no buffering, echo disabled,
// no terminal escape codes nor signals interpreted)
func SetRawModeStdin() error {
if oldStateStdin != nil {
panic("terminal already in RAW mode")
}
if !IsTerminal() {
return errors.New(tr("not running in a terminal"))
}
old, err := term.MakeRaw(int(os.Stdin.Fd()))
if err != nil {
return err
}
oldStateStdin = old
return nil
}
// RestoreModeStdin restore the terminal settings to the normal non-RAW state. This
// function must be called after SetRawModeStdin to not leave the terminal in an
// undefined state.
func RestoreModeStdin() {
if oldStateStdin == nil {
return
}
_ = term.Restore(int(os.Stdin.Fd()), oldStateStdin)
oldStateStdin = nil
}
// IsTerminal returns true if there is an interactive terminal
func IsTerminal() bool {
return term.IsTerminal(int(os.Stdin.Fd()))
}
// InputUserField prompts the user to input the provided user field.
func InputUserField(prompt string, secret bool) (string, error) {
if format != Text {
return "", errors.New(tr("user input not supported for the '%s' output format", format))
}
if !IsTerminal() {
return "", errors.New(tr("user input not supported in non interactive mode"))
}
fmt.Fprintf(stdOut, "%s: ", prompt)
if secret {
// Read and return a password (no characted echoed on terminal)
value, err := term.ReadPassword(int(os.Stdin.Fd()))
fmt.Fprintln(stdOut)
return string(value), err
}
// Read and return an input line
sc := bufio.NewScanner(os.Stdin)
sc.Scan()
return sc.Text(), sc.Err()
}
// ExitWhenParentProcessEnds waits until the controlling parent process ends and then exits
// the current process. This is useful to terminate the current process when it is daemonized
// and the controlling parent process is terminated to avoid leaving zombie processes.
// It is recommended to call this function as a goroutine.
func ExitWhenParentProcessEnds() {
// Stdin is closed when the controlling parent process ends
_, _ = io.Copy(io.Discard, os.Stdin)
os.Exit(0)
}