| 
									
										
										
										
											2019-07-04 01:36:35 -06:00
										 |  |  | package manager | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io/ioutil" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"path/filepath" | 
					
						
							|  |  |  | 	"text/template" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-04 01:36:35 -06:00
										 |  |  | 	"git.rootprojects.org/root/go-serviceman/manager/static" | 
					
						
							| 
									
										
										
										
											2019-07-02 23:51:30 -06:00
										 |  |  | 	"git.rootprojects.org/root/go-serviceman/service" | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-05 12:48:58 -06:00
										 |  |  | const ( | 
					
						
							|  |  |  | 	srvExt      = ".plist" | 
					
						
							|  |  |  | 	srvSysPath  = "/Library/LaunchDaemons" | 
					
						
							|  |  |  | 	srvUserPath = "Library/LaunchAgents" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var srvLen int | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func init() { | 
					
						
							|  |  |  | 	srvLen = len(srvExt) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-10 01:16:45 -06:00
										 |  |  | func start(conf *service.Service) error { | 
					
						
							|  |  |  | 	system := conf.System | 
					
						
							|  |  |  | 	home := conf.Home | 
					
						
							|  |  |  | 	rdns := conf.ReverseDNS | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	service, err := getService(system, home, rdns) | 
					
						
							| 
									
										
										
										
											2019-07-05 12:48:58 -06:00
										 |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cmds := []Runnable{ | 
					
						
							|  |  |  | 		Runnable{ | 
					
						
							|  |  |  | 			Exec: "launchctl", | 
					
						
							|  |  |  | 			Args: []string{"unload", "-w", service}, | 
					
						
							|  |  |  | 			Must: false, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		Runnable{ | 
					
						
							| 
									
										
										
										
											2019-07-07 02:14:25 -06:00
										 |  |  | 			Exec:     "launchctl", | 
					
						
							|  |  |  | 			Args:     []string{"load", "-w", service}, | 
					
						
							|  |  |  | 			Must:     true, | 
					
						
							|  |  |  | 			Badwords: []string{"No such file or directory", "service already loaded"}, | 
					
						
							| 
									
										
										
										
											2019-07-05 12:48:58 -06:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 	cmds = adjustPrivs(system, cmds) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-10 01:16:45 -06:00
										 |  |  | 	typ := "USER" | 
					
						
							|  |  |  | 	if system { | 
					
						
							|  |  |  | 		typ = "SYSTEM" | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	fmt.Printf("Starting launchd %s service...\n\n", typ) | 
					
						
							| 
									
										
										
										
											2019-07-10 01:16:45 -06:00
										 |  |  | 	for i := range cmds { | 
					
						
							|  |  |  | 		exe := cmds[i] | 
					
						
							|  |  |  | 		fmt.Println("\t" + exe.String()) | 
					
						
							|  |  |  | 		err := exe.Run() | 
					
						
							|  |  |  | 		if nil != err { | 
					
						
							|  |  |  | 			return err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fmt.Println() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func stop(conf *service.Service) error { | 
					
						
							|  |  |  | 	system := conf.System | 
					
						
							|  |  |  | 	home := conf.Home | 
					
						
							|  |  |  | 	rdns := conf.ReverseDNS | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	service, err := getService(system, home, rdns) | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cmds := []Runnable{ | 
					
						
							|  |  |  | 		Runnable{ | 
					
						
							|  |  |  | 			Exec:     "launchctl", | 
					
						
							|  |  |  | 			Args:     []string{"unload", service}, | 
					
						
							|  |  |  | 			Must:     false, | 
					
						
							|  |  |  | 			Badwords: []string{"No such file or directory", "Cound not find specified service"}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cmds = adjustPrivs(system, cmds) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fmt.Println() | 
					
						
							|  |  |  | 	typ := "USER" | 
					
						
							|  |  |  | 	if system { | 
					
						
							|  |  |  | 		typ = "SYSTEM" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fmt.Printf("Stopping launchd %s service...\n", typ) | 
					
						
							| 
									
										
										
										
											2019-07-05 12:48:58 -06:00
										 |  |  | 	for i := range cmds { | 
					
						
							|  |  |  | 		exe := cmds[i] | 
					
						
							| 
									
										
										
										
											2019-07-07 06:55:01 +00:00
										 |  |  | 		fmt.Println("\t" + exe.String()) | 
					
						
							| 
									
										
										
										
											2019-07-05 12:48:58 -06:00
										 |  |  | 		err := exe.Run() | 
					
						
							|  |  |  | 		if nil != err { | 
					
						
							|  |  |  | 			return err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fmt.Println() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | func Render(c *service.Service) ([]byte, error) { | 
					
						
							|  |  |  | 	// Create service file from template | 
					
						
							|  |  |  | 	b, err := static.ReadFile("dist/Library/LaunchDaemons/_rdns_.plist.tmpl") | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	s := string(b) | 
					
						
							|  |  |  | 	rw := &bytes.Buffer{} | 
					
						
							|  |  |  | 	// not sure what the template name does, but whatever | 
					
						
							|  |  |  | 	tmpl, err := template.New("service").Parse(s) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	err = tmpl.Execute(rw, c) | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return rw.Bytes(), nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func install(c *service.Service) (string, error) { | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	// Darwin-specific config options | 
					
						
							|  |  |  | 	if c.PrivilegedPorts { | 
					
						
							|  |  |  | 		if !c.System { | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 			return "", fmt.Errorf("You must use root-owned LaunchDaemons (not user-owned LaunchAgents) to use priveleged ports on OS X") | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-05 12:48:58 -06:00
										 |  |  | 	plistDir := srvSysPath | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	if !c.System { | 
					
						
							| 
									
										
										
										
											2019-07-05 12:48:58 -06:00
										 |  |  | 		plistDir = filepath.Join(c.Home, srvUserPath) | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Check paths first | 
					
						
							| 
									
										
										
										
											2019-07-02 00:25:16 -06:00
										 |  |  | 	err := os.MkdirAll(filepath.Dir(plistDir), 0755) | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	if nil != err { | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 		return "", err | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	b, err := Render(c) | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	if nil != err { | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 		return "", err | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Write the file out | 
					
						
							|  |  |  | 	// TODO rdns | 
					
						
							| 
									
										
										
										
											2019-07-02 00:02:09 -06:00
										 |  |  | 	plistName := c.ReverseDNS + ".plist" | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	plistPath := filepath.Join(plistDir, plistName) | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	if err := ioutil.WriteFile(plistPath, b, 0644); err != nil { | 
					
						
							|  |  |  | 		return "", fmt.Errorf("Error writing %s: %v", plistPath, err) | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-05 12:48:58 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// TODO --no-start | 
					
						
							| 
									
										
										
										
											2019-07-10 01:16:45 -06:00
										 |  |  | 	err = start(c) | 
					
						
							| 
									
										
										
										
											2019-07-05 12:48:58 -06:00
										 |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		fmt.Printf("If things don't go well you should be able to get additional logging from launchctl:\n") | 
					
						
							|  |  |  | 		fmt.Printf("\tsudo launchctl log level debug\n") | 
					
						
							|  |  |  | 		fmt.Printf("\ttail -f /var/log/system.log\n") | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 		return "", err | 
					
						
							| 
									
										
										
										
											2019-07-02 00:25:16 -06:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	return "launchd", nil | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | } |