Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 1 | // 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 Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 5 | // extendable by adding new 'commands' that implement the API defined |
| 6 | // by veyron/lib/modules. |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 7 | package main |
| 8 | |
| 9 | import ( |
| 10 | "bufio" |
| 11 | "flag" |
| 12 | "fmt" |
| 13 | "os" |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 14 | "strconv" |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 15 | "strings" |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 16 | "time" |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 17 | "unicode" |
| 18 | |
Jiri Simsa | 764efb7 | 2014-12-25 20:57:03 -0800 | [diff] [blame] | 19 | "v.io/core/veyron2" |
Matt Rosencrantz | d599e38 | 2015-01-12 11:13:32 -0800 | [diff] [blame] | 20 | "v.io/core/veyron2/context" |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 21 | |
Jiri Simsa | 764efb7 | 2014-12-25 20:57:03 -0800 | [diff] [blame] | 22 | "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 Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 26 | ) |
| 27 | |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 28 | type cmdState struct { |
| 29 | modules.Handle |
| 30 | *expect.Session |
| 31 | line string |
| 32 | } |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 33 | |
| 34 | var ( |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 35 | interactive bool |
Cosmos Nicolaou | a1a0b16 | 2015-01-30 10:57:37 -0800 | [diff] [blame] | 36 | filename string |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 37 | handles map[string]*cmdState |
Cosmos Nicolaou | b44f239 | 2014-10-28 22:41:49 -0700 | [diff] [blame] | 38 | jsonDict map[string]string |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 39 | ) |
| 40 | |
| 41 | func init() { |
| 42 | flag.BoolVar(&interactive, "interactive", true, "set interactive/batch mode") |
Cosmos Nicolaou | a1a0b16 | 2015-01-30 10:57:37 -0800 | [diff] [blame] | 43 | flag.StringVar(&filename, "file", "", "command file") |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 44 | handles = make(map[string]*cmdState) |
Cosmos Nicolaou | b44f239 | 2014-10-28 22:41:49 -0700 | [diff] [blame] | 45 | jsonDict = make(map[string]string) |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 46 | flag.Usage = usage |
| 47 | } |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 48 | |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 49 | var usage = func() { |
| 50 | fmt.Println( |
| 51 | `Welcome to this simple shell that lets you run mount tables, a simple server |
| 52 | and 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 |
| 54 | get specific help about that command. The shell provides environment variables |
| 55 | with expansion and intrinsic support for managing subprocess, but it does not |
| 56 | provide any flow control commands. |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 57 | |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 58 | All commands, except builtin ones (such as help, set, eval etc) are run |
| 59 | asynchronously in background. That is, the prompt returns as soon as they are |
| 60 | started and no output is displayed from them unless an error is encountered |
| 61 | when they are being started. Each input line is numbered and that number is |
| 62 | used to refer to the standard output of previous started commands. The variable |
| 63 | _ always contains the number of the immediately preceeding line. It is |
| 64 | possible to read the output of a command (using the 'read' builtin) and assign |
| 65 | it that output to an environment variable. The 'eval' builtin parses output of |
| 66 | the form <var>=<val>. In this way subproccess may be started, their output |
| 67 | read and used to configure subsequent subprocesses. For example: |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 68 | |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 69 | 1> time |
| 70 | 2> read 1 t |
| 71 | 3> print $t |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 72 | |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 73 | will print the first line of output from the time command, as will the |
| 74 | following: |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 75 | |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 76 | or: |
| 77 | time |
| 78 | read $_ t |
| 79 | print $t |
| 80 | |
| 81 | The eval builtin is used to directly to assign to variables specified |
| 82 | in the output of the command. For example, if the root command |
| 83 | prints out MT_NAME=foo then eval will set MT_NAME to foo as follows: |
| 84 | |
| 85 | root |
| 86 | eval $_ |
| 87 | print $MT_NAME |
| 88 | |
| 89 | will print the value of MT_NAME that is output by the root command. |
| 90 | `) |
| 91 | flag.PrintDefaults() |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 92 | } |
| 93 | |
| 94 | func prompt(lineno int) { |
| 95 | if interactive { |
| 96 | fmt.Printf("%d> ", lineno) |
| 97 | } |
| 98 | } |
| 99 | |
Matt Rosencrantz | d599e38 | 2015-01-12 11:13:32 -0800 | [diff] [blame] | 100 | var ctx *context.T |
Matt Rosencrantz | c13446b | 2014-12-03 10:37:00 -0800 | [diff] [blame] | 101 | |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 102 | func main() { |
Suharsh Sivakumar | 19fbf99 | 2015-01-23 11:02:27 -0800 | [diff] [blame] | 103 | var shutdown veyron2.Shutdown |
| 104 | ctx, shutdown = veyron2.Init() |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 105 | |
Cosmos Nicolaou | a1a0b16 | 2015-01-30 10:57:37 -0800 | [diff] [blame] | 106 | 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 Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 117 | // 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 Nicolaou | d4f0056 | 2014-11-17 20:35:48 -0800 | [diff] [blame] | 120 | if modules.IsModulesProcess() { |
Suharsh Sivakumar | 19fbf99 | 2015-01-23 11:02:27 -0800 | [diff] [blame] | 121 | shutdown() |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 122 | // Subprocess, run the requested command. |
| 123 | if err := modules.Dispatch(); err != nil { |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 124 | fmt.Fprintf(os.Stderr, "failed: %v\n", err) |
Cosmos Nicolaou | 4e213d7 | 2014-10-26 22:21:52 -0700 | [diff] [blame] | 125 | os.Exit(1) |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 126 | } |
| 127 | return |
| 128 | } |
Suharsh Sivakumar | 19fbf99 | 2015-01-23 11:02:27 -0800 | [diff] [blame] | 129 | defer shutdown() |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 130 | |
Ryan Brown | a08a221 | 2015-01-15 15:40:10 -0800 | [diff] [blame] | 131 | shell, err := modules.NewShell(ctx, nil) |
Cosmos Nicolaou | 344cc4a | 2014-11-26 15:38:43 -0800 | [diff] [blame] | 132 | if err != nil { |
| 133 | fmt.Fprintf(os.Stderr, "unexpected error: %s\n", err) |
| 134 | os.Exit(1) |
| 135 | } |
Cosmos Nicolaou | bbae388 | 2014-10-02 22:58:19 -0700 | [diff] [blame] | 136 | defer shell.Cleanup(os.Stderr, os.Stderr) |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 137 | |
Cosmos Nicolaou | a1a0b16 | 2015-01-30 10:57:37 -0800 | [diff] [blame] | 138 | scanner := bufio.NewScanner(input) |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 139 | lineno := 1 |
| 140 | prompt(lineno) |
| 141 | for scanner.Scan() { |
| 142 | line := scanner.Text() |
| 143 | if !strings.HasPrefix(line, "#") && len(line) > 0 { |
Cosmos Nicolaou | 3574f12 | 2014-06-11 23:07:15 -0700 | [diff] [blame] | 144 | if line == "eof" { |
| 145 | break |
| 146 | } |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 147 | if err := process(shell, line, lineno); err != nil { |
| 148 | fmt.Printf("ERROR: %d> %q: %v\n", lineno, line, err) |
Cosmos Nicolaou | b44f239 | 2014-10-28 22:41:49 -0700 | [diff] [blame] | 149 | if !interactive { |
| 150 | os.Exit(1) |
| 151 | } |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 152 | } |
| 153 | } |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 154 | shell.SetVar("_", strconv.Itoa(lineno)) |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 155 | lineno++ |
| 156 | prompt(lineno) |
| 157 | } |
| 158 | if err := scanner.Err(); err != nil { |
| 159 | fmt.Printf("error reading input: %v\n", err) |
| 160 | } |
| 161 | |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 162 | } |
| 163 | |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 164 | func 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 | |
| 174 | func process(sh *modules.Shell, line string, lineno int) error { |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 175 | 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 Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 190 | sub, err := subVariables(sh, args) |
Cosmos Nicolaou | 4e213d7 | 2014-10-26 22:21:52 -0700 | [diff] [blame] | 191 | if err != nil { |
| 192 | return err |
| 193 | } |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 194 | 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 Nicolaou | 4e213d7 | 2014-10-26 22:21:52 -0700 | [diff] [blame] | 206 | return fmt.Errorf("%s : %s", err, l) |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 207 | } |
| 208 | output(lineno, l) |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 209 | } else { |
Cosmos Nicolaou | 612ad38 | 2014-10-29 19:41:35 -0700 | [diff] [blame] | 210 | handle, err := sh.Start(name, nil, sub...) |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 211 | 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 Nicolaou | b44f239 | 2014-10-28 22:41:49 -0700 | [diff] [blame] | 219 | output(lineno, line) |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 220 | } |
| 221 | return nil |
| 222 | } |
| 223 | |
| 224 | // splitQuotedFields a line into fields, allowing for quoted strings. |
| 225 | func 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 Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 263 | // args with values from the Shell. |
| 264 | func subVariables(sh *modules.Shell, args []string) ([]string, error) { |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 265 | var results []string |
| 266 | for _, a := range args { |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 267 | if r, err := subVariablesInArgument(sh, a); err != nil { |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 268 | 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 Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 282 | func subVariablesInArgument(sh *modules.Shell, a string) (string, error) { |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 283 | 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 Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 302 | end = strings.IndexAny(p, "\t/,:!= ") |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 303 | if end < 0 { |
| 304 | end = len(p) |
| 305 | } |
| 306 | rem = end |
| 307 | } |
| 308 | vn = p[start:end] |
| 309 | r := p[rem:] |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 310 | v, present := sh.GetVar(vn) |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 311 | if !present { |
Matt Rosencrantz | 0dd7ce0 | 2014-09-23 11:34:26 -0700 | [diff] [blame] | 312 | return a, fmt.Errorf("unknown variable: %q", vn) |
Cosmos Nicolaou | 7c659ac | 2014-06-09 22:47:04 -0700 | [diff] [blame] | 313 | } |
| 314 | result += v |
| 315 | result += r |
| 316 | } |
| 317 | return result, nil |
| 318 | } |