blob: faa02f81dbd9c054f38834677cd716e0fa3eaa1c [file] [log] [blame]
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -07001// This app provides a simple scripted environment for running common veyron
2// services as subprocesses and testing interactions between them. It is
3// structured as an interpreter, with global variables and variable
4// expansion, but no control flow. The command set that it supports is
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -07005// extendable by adding new 'commands' that implement the API defined
6// by veyron/lib/modules.
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -07007package main
8
9import (
10 "bufio"
11 "flag"
12 "fmt"
13 "os"
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -070014 "strconv"
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -070015 "strings"
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -070016 "time"
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -070017 "unicode"
18
Jiri Simsa764efb72014-12-25 20:57:03 -080019 "v.io/core/veyron2"
Matt Rosencrantzd599e382015-01-12 11:13:32 -080020 "v.io/core/veyron2/context"
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -070021
Jiri Simsa764efb72014-12-25 20:57:03 -080022 "v.io/core/veyron/lib/expect"
23 "v.io/core/veyron/lib/modules"
24 _ "v.io/core/veyron/lib/modules/core"
25 _ "v.io/core/veyron/profiles"
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -070026)
27
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -070028type cmdState struct {
29 modules.Handle
30 *expect.Session
31 line string
32}
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -070033
34var (
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -070035 interactive bool
Cosmos Nicolaoua1a0b162015-01-30 10:57:37 -080036 filename string
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -070037 handles map[string]*cmdState
Cosmos Nicolaoub44f2392014-10-28 22:41:49 -070038 jsonDict map[string]string
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -070039)
40
41func init() {
42 flag.BoolVar(&interactive, "interactive", true, "set interactive/batch mode")
Cosmos Nicolaoua1a0b162015-01-30 10:57:37 -080043 flag.StringVar(&filename, "file", "", "command file")
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -070044 handles = make(map[string]*cmdState)
Cosmos Nicolaoub44f2392014-10-28 22:41:49 -070045 jsonDict = make(map[string]string)
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -070046 flag.Usage = usage
47}
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -070048
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -070049var usage = func() {
50 fmt.Println(
51 `Welcome to this simple shell that lets you run mount tables, a simple server
52and sundry other commands from an interactive command line or as scripts. Type
53'help' at the prompt to see a list of available commands, or 'help command' to
54get specific help about that command. The shell provides environment variables
55with expansion and intrinsic support for managing subprocess, but it does not
56provide any flow control commands.
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -070057
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -070058All commands, except builtin ones (such as help, set, eval etc) are run
59asynchronously in background. That is, the prompt returns as soon as they are
60started and no output is displayed from them unless an error is encountered
61when they are being started. Each input line is numbered and that number is
62used to refer to the standard output of previous started commands. The variable
63_ always contains the number of the immediately preceeding line. It is
64possible to read the output of a command (using the 'read' builtin) and assign
65it that output to an environment variable. The 'eval' builtin parses output of
66the form <var>=<val>. In this way subproccess may be started, their output
67read and used to configure subsequent subprocesses. For example:
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -070068
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700691> time
702> read 1 t
713> print $t
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -070072
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -070073will print the first line of output from the time command, as will the
74following:
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -070075
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -070076or:
77time
78read $_ t
79print $t
80
81The eval builtin is used to directly to assign to variables specified
82in the output of the command. For example, if the root command
83prints out MT_NAME=foo then eval will set MT_NAME to foo as follows:
84
85root
86eval $_
87print $MT_NAME
88
89will print the value of MT_NAME that is output by the root command.
90`)
91 flag.PrintDefaults()
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -070092}
93
94func prompt(lineno int) {
95 if interactive {
96 fmt.Printf("%d> ", lineno)
97 }
98}
99
Matt Rosencrantzd599e382015-01-12 11:13:32 -0800100var ctx *context.T
Matt Rosencrantzc13446b2014-12-03 10:37:00 -0800101
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700102func main() {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800103 var shutdown veyron2.Shutdown
104 ctx, shutdown = veyron2.Init()
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700105
Cosmos Nicolaoua1a0b162015-01-30 10:57:37 -0800106 input := os.Stdin
107 if len(filename) > 0 {
108 f, err := os.Open(filename)
109 if err != nil {
110 fmt.Fprintf(os.Stderr, "unexpected error: %s\n", err)
111 os.Exit(1)
112 }
113 input = f
114 interactive = false
115 }
116
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700117 // Subprocesses commands are run by fork/execing this binary
118 // so we must test to see if this instance is a subprocess or the
119 // the original command line instance.
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -0800120 if modules.IsModulesProcess() {
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800121 shutdown()
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700122 // Subprocess, run the requested command.
123 if err := modules.Dispatch(); err != nil {
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700124 fmt.Fprintf(os.Stderr, "failed: %v\n", err)
Cosmos Nicolaou4e213d72014-10-26 22:21:52 -0700125 os.Exit(1)
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700126 }
127 return
128 }
Suharsh Sivakumar19fbf992015-01-23 11:02:27 -0800129 defer shutdown()
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700130
Ryan Browna08a2212015-01-15 15:40:10 -0800131 shell, err := modules.NewShell(ctx, nil)
Cosmos Nicolaou344cc4a2014-11-26 15:38:43 -0800132 if err != nil {
133 fmt.Fprintf(os.Stderr, "unexpected error: %s\n", err)
134 os.Exit(1)
135 }
Cosmos Nicolaoubbae3882014-10-02 22:58:19 -0700136 defer shell.Cleanup(os.Stderr, os.Stderr)
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700137
Cosmos Nicolaoua1a0b162015-01-30 10:57:37 -0800138 scanner := bufio.NewScanner(input)
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700139 lineno := 1
140 prompt(lineno)
141 for scanner.Scan() {
142 line := scanner.Text()
143 if !strings.HasPrefix(line, "#") && len(line) > 0 {
Cosmos Nicolaou3574f122014-06-11 23:07:15 -0700144 if line == "eof" {
145 break
146 }
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700147 if err := process(shell, line, lineno); err != nil {
148 fmt.Printf("ERROR: %d> %q: %v\n", lineno, line, err)
Cosmos Nicolaoub44f2392014-10-28 22:41:49 -0700149 if !interactive {
150 os.Exit(1)
151 }
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700152 }
153 }
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700154 shell.SetVar("_", strconv.Itoa(lineno))
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700155 lineno++
156 prompt(lineno)
157 }
158 if err := scanner.Err(); err != nil {
159 fmt.Printf("error reading input: %v\n", err)
160 }
161
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700162}
163
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700164func output(lineno int, line string) {
165 if len(line) > 0 {
166 if !interactive {
167 fmt.Printf("%d> ", lineno)
168 }
169 line = strings.TrimSuffix(line, "\n")
170 fmt.Printf("%s\n", line)
171 }
172}
173
174func process(sh *modules.Shell, line string, lineno int) error {
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700175 fields, err := splitQuotedFields(line)
176 if err != nil {
177 return err
178 }
179 if len(fields) == 0 {
180 return fmt.Errorf("no input")
181 }
182 name := fields[0]
183
184 var args []string
185 if len(fields) > 1 {
186 args = fields[1:]
187 } else {
188 args = []string{}
189 }
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700190 sub, err := subVariables(sh, args)
Cosmos Nicolaou4e213d72014-10-26 22:21:52 -0700191 if err != nil {
192 return err
193 }
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700194 if cmd := builtins[name]; cmd != nil {
195 if cmd.nargs >= 0 && len(sub) != cmd.nargs {
196 return fmt.Errorf("wrong (%d) # args for %q: usage %s", len(sub), name, cmd.usage)
197 }
198 l := ""
199 var err error
200 if cmd.needsHandle {
201 l, err = handleWrapper(sh, cmd.fn, sub...)
202 } else {
203 l, err = cmd.fn(sh, nil, sub...)
204 }
205 if err != nil {
Cosmos Nicolaou4e213d72014-10-26 22:21:52 -0700206 return fmt.Errorf("%s : %s", err, l)
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700207 }
208 output(lineno, l)
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700209 } else {
Cosmos Nicolaou612ad382014-10-29 19:41:35 -0700210 handle, err := sh.Start(name, nil, sub...)
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700211 if err != nil {
212 return err
213 }
214 handles[strconv.Itoa(lineno)] = &cmdState{
215 handle,
216 expect.NewSession(nil, handle.Stdout(), time.Minute),
217 line,
218 }
Cosmos Nicolaoub44f2392014-10-28 22:41:49 -0700219 output(lineno, line)
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700220 }
221 return nil
222}
223
224// splitQuotedFields a line into fields, allowing for quoted strings.
225func splitQuotedFields(line string) ([]string, error) {
226 fields := []string{}
227 inquote := false
228 var field []rune
229 for _, c := range line {
230 switch {
231 case c == '"':
232 if inquote {
233 fields = append(fields, string(field))
234 field = nil
235 inquote = false
236 } else {
237 inquote = true
238 }
239 case unicode.IsSpace(c):
240 if inquote {
241 field = append(field, c)
242 } else {
243 if len(field) > 0 {
244 fields = append(fields, string(field))
245 }
246 field = nil
247 }
248 default:
249 field = append(field, c)
250 }
251 }
252 if inquote {
253 return nil, fmt.Errorf("unterminated quoted input")
254 }
255
256 if len(field) > 0 {
257 fields = append(fields, string(field))
258 }
259 return fields, nil
260}
261
262// subVariables substitutes variables that occur in the string slice
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700263// args with values from the Shell.
264func subVariables(sh *modules.Shell, args []string) ([]string, error) {
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700265 var results []string
266 for _, a := range args {
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700267 if r, err := subVariablesInArgument(sh, a); err != nil {
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700268 return results, err
269 } else {
270 results = append(results, r)
271 }
272 }
273 return results, nil
274}
275
276// subVariablesInArgument substitutes variables that occur in the string
277// parameter with values from vars.
278//
279// A variable, is introduced by $, terminated by \t, space, / , : or !.
280// Variables may also be enclosed by {} (as in ${VAR}) to allow for embedding
281// within strings.
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700282func subVariablesInArgument(sh *modules.Shell, a string) (string, error) {
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700283 first := strings.Index(a, "$")
284 if first < 0 {
285 return a, nil
286 }
287 parts := strings.Split(a, "$")
288 result := parts[0]
289 vn := ""
290 rem := 0
291 for _, p := range parts[1:] {
292 start := 0
293 end := -1
294 if strings.HasPrefix(p, "{") {
295 start = 1
296 end = strings.Index(p, "}")
297 if end < 0 {
298 return "", fmt.Errorf("unterminated variable: %q", p)
299 }
300 rem = end + 1
301 } else {
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700302 end = strings.IndexAny(p, "\t/,:!= ")
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700303 if end < 0 {
304 end = len(p)
305 }
306 rem = end
307 }
308 vn = p[start:end]
309 r := p[rem:]
Cosmos Nicolaou9c9918d2014-09-23 08:45:56 -0700310 v, present := sh.GetVar(vn)
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700311 if !present {
Matt Rosencrantz0dd7ce02014-09-23 11:34:26 -0700312 return a, fmt.Errorf("unknown variable: %q", vn)
Cosmos Nicolaou7c659ac2014-06-09 22:47:04 -0700313 }
314 result += v
315 result += r
316 }
317 return result, nil
318}