blob: c0309c0926712e969626fca3d16a1fb40b7643d5 [file] [log] [blame]
Jiri Simsad7616c92015-03-24 23:44:30 -07001// Copyright 2015 The Vanadium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -08005package modules
6
7import (
8 "flag"
9 "fmt"
10 "io"
Todd Wang5507c832015-05-15 22:59:23 -070011 "io/ioutil"
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080012 "os"
Todd Wang95873902015-05-22 14:21:30 -070013 "path/filepath"
14 "runtime"
15 "strconv"
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080016 "time"
17
Cosmos Nicolaoue3b19322015-06-18 16:05:08 -070018 "v.io/x/ref/internal/logger"
Jiri Simsaffceefa2015-02-28 11:03:34 -080019 vexec "v.io/x/ref/lib/exec"
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080020)
21
Todd Wang95873902015-05-22 14:21:30 -070022// Program is a symbolic representation of a registered Main function.
23type Program string
24
25type programInfo struct {
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080026 main Main
Todd Wang95873902015-05-22 14:21:30 -070027 factory func() *execHandle
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080028}
29
Todd Wang95873902015-05-22 14:21:30 -070030type programRegistry struct {
31 programs []*programInfo
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080032}
33
Todd Wang95873902015-05-22 14:21:30 -070034var registry = new(programRegistry)
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080035
Todd Wang95873902015-05-22 14:21:30 -070036func (r *programRegistry) addProgram(main Main, description string) Program {
37 prog := strconv.Itoa(len(r.programs))
38 factory := func() *execHandle { return newExecHandle(prog, description) }
39 r.programs = append(r.programs, &programInfo{main, factory})
40 return Program(prog)
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080041}
42
Todd Wang95873902015-05-22 14:21:30 -070043func (r *programRegistry) getProgram(prog Program) *programInfo {
44 index, err := strconv.Atoi(string(prog))
45 if err != nil || index < 0 || index >= len(r.programs) {
46 return nil
47 }
48 return r.programs[index]
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080049}
50
Todd Wang95873902015-05-22 14:21:30 -070051func (r *programRegistry) getExternalProgram(prog Program) *programInfo {
52 h := newExecHandleExternal(string(prog))
53 return &programInfo{
Todd Wang5507c832015-05-15 22:59:23 -070054 factory: func() *execHandle { return h },
Cosmos Nicolaou42a17362015-03-10 16:40:18 -070055 }
56}
57
Todd Wang95873902015-05-22 14:21:30 -070058func (r *programRegistry) String() string {
59 var s string
60 for _, info := range r.programs {
61 h := info.factory()
62 s += fmt.Sprintf("%s: %s\n", h.entryPoint, h.desc)
63 }
64 return s
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080065}
66
Todd Wang95873902015-05-22 14:21:30 -070067// Register adds a new program to the registry that will be run as a subprocess.
68// It must be called before Dispatch is called, typically from an init function.
69func Register(main Main, description string) Program {
70 if _, file, line, ok := runtime.Caller(1); ok {
71 description = fmt.Sprintf("%s:%d %s", shortFile(file), line, description)
72 }
73 return registry.addProgram(main, description)
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080074}
75
Todd Wang95873902015-05-22 14:21:30 -070076// shortFile returns the last 3 components of the given file name.
77func shortFile(file string) string {
78 var short string
79 for i := 0; i < 3; i++ {
80 short = filepath.Join(filepath.Base(file), short)
81 file = filepath.Dir(file)
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080082 }
Todd Wang95873902015-05-22 14:21:30 -070083 return short
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080084}
85
Asim Shankar59b8b692015-03-30 01:23:36 -070086const shellEntryPoint = "V23_SHELL_HELPER_PROCESS_ENTRY_POINT"
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080087
Todd Wang95873902015-05-22 14:21:30 -070088// IsChildProcess returns true if this process was started by the modules
89// package.
90func IsChildProcess() bool {
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -080091 return os.Getenv(shellEntryPoint) != ""
92}
93
Todd Wang95873902015-05-22 14:21:30 -070094// Dispatch executes the requested subprocess program from within a subprocess.
95// Returns an error if it is executed by a process that does not specify an
96// entry point in its environment.
Cosmos Nicolaouec6ed652015-02-13 14:31:06 -080097func Dispatch() error {
Cosmos Nicolaouec6ed652015-02-13 14:31:06 -080098 return registry.dispatch()
99}
100
Todd Wang95873902015-05-22 14:21:30 -0700101// DispatchAndExitIfChild is a convenience function with three possible results:
102// * os.Exit(0) if called within a child process, and the dispatch succeeds.
103// * os.Exit(1) if called within a child process, and the dispatch fails.
104// * return with no side-effects, if not called within a child process.
105func DispatchAndExitIfChild() {
106 if IsChildProcess() {
107 if err := Dispatch(); err != nil {
108 fmt.Fprintf(os.Stderr, "modules.Dispatch failed: %v\n", err)
109 os.Exit(1)
110 }
111 os.Exit(0)
Cosmos Nicolaou90610bd2014-12-02 22:31:04 -0800112 }
Cosmos Nicolaou90610bd2014-12-02 22:31:04 -0800113}
114
Todd Wang95873902015-05-22 14:21:30 -0700115func (r *programRegistry) dispatch() error {
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -0800116 ch, err := vexec.GetChildHandle()
117 if err != nil {
118 // This is for debugging only. It's perfectly reasonable for this
119 // error to occur if the process is started by a means other
120 // than the exec library.
Cosmos Nicolaoue3b19322015-06-18 16:05:08 -0700121 logger.Global().VI(1).Infof("failed to get child handle: %s", err)
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -0800122 }
123
124 // Only signal that the child is ready or failed if we successfully get
125 // a child handle. We most likely failed to get a child handle
126 // because the subprocess was run directly from the command line.
Todd Wang95873902015-05-22 14:21:30 -0700127 prog := os.Getenv(shellEntryPoint)
128 if prog == "" {
129 err := fmt.Errorf("Failed to find entrypoint %q", prog)
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -0800130 if ch != nil {
131 ch.SetFailed(err)
132 }
133 return err
134 }
135
Todd Wang95873902015-05-22 14:21:30 -0700136 m := registry.getProgram(Program(prog))
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -0800137 if m == nil {
Todd Wang95873902015-05-22 14:21:30 -0700138 err := fmt.Errorf("%s: not registered\n%s", prog, registry.String())
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -0800139 if ch != nil {
140 ch.SetFailed(err)
141 }
142 return err
143 }
144
145 if ch != nil {
146 ch.SetReady()
147 }
148
149 go func(pid int) {
150 for {
151 _, err := os.FindProcess(pid)
152 if err != nil {
Cosmos Nicolaoue3b19322015-06-18 16:05:08 -0700153 logger.Global().Fatalf("Looks like our parent exited: %v", err)
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -0800154 }
155 time.Sleep(time.Second)
156 }
157 }(os.Getppid())
158
Suharsh Sivakumar9d17e4a2015-02-02 22:42:16 -0800159 flag.Parse()
Todd Wang95873902015-05-22 14:21:30 -0700160 return m.main(EnvFromOS(), flag.Args()...)
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -0800161}
162
Bogdan Capritac7e72b62015-01-07 19:22:23 -0800163// WaitForEOF returns when a read on its io.Reader parameter returns io.EOF
Todd Wang5507c832015-05-15 22:59:23 -0700164func WaitForEOF(r io.Reader) {
165 io.Copy(ioutil.Discard, r)
Cosmos Nicolaoud4f00562014-11-17 20:35:48 -0800166}