| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | // Copyright (c) 2020 Tailscale Inc & AUTHORS All rights reserved. | 
					
						
							|  |  |  | // Use of this source code is governed by a BSD-style | 
					
						
							|  |  |  | // license that can be found in the LICENSE file. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 07:56:38 -07:00
										 |  |  | // +build darwin,amd64 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | package portlist | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bufio" | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"log" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2020-03-13 20:53:58 -07:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	exec "tailscale.com/tempfork/osexec" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // We have to run netstat, which is a bit expensive, so don't do it too often. | 
					
						
							| 
									
										
										
										
											2020-03-13 20:53:58 -07:00
										 |  |  | const pollInterval = 5 * time.Second | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func listPorts() (List, error) { | 
					
						
							|  |  |  | 	return listPortsNetstat("-na") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // In theory, lsof could replace the function of both listPorts() and | 
					
						
							|  |  |  | // addProcesses(), since it provides a superset of the netstat output. | 
					
						
							|  |  |  | // However, "netstat -na" runs ~100x faster than lsof on my machine, so | 
					
						
							|  |  |  | // we should do it only if the list of open ports has actually changed. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // TODO(apenwarr): this fails in a macOS sandbox (ie. our usual case). | 
					
						
							|  |  |  | // We might as well just delete this code if we can't find a solution. | 
					
						
							|  |  |  | func addProcesses(pl []Port) ([]Port, error) { | 
					
						
							|  |  |  | 	exe, err := exec.LookPath("lsof") | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, fmt.Errorf("lsof: lookup: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-03-17 20:19:39 -07:00
										 |  |  | 	output, err := exec.Command(exe, "-F", "-n", "-P", "-O", "-S2", "-T", "-i4", "-i6").Output() | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		xe, ok := err.(*exec.ExitError) | 
					
						
							|  |  |  | 		stderr := "" | 
					
						
							|  |  |  | 		if ok { | 
					
						
							|  |  |  | 			stderr = strings.TrimSpace(string(xe.Stderr)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// fails when run in a macOS sandbox, so make this non-fatal. | 
					
						
							|  |  |  | 		log.Printf("portlist: lsof: %v (%q)\n", err, stderr) | 
					
						
							|  |  |  | 		return pl, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	type ProtoPort struct { | 
					
						
							|  |  |  | 		proto string | 
					
						
							|  |  |  | 		port  uint16 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	m := map[ProtoPort]*Port{} | 
					
						
							|  |  |  | 	for i := range pl { | 
					
						
							|  |  |  | 		pp := ProtoPort{pl[i].Proto, pl[i].Port} | 
					
						
							|  |  |  | 		m[pp] = &pl[i] | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	r := bytes.NewReader(output) | 
					
						
							|  |  |  | 	scanner := bufio.NewScanner(r) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var cmd, proto string | 
					
						
							|  |  |  | 	for scanner.Scan() { | 
					
						
							|  |  |  | 		line := scanner.Text() | 
					
						
							| 
									
										
										
										
											2020-03-17 20:19:39 -07:00
										 |  |  | 		if line == "" { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		field, val := line[0], line[1:] | 
					
						
							|  |  |  | 		switch field { | 
					
						
							|  |  |  | 		case 'p': | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 			// starting a new process | 
					
						
							|  |  |  | 			cmd = "" | 
					
						
							|  |  |  | 			proto = "" | 
					
						
							| 
									
										
										
										
											2020-03-17 20:19:39 -07:00
										 |  |  | 		case 'c': | 
					
						
							|  |  |  | 			cmd = val | 
					
						
							|  |  |  | 		case 'P': | 
					
						
							|  |  |  | 			proto = strings.ToLower(val) | 
					
						
							|  |  |  | 		case 'n': | 
					
						
							|  |  |  | 			if strings.Contains(val, "->") { | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// a listening port | 
					
						
							|  |  |  | 			port := parsePort(val) | 
					
						
							|  |  |  | 			if port > 0 { | 
					
						
							|  |  |  | 				pp := ProtoPort{proto, uint16(port)} | 
					
						
							|  |  |  | 				p := m[pp] | 
					
						
							|  |  |  | 				if p != nil { | 
					
						
							|  |  |  | 					p.Process = cmd | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					fmt.Fprintf(os.Stderr, "weird: missing %v\n", pp) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return pl, nil | 
					
						
							|  |  |  | } |