| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | //go:generate go run -mod=vendor git.rootprojects.org/root/go-gitver | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | // main runs the things and does the stuff | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | package main | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	"encoding/json" | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	"flag" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	"os/exec" | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	"os/user" | 
					
						
							|  |  |  | 	"path/filepath" | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	"strings" | 
					
						
							|  |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	"unicode/utf8" | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-04 01:36:35 -06:00
										 |  |  | 	"git.rootprojects.org/root/go-serviceman/manager" | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	"git.rootprojects.org/root/go-serviceman/runner" | 
					
						
							|  |  |  | 	"git.rootprojects.org/root/go-serviceman/service" | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var GitRev = "000000000" | 
					
						
							| 
									
										
										
										
											2019-07-10 02:01:12 -06:00
										 |  |  | var GitVersion = "v0.3.2-pre+dirty" | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | var GitTimestamp = time.Now().Format(time.RFC3339) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | func usage() { | 
					
						
							| 
									
										
										
										
											2019-07-10 01:16:45 -06:00
										 |  |  | 	fmt.Println("Usage:") | 
					
						
							|  |  |  | 	fmt.Println("\tserviceman <command> --help") | 
					
						
							|  |  |  | 	fmt.Println("\tserviceman add ./foo-app -- --foo-arg") | 
					
						
							|  |  |  | 	fmt.Println("\tserviceman run --config ./foo-app.json") | 
					
						
							|  |  |  | 	fmt.Println("\tserviceman start <name>") | 
					
						
							|  |  |  | 	fmt.Println("\tserviceman stop <name>") | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | func main() { | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	if len(os.Args) < 2 { | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "Too few arguments: %s\n", strings.Join(os.Args, " ")) | 
					
						
							|  |  |  | 		usage() | 
					
						
							|  |  |  | 		os.Exit(1) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	top := os.Args[1] | 
					
						
							|  |  |  | 	os.Args = append(os.Args[:1], os.Args[2:]...) | 
					
						
							|  |  |  | 	switch top { | 
					
						
							| 
									
										
										
										
											2019-07-04 01:36:35 -06:00
										 |  |  | 	case "version": | 
					
						
							|  |  |  | 		fmt.Println(GitVersion, GitTimestamp, GitRev) | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	case "run": | 
					
						
							|  |  |  | 		run() | 
					
						
							| 
									
										
										
										
											2019-07-10 01:16:45 -06:00
										 |  |  | 	case "add": | 
					
						
							|  |  |  | 		add() | 
					
						
							|  |  |  | 	case "start": | 
					
						
							|  |  |  | 		start() | 
					
						
							|  |  |  | 	case "stop": | 
					
						
							|  |  |  | 		stop() | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	default: | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "Unknown argument %s\n", top) | 
					
						
							|  |  |  | 		usage() | 
					
						
							|  |  |  | 		os.Exit(1) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-04 01:36:35 -06:00
										 |  |  | func add() { | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	conf := &service.Service{ | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 		Restart: true, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-02 00:02:09 -06:00
										 |  |  | 	force := false | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	forUser := false | 
					
						
							|  |  |  | 	forSystem := false | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	dryrun := false | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	flag.StringVar(&conf.Title, "title", "", "a human-friendly name for the service") | 
					
						
							|  |  |  | 	flag.StringVar(&conf.Desc, "desc", "", "a human-friendly description of the service (ex: Foo App)") | 
					
						
							|  |  |  | 	flag.StringVar(&conf.Name, "name", "", "a computer-friendly name for the service (ex: foo-app)") | 
					
						
							|  |  |  | 	flag.StringVar(&conf.URL, "url", "", "the documentation on home page of the service") | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	flag.StringVar(&conf.Workdir, "workdir", "", "the directory in which the service should be started (if supported)") | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	flag.StringVar(&conf.ReverseDNS, "rdns", "", "a plist-friendly Reverse DNS name for launchctl (ex: com.example.foo-app)") | 
					
						
							| 
									
										
										
										
											2019-07-04 01:36:35 -06:00
										 |  |  | 	flag.BoolVar(&forSystem, "system", false, "attempt to add system service as an unprivileged/unelevated user") | 
					
						
							|  |  |  | 	flag.BoolVar(&forUser, "user", false, "add user space / user mode service even when admin/root/sudo/elevated") | 
					
						
							| 
									
										
										
										
											2019-07-02 00:02:09 -06:00
										 |  |  | 	flag.BoolVar(&force, "force", false, "if the interpreter or executable doesn't exist, or things don't make sense, try anyway") | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	flag.StringVar(&conf.User, "username", "", "run the service as this user") | 
					
						
							|  |  |  | 	flag.StringVar(&conf.Group, "groupname", "", "run the service as this group") | 
					
						
							|  |  |  | 	flag.BoolVar(&conf.PrivilegedPorts, "cap-net-bind", false, "this service should have access to privileged ports") | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	flag.BoolVar(&dryrun, "dryrun", false, "output the service file without modifying anything on disk") | 
					
						
							|  |  |  | 	flag.Usage = func() { | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "Usage of %s:\n\n", os.Args[0]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		flag.PrintDefaults() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "Flags and arguments after \"--\" will be completely ignored by serviceman\n", os.Args[0]) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	flag.Parse() | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	flagargs := flag.Args() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// You must have something to run, duh | 
					
						
							|  |  |  | 	n := len(flagargs) | 
					
						
							|  |  |  | 	if 0 == n { | 
					
						
							|  |  |  | 		fmt.Println("Usage: serviceman add ./foo-app --foo-arg") | 
					
						
							|  |  |  | 		os.Exit(2) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if forUser && forSystem { | 
					
						
							|  |  |  | 		fmt.Println("Pfff! You can't --user AND --system! What are you trying to pull?") | 
					
						
							|  |  |  | 		os.Exit(1) | 
					
						
							| 
									
										
										
										
											2019-07-02 00:02:09 -06:00
										 |  |  | 		return | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// There are three groups of flags | 
					
						
							|  |  |  | 	// serviceman --flag1 arg1 non-flag-arg --child1 -- --raw1 -- --raw2 | 
					
						
							|  |  |  | 	//  serviceman --flag1 arg1   // these belong to serviceman | 
					
						
							|  |  |  | 	//  non-flag-arg --child1     // these will be interpretted | 
					
						
							|  |  |  | 	//  --                        // separator | 
					
						
							|  |  |  | 	//  --raw1 -- --raw2          // after the separater (including additional separators) will be ignored | 
					
						
							|  |  |  | 	rawargs := []string{} | 
					
						
							|  |  |  | 	for i := range flagargs { | 
					
						
							|  |  |  | 		if "--" == flagargs[i] { | 
					
						
							|  |  |  | 			if len(flagargs) > i+1 { | 
					
						
							|  |  |  | 				rawargs = flagargs[i+1:] | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			flagargs = flagargs[:i] | 
					
						
							|  |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Assumptions | 
					
						
							|  |  |  | 	ass := []string{} | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	if forUser { | 
					
						
							|  |  |  | 		conf.System = false | 
					
						
							|  |  |  | 	} else if forSystem { | 
					
						
							|  |  |  | 		conf.System = true | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2019-07-04 01:36:35 -06:00
										 |  |  | 		conf.System = manager.IsPrivileged() | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 		if conf.System { | 
					
						
							|  |  |  | 			ass = append(ass, "# Because you're a privileged user") | 
					
						
							|  |  |  | 			ass = append(ass, "  --system") | 
					
						
							|  |  |  | 			ass = append(ass, "") | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			ass = append(ass, "# Because you're a unprivileged user") | 
					
						
							|  |  |  | 			ass = append(ass, "  --user") | 
					
						
							|  |  |  | 			ass = append(ass, "") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if "" == conf.Workdir { | 
					
						
							|  |  |  | 		dir, _ := os.Getwd() | 
					
						
							|  |  |  | 		conf.Workdir = dir | 
					
						
							|  |  |  | 		ass = append(ass, "# Because this is your current working directory") | 
					
						
							|  |  |  | 		ass = append(ass, fmt.Sprintf("  --workdir %s", conf.Workdir)) | 
					
						
							|  |  |  | 		ass = append(ass, "") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if "" == conf.Name { | 
					
						
							|  |  |  | 		name, _ := os.Getwd() | 
					
						
							|  |  |  | 		base := filepath.Base(name) | 
					
						
							|  |  |  | 		ext := filepath.Ext(base) | 
					
						
							|  |  |  | 		n := (len(base) - len(ext)) | 
					
						
							|  |  |  | 		name = base[:n] | 
					
						
							|  |  |  | 		if "" == name { | 
					
						
							|  |  |  | 			name = base | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		conf.Name = name | 
					
						
							|  |  |  | 		ass = append(ass, "# Because this is the name of your current working directory") | 
					
						
							|  |  |  | 		ass = append(ass, fmt.Sprintf("  --name %s", conf.Name)) | 
					
						
							|  |  |  | 		ass = append(ass, "") | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	exepath, err := findExec(flagargs[0], force) | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "%s\n", err) | 
					
						
							|  |  |  | 		os.Exit(3) | 
					
						
							| 
									
										
										
										
											2019-07-02 00:02:09 -06:00
										 |  |  | 		return | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	flagargs[0] = exepath | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	exeargs, err := testScript(flagargs[0], force) | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	if nil != err { | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 		fmt.Fprintf(os.Stderr, "%s\n", err) | 
					
						
							|  |  |  | 		os.Exit(3) | 
					
						
							|  |  |  | 		return | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	flagargs = append(exeargs, flagargs...) | 
					
						
							|  |  |  | 	// TODO | 
					
						
							|  |  |  | 	for i := range flagargs { | 
					
						
							|  |  |  | 		arg := flagargs[i] | 
					
						
							|  |  |  | 		arg = filepath.ToSlash(arg) | 
					
						
							|  |  |  | 		// Paths considered to be anything starting with ./, .\, /, \, C: | 
					
						
							|  |  |  | 		if "." == arg || strings.Contains(arg, "/") { | 
					
						
							|  |  |  | 			//if "." == arg || (len(arg) >= 2 && "./" == arg[:2] || '/' == arg[0] || "C:" == strings.ToUpper(arg[:1])) { | 
					
						
							|  |  |  | 			var err error | 
					
						
							|  |  |  | 			arg, err = filepath.Abs(arg) | 
					
						
							|  |  |  | 			if nil == err { | 
					
						
							|  |  |  | 				_, err = os.Stat(arg) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if nil != err { | 
					
						
							|  |  |  | 				fmt.Printf("%q appears to be a file path, but %q could not be read\n", flagargs[i], arg) | 
					
						
							|  |  |  | 				if !force { | 
					
						
							|  |  |  | 					os.Exit(7) | 
					
						
							|  |  |  | 					return | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if '\\' != os.PathSeparator { | 
					
						
							|  |  |  | 				// Convert paths back to .\ for Windows | 
					
						
							|  |  |  | 				arg = filepath.FromSlash(arg) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// Lookin' good | 
					
						
							|  |  |  | 			flagargs[i] = arg | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	// We won't bother with Interpreter here | 
					
						
							|  |  |  | 	// (it's really just for documentation), | 
					
						
							|  |  |  | 	// but we will add any and all unchecked args to the full slice | 
					
						
							|  |  |  | 	conf.Exec = flagargs[0] | 
					
						
							|  |  |  | 	conf.Argv = append(flagargs[1:], rawargs...) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// TODO update docs: go to the work directory | 
					
						
							|  |  |  | 	// TODO test with "npm start" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	conf.NormalizeWithoutPath() | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-03 00:43:59 -06:00
										 |  |  | 	//fmt.Printf("\n%#v\n\n", conf) | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 	if conf.System && !manager.IsPrivileged() { | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "Warning: You may need to use 'sudo' to add %q as a privileged system service.\n", conf.Name) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	if len(ass) > 0 { | 
					
						
							|  |  |  | 		fmt.Println("OPTIONS: Making some assumptions...\n") | 
					
						
							|  |  |  | 		for i := range ass { | 
					
						
							|  |  |  | 			fmt.Println("\t" + ass[i]) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Find who this is running as | 
					
						
							|  |  |  | 	// And pretty print the command to run | 
					
						
							|  |  |  | 	runAs := conf.User | 
					
						
							|  |  |  | 	var wasflag bool | 
					
						
							|  |  |  | 	fmt.Printf("COMMAND: Service %q will be run like this (more or less):\n\n", conf.Title) | 
					
						
							|  |  |  | 	if conf.System { | 
					
						
							|  |  |  | 		if "" == runAs { | 
					
						
							|  |  |  | 			runAs = "root" | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		fmt.Printf("\t# Starts on system boot, as %q\n", runAs) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		u, _ := user.Current() | 
					
						
							|  |  |  | 		runAs = u.Name | 
					
						
							|  |  |  | 		if "" == runAs { | 
					
						
							|  |  |  | 			runAs = u.Username | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		fmt.Printf("\t# Starts as %q, when %q logs in\n", runAs, u.Username) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	//fmt.Printf("\tpushd %s\n", conf.Workdir) | 
					
						
							|  |  |  | 	fmt.Printf("\t%s\n", conf.Exec) | 
					
						
							|  |  |  | 	for i := range conf.Argv { | 
					
						
							|  |  |  | 		arg := conf.Argv[i] | 
					
						
							|  |  |  | 		if '-' == arg[0] { | 
					
						
							|  |  |  | 			if wasflag { | 
					
						
							|  |  |  | 				fmt.Println() | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			wasflag = true | 
					
						
							|  |  |  | 			fmt.Printf("\t\t%s", arg) | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			if wasflag { | 
					
						
							|  |  |  | 				fmt.Printf(" %s\n", arg) | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				fmt.Printf("\t\t%s\n", arg) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			wasflag = false | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if wasflag { | 
					
						
							|  |  |  | 		fmt.Println() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fmt.Println() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// TODO output config without installing | 
					
						
							|  |  |  | 	if dryrun { | 
					
						
							|  |  |  | 		b, err := manager.Render(conf) | 
					
						
							|  |  |  | 		if nil != err { | 
					
						
							|  |  |  | 			fmt.Fprintf(os.Stderr, "Error rendering: %s\n", err) | 
					
						
							|  |  |  | 			os.Exit(10) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		fmt.Println(string(b)) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fmt.Printf("LAUNCHER: ") | 
					
						
							|  |  |  | 	servicetype, err := manager.Install(conf) | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 		fmt.Fprintf(os.Stderr, "%s\n", err) | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 		os.Exit(500) | 
					
						
							|  |  |  | 		return | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-04 03:02:27 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	fmt.Printf("LOGS: ") | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 	printLogMessage(conf) | 
					
						
							| 
									
										
										
										
											2019-07-05 12:48:58 -06:00
										 |  |  | 	fmt.Println() | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	servicemode := "USER MODE" | 
					
						
							|  |  |  | 	if conf.System { | 
					
						
							|  |  |  | 		servicemode = "SYSTEM" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fmt.Printf( | 
					
						
							|  |  |  | 		"SUCCESS:\n\n\t%q started as a %q %s service, running as %q\n", | 
					
						
							|  |  |  | 		conf.Name, | 
					
						
							|  |  |  | 		servicetype, | 
					
						
							|  |  |  | 		servicemode, | 
					
						
							|  |  |  | 		runAs, | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 	fmt.Println() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func findExec(exe string, force bool) (string, error) { | 
					
						
							|  |  |  | 	// ex: node => /usr/local/bin/node | 
					
						
							|  |  |  | 	// ex: ./demo.js => /Users/aj/project/demo.js | 
					
						
							|  |  |  | 	exepath, err := exec.LookPath(exe) | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		var msg string | 
					
						
							|  |  |  | 		if strings.Contains(filepath.ToSlash(exe), "/") { | 
					
						
							|  |  |  | 			if _, err := os.Stat(exe); err != nil { | 
					
						
							|  |  |  | 				msg = fmt.Sprintf("Error: '%s' could not be found in PATH or working directory.\n", exe) | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				msg = fmt.Sprintf("Error: '%s' is not an executable.\nYou may be able to fix that. Try running this:\n\tchmod a+x %s\n", exe, exe) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			if _, err := os.Stat(exe); err != nil { | 
					
						
							|  |  |  | 				msg = fmt.Sprintf("Error: '%s' could not be found in PATH", exe) | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				msg = fmt.Sprintf("Error: '%s' could not be found in PATH, did you mean './%s'?\n", exe, exe) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if !force { | 
					
						
							|  |  |  | 			return "", fmt.Errorf(msg) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "%s\n", msg) | 
					
						
							|  |  |  | 		return exe, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// ex: \Users\aj\project\demo.js => /Users/aj/project/demo.js | 
					
						
							|  |  |  | 	// Can't have an error here when lookpath succeeded | 
					
						
							|  |  |  | 	exepath, _ = filepath.Abs(filepath.ToSlash(exepath)) | 
					
						
							|  |  |  | 	return exepath, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func testScript(exepath string, force bool) ([]string, error) { | 
					
						
							|  |  |  | 	f, err := os.Open(exepath) | 
					
						
							|  |  |  | 	b := make([]byte, 256) | 
					
						
							|  |  |  | 	if nil == err { | 
					
						
							|  |  |  | 		_, err = f.Read(b) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if nil != err || len(b) < len("#!/x") { | 
					
						
							|  |  |  | 		msg := fmt.Sprintf("Error when testing if '%s' is a binary or script: could not read file: %s\n", exepath, err) | 
					
						
							|  |  |  | 		if !force { | 
					
						
							|  |  |  | 			return nil, fmt.Errorf(msg) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "%s\n", msg) | 
					
						
							|  |  |  | 		return nil, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Nott sure if this is more readable and idiomatic as if else or switch | 
					
						
							|  |  |  | 	// However, the order matters | 
					
						
							|  |  |  | 	switch { | 
					
						
							|  |  |  | 	case utf8.Valid(b): | 
					
						
							|  |  |  | 		// Looks like an executable script | 
					
						
							|  |  |  | 		if "#!/" == string(b[:3]) { | 
					
						
							|  |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		msg := fmt.Sprintf("Error: %q looks like a script, but we don't know the interpreter.\nYou can probably fix this by...\n"+ | 
					
						
							|  |  |  | 			"\tExplicitly naming the interpreter (ex: 'python my-script.py' instead of just 'my-script.py')\n"+ | 
					
						
							|  |  |  | 			"\tPlacing a hashbang at the top of the script (ex: '#!/usr/bin/env python')", exepath) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if !force { | 
					
						
							|  |  |  | 			return nil, fmt.Errorf(msg) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return nil, nil | 
					
						
							|  |  |  | 	case "#!/" != string(b[:3]): | 
					
						
							|  |  |  | 		// Looks like a normal binary | 
					
						
							|  |  |  | 		return nil, nil | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		// Looks like a corrupt script file | 
					
						
							|  |  |  | 		msg := "Error: It looks like you've specified a corrupt script file." | 
					
						
							|  |  |  | 		if !force { | 
					
						
							|  |  |  | 			return nil, fmt.Errorf(msg) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return nil, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Deal with #!/whatever | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Get that first line | 
					
						
							|  |  |  | 	// "#!/usr/bin/env node" => ["/usr/bin/env", "node"] | 
					
						
							|  |  |  | 	// "#!/usr/bin/node --harmony => ["/usr/bin/node", "--harmony"] | 
					
						
							|  |  |  | 	s := string(b[2:]) // strip leading #! | 
					
						
							|  |  |  | 	s = strings.Split(strings.Replace(s, "\r\n", "\n", -1), "\n")[0] | 
					
						
							|  |  |  | 	allargs := strings.Split(strings.TrimSpace(s), " ") | 
					
						
							|  |  |  | 	args := []string{} | 
					
						
							|  |  |  | 	for i := range allargs { | 
					
						
							|  |  |  | 		arg := strings.TrimSpace(allargs[i]) | 
					
						
							|  |  |  | 		if "" != arg { | 
					
						
							|  |  |  | 			args = append(args, arg) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if strings.HasSuffix(args[0], "/env") && len(args) > 1 { | 
					
						
							|  |  |  | 		// TODO warn that "env" is probably not an executable if 1 = len(args)? | 
					
						
							|  |  |  | 		args = args[1:] | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	exepath, err = findExec(args[0], force) | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	args[0] = exepath | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return args, nil | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-10 01:16:45 -06:00
										 |  |  | func start() { | 
					
						
							|  |  |  | 	forUser := false | 
					
						
							|  |  |  | 	forSystem := false | 
					
						
							|  |  |  | 	flag.BoolVar(&forSystem, "system", false, "attempt to add system service as an unprivileged/unelevated user") | 
					
						
							|  |  |  | 	flag.BoolVar(&forUser, "user", false, "add user space / user mode service even when admin/root/sudo/elevated") | 
					
						
							|  |  |  | 	flag.Parse() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	args := flag.Args() | 
					
						
							|  |  |  | 	if 1 != len(args) { | 
					
						
							|  |  |  | 		fmt.Println("Usage: serviceman start <name>") | 
					
						
							|  |  |  | 		os.Exit(1) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if forUser && forSystem { | 
					
						
							|  |  |  | 		fmt.Println("Pfff! You can't --user AND --system! What are you trying to pull?") | 
					
						
							|  |  |  | 		os.Exit(1) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	conf := &service.Service{ | 
					
						
							|  |  |  | 		Name:    args[0], | 
					
						
							|  |  |  | 		Restart: false, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if forUser { | 
					
						
							|  |  |  | 		conf.System = false | 
					
						
							|  |  |  | 	} else if forSystem { | 
					
						
							|  |  |  | 		conf.System = true | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		conf.System = manager.IsPrivileged() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	conf.NormalizeWithoutPath() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err := manager.Start(conf) | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "%s\n", err) | 
					
						
							|  |  |  | 		os.Exit(500) | 
					
						
							|  |  |  | 		return | 
					
						
							| 
									
										
										
										
											2019-07-10 01:16:45 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func stop() { | 
					
						
							|  |  |  | 	forUser := false | 
					
						
							|  |  |  | 	forSystem := false | 
					
						
							|  |  |  | 	flag.BoolVar(&forSystem, "system", false, "attempt to add system service as an unprivileged/unelevated user") | 
					
						
							|  |  |  | 	flag.BoolVar(&forUser, "user", false, "add user space / user mode service even when admin/root/sudo/elevated") | 
					
						
							|  |  |  | 	flag.Parse() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	args := flag.Args() | 
					
						
							|  |  |  | 	if 1 != len(args) { | 
					
						
							|  |  |  | 		fmt.Println("Usage: serviceman stop <name>") | 
					
						
							|  |  |  | 		os.Exit(1) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if forUser && forSystem { | 
					
						
							|  |  |  | 		fmt.Println("Pfff! You can't --user AND --system! What are you trying to pull?") | 
					
						
							|  |  |  | 		os.Exit(1) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	conf := &service.Service{ | 
					
						
							|  |  |  | 		Name:    args[0], | 
					
						
							|  |  |  | 		Restart: false, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if forUser { | 
					
						
							|  |  |  | 		conf.System = false | 
					
						
							|  |  |  | 	} else if forSystem { | 
					
						
							|  |  |  | 		conf.System = true | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		conf.System = manager.IsPrivileged() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	conf.NormalizeWithoutPath() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err := manager.Stop(conf); nil != err { | 
					
						
							|  |  |  | 		fmt.Println(err) | 
					
						
							|  |  |  | 		os.Exit(127) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | func run() { | 
					
						
							|  |  |  | 	var confpath string | 
					
						
							|  |  |  | 	var daemonize bool | 
					
						
							|  |  |  | 	flag.StringVar(&confpath, "config", "", "path to a config file to run") | 
					
						
							|  |  |  | 	flag.BoolVar(&daemonize, "daemon", false, "spawn a child process that lives in the background, and exit") | 
					
						
							|  |  |  | 	flag.Parse() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if "" == confpath { | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 		fmt.Fprintf(os.Stderr, "%s\n", strings.Join(flag.Args(), " ")) | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 		fmt.Fprintf(os.Stderr, "--config /path/to/config.json is required\n") | 
					
						
							|  |  |  | 		usage() | 
					
						
							|  |  |  | 		os.Exit(1) | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	b, err := ioutil.ReadFile(confpath) | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "Couldn't read config file: %s\n", err) | 
					
						
							|  |  |  | 		os.Exit(400) | 
					
						
							| 
									
										
										
										
											2019-07-02 00:02:09 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	s := &service.Service{} | 
					
						
							|  |  |  | 	err = json.Unmarshal(b, s) | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "Couldn't JSON parse config file: %s\n", err) | 
					
						
							|  |  |  | 		os.Exit(400) | 
					
						
							| 
									
										
										
										
											2019-07-02 00:02:09 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	m := map[string]interface{}{} | 
					
						
							|  |  |  | 	err = json.Unmarshal(b, &m) | 
					
						
							| 
									
										
										
										
											2019-07-02 00:02:09 -06:00
										 |  |  | 	if nil != err { | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 		fmt.Fprintf(os.Stderr, "Couldn't JSON parse config file: %s\n", err) | 
					
						
							|  |  |  | 		os.Exit(400) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-02 00:02:09 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	// default Restart to true | 
					
						
							|  |  |  | 	if _, ok := m["restart"]; !ok { | 
					
						
							|  |  |  | 		s.Restart = true | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	if "" == s.Exec { | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "Missing exec\n") | 
					
						
							|  |  |  | 		os.Exit(400) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-10 01:16:45 -06:00
										 |  |  | 	force := false | 
					
						
							|  |  |  | 	s.Normalize(force) | 
					
						
							| 
									
										
										
										
											2019-07-04 03:02:27 -06:00
										 |  |  | 	fmt.Printf("All output will be directed to the logs at:\n\t%s\n", s.Logdir) | 
					
						
							| 
									
										
										
										
											2019-07-03 00:43:59 -06:00
										 |  |  | 	err = os.MkdirAll(s.Logdir, 0755) | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "%s\n", err) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	if !daemonize { | 
					
						
							| 
									
										
										
										
											2019-07-03 02:11:50 -06:00
										 |  |  | 		//fmt.Fprintf(os.Stdout, "Running %s %s %s\n", s.Interpreter, s.Exec, strings.Join(s.Argv, " ")) | 
					
						
							| 
									
										
										
										
											2019-07-10 01:16:45 -06:00
										 |  |  | 		if err := runner.Start(s); nil != err { | 
					
						
							|  |  |  | 			fmt.Println("Error:", err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	manager.Run(os.Args[0], "run", "--config", confpath) | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | } |