| 
									
										
										
										
											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
										 |  |  | var ( | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 	srvLen     int | 
					
						
							|  |  |  | 	srvExt     = ".service" | 
					
						
							|  |  |  | 	srvSysPath = "/etc/systemd/system" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Not sure which of these it's supposed to be... | 
					
						
							|  |  |  | 	// * ~/.local/share/systemd/user/watchdog.service | 
					
						
							|  |  |  | 	// * ~/.config/systemd/user/watchdog.service | 
					
						
							|  |  |  | 	// https://wiki.archlinux.org/index.php/Systemd/User | 
					
						
							|  |  |  | 	// This seems to work on Ubuntu | 
					
						
							|  |  |  | 	srvUserPath = ".config/systemd/user" | 
					
						
							| 
									
										
										
										
											2019-07-05 12:48:58 -06:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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 | 
					
						
							|  |  |  | 	name := conf.ReverseDNS | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_, err := getService(system, home, name) | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var cmds []Runnable | 
					
						
							|  |  |  | 	if system { | 
					
						
							|  |  |  | 		cmds = []Runnable{ | 
					
						
							|  |  |  | 			Runnable{ | 
					
						
							|  |  |  | 				Exec: "systemctl", | 
					
						
							|  |  |  | 				Args: []string{"daemon-reload"}, | 
					
						
							|  |  |  | 				Must: false, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			Runnable{ | 
					
						
							|  |  |  | 				Exec: "systemctl", | 
					
						
							|  |  |  | 				Args: []string{"stop", name + ".service"}, | 
					
						
							|  |  |  | 				Must: false, | 
					
						
							|  |  |  | 			}, | 
					
						
							| 
									
										
										
										
											2019-07-10 02:00:43 -06:00
										 |  |  | 			Runnable{ | 
					
						
							|  |  |  | 				Exec:     "systemctl", | 
					
						
							|  |  |  | 				Args:     []string{"enable", name + ".service"}, | 
					
						
							|  |  |  | 				Badwords: []string{"not found", "failed"}, | 
					
						
							|  |  |  | 				Must:     true, | 
					
						
							|  |  |  | 			}, | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 			Runnable{ | 
					
						
							|  |  |  | 				Exec:     "systemctl", | 
					
						
							|  |  |  | 				Args:     []string{"start", name + ".service"}, | 
					
						
							|  |  |  | 				Badwords: []string{"not found", "failed"}, | 
					
						
							|  |  |  | 				Must:     true, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		cmds = []Runnable{ | 
					
						
							|  |  |  | 			Runnable{ | 
					
						
							|  |  |  | 				Exec: "systemctl", | 
					
						
							|  |  |  | 				Args: []string{"--user", "daemon-reload"}, | 
					
						
							|  |  |  | 				Must: false, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			Runnable{ | 
					
						
							|  |  |  | 				Exec: "systemctl", | 
					
						
							|  |  |  | 				Args: []string{"stop", "--user", name + ".service"}, | 
					
						
							|  |  |  | 				Must: false, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			Runnable{ | 
					
						
							|  |  |  | 				Exec:     "systemctl", | 
					
						
							|  |  |  | 				Args:     []string{"start", "--user", name + ".service"}, | 
					
						
							|  |  |  | 				Badwords: []string{"not found", "failed"}, | 
					
						
							|  |  |  | 				Must:     true, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cmds = adjustPrivs(system, cmds) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-10 01:16:45 -06:00
										 |  |  | 	typ := "USER MODE" | 
					
						
							|  |  |  | 	if system { | 
					
						
							|  |  |  | 		typ = "SYSTEM" | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	fmt.Printf("Starting systemd %s service unit...\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 | 
					
						
							|  |  |  | 	name := conf.ReverseDNS | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_, err := getService(system, home, name) | 
					
						
							|  |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var cmds []Runnable | 
					
						
							|  |  |  | 	badwords := []string{"Failed to stop"} | 
					
						
							|  |  |  | 	if system { | 
					
						
							|  |  |  | 		cmds = []Runnable{ | 
					
						
							|  |  |  | 			Runnable{ | 
					
						
							|  |  |  | 				Exec:     "systemctl", | 
					
						
							|  |  |  | 				Args:     []string{"stop", name + ".service"}, | 
					
						
							|  |  |  | 				Must:     true, | 
					
						
							|  |  |  | 				Badwords: badwords, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		cmds = []Runnable{ | 
					
						
							|  |  |  | 			Runnable{ | 
					
						
							|  |  |  | 				Exec:     "systemctl", | 
					
						
							|  |  |  | 				Args:     []string{"stop", "--user", name + ".service"}, | 
					
						
							|  |  |  | 				Must:     true, | 
					
						
							|  |  |  | 				Badwords: badwords, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cmds = adjustPrivs(system, cmds) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fmt.Println() | 
					
						
							|  |  |  | 	typ := "USER MODE" | 
					
						
							|  |  |  | 	if system { | 
					
						
							|  |  |  | 		typ = "SYSTEM" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fmt.Printf("Stopping systemd %s service...\n", typ) | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 	for i := range cmds { | 
					
						
							|  |  |  | 		exe := cmds[i] | 
					
						
							| 
									
										
										
										
											2019-07-07 06:55:01 +00:00
										 |  |  | 		fmt.Println("\t" + exe.String()) | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00: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/etc/systemd/system/_name_.service.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
										 |  |  | 	// Linux-specific config options | 
					
						
							|  |  |  | 	if c.System { | 
					
						
							|  |  |  | 		if "" == c.User { | 
					
						
							|  |  |  | 			c.User = "root" | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if "" == c.Group { | 
					
						
							|  |  |  | 		c.Group = c.User | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Check paths first | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 	serviceDir := srvSysPath | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	if !c.System { | 
					
						
							| 
									
										
										
										
											2019-07-05 12:48:58 -06:00
										 |  |  | 		serviceDir = filepath.Join(c.Home, srvUserPath) | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 		err := os.MkdirAll(serviceDir, 0755) | 
					
						
							| 
									
										
										
										
											2019-07-02 00:25:16 -06:00
										 |  |  | 		if nil != err { | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 	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 | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 	serviceName := c.Name + ".service" | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	servicePath := filepath.Join(serviceDir, serviceName) | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	if err := ioutil.WriteFile(servicePath, b, 0644); err != nil { | 
					
						
							|  |  |  | 		return "", fmt.Errorf("Error writing %s: %v", servicePath, err) | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 	// TODO --no-start | 
					
						
							| 
									
										
										
										
											2019-07-10 01:16:45 -06:00
										 |  |  | 	err = start(c) | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 	if nil != err { | 
					
						
							|  |  |  | 		sudo := "" | 
					
						
							|  |  |  | 		// --user-unit rather than --user --unit for older systemd | 
					
						
							|  |  |  | 		unit := "--user-unit" | 
					
						
							|  |  |  | 		if c.System { | 
					
						
							|  |  |  | 			sudo = "sudo " | 
					
						
							|  |  |  | 			unit = "--unit" | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		fmt.Printf("If things don't go well you should be able to get additional logging from journalctl:\n") | 
					
						
							|  |  |  | 		fmt.Printf("\t%sjournalctl -xe %s %s.service\n", sudo, unit, c.Name) | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 		return "", err | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-07 06:48:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 20:50:00 -06:00
										 |  |  | 	return "systemd", nil | 
					
						
							| 
									
										
										
										
											2019-07-01 02:44:48 -06:00
										 |  |  | } |