| 
									
										
										
											
												go.mod, cmd/tailscaled, ipn/localapi, util/osdiag, util/winutil, util/winutil/authenticode: add Windows module list to OS-specific logs that are written upon bugreport
* We update wingoes to pick up new version information functionality
  (See pe/version.go in the https://github.com/dblohm7/wingoes repo);
* We move the existing LogSupportInfo code (including necessary syscall
  stubs) out of util/winutil into a new package, util/osdiag, and implement
  the public LogSupportInfo function may be implemented for other platforms
  as needed;
* We add a new reason argument to LogSupportInfo and wire that into
  localapi's bugreport implementation;
* We add module information to the Windows implementation of LogSupportInfo
  when reason indicates a bugreport. We enumerate all loaded modules in our
  process, and for each one we gather debug, authenticode signature, and
  version information.
Fixes #7802
Signed-off-by: Aaron Klotz <aaron@tailscale.com>
											
										 
											2023-06-26 11:50:45 -06:00
										 |  |  | // Copyright (c) Tailscale Inc & AUTHORS | 
					
						
							|  |  |  | // SPDX-License-Identifier: BSD-3-Clause | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package osdiag | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"errors" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2023-08-17 09:40:19 -07:00
										 |  |  | 	"maps" | 
					
						
							| 
									
										
										
											
												go.mod, cmd/tailscaled, ipn/localapi, util/osdiag, util/winutil, util/winutil/authenticode: add Windows module list to OS-specific logs that are written upon bugreport
* We update wingoes to pick up new version information functionality
  (See pe/version.go in the https://github.com/dblohm7/wingoes repo);
* We move the existing LogSupportInfo code (including necessary syscall
  stubs) out of util/winutil into a new package, util/osdiag, and implement
  the public LogSupportInfo function may be implemented for other platforms
  as needed;
* We add a new reason argument to LogSupportInfo and wire that into
  localapi's bugreport implementation;
* We add module information to the Windows implementation of LogSupportInfo
  when reason indicates a bugreport. We enumerate all loaded modules in our
  process, and for each one we gather debug, authenticode signature, and
  version information.
Fixes #7802
Signed-off-by: Aaron Klotz <aaron@tailscale.com>
											
										 
											2023-06-26 11:50:45 -06:00
										 |  |  | 	"strings" | 
					
						
							|  |  |  | 	"testing" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"golang.org/x/sys/windows/registry" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func makeLongBinaryValue() []byte { | 
					
						
							|  |  |  | 	buf := make([]byte, maxBinaryValueLen*2) | 
					
						
							|  |  |  | 	for i, _ := range buf { | 
					
						
							|  |  |  | 		buf[i] = byte(i % 0xFF) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return buf | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var testData = map[string]any{ | 
					
						
							|  |  |  | 	"":                              "I am the default", | 
					
						
							|  |  |  | 	"StringEmpty":                   "", | 
					
						
							|  |  |  | 	"StringShort":                   "Hello", | 
					
						
							|  |  |  | 	"StringLong":                    strings.Repeat("7", initialValueBufLen+1), | 
					
						
							|  |  |  | 	"MultiStringEmpty":              []string{}, | 
					
						
							|  |  |  | 	"MultiStringSingle":             []string{"Foo"}, | 
					
						
							|  |  |  | 	"MultiStringSingleEmpty":        []string{""}, | 
					
						
							|  |  |  | 	"MultiString":                   []string{"Foo", "Bar", "Baz"}, | 
					
						
							|  |  |  | 	"MultiStringWithEmptyBeginning": []string{"", "Foo", "Bar"}, | 
					
						
							|  |  |  | 	"MultiStringWithEmptyMiddle":    []string{"Foo", "", "Bar"}, | 
					
						
							|  |  |  | 	"MultiStringWithEmptyEnd":       []string{"Foo", "Bar", ""}, | 
					
						
							|  |  |  | 	"DWord":                         uint32(0x12345678), | 
					
						
							|  |  |  | 	"QWord":                         uint64(0x123456789abcdef0), | 
					
						
							|  |  |  | 	"BinaryEmpty":                   []byte{}, | 
					
						
							|  |  |  | 	"BinaryShort":                   []byte{0x01, 0x02, 0x03, 0x04}, | 
					
						
							|  |  |  | 	"BinaryLong":                    makeLongBinaryValue(), | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ( | 
					
						
							|  |  |  | 	keyNameTest    = `SOFTWARE\Tailscale Test` | 
					
						
							|  |  |  | 	subKeyNameTest = "SubKey" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func setValues(t *testing.T, k registry.Key) { | 
					
						
							|  |  |  | 	for vk, v := range testData { | 
					
						
							|  |  |  | 		var err error | 
					
						
							|  |  |  | 		switch tv := v.(type) { | 
					
						
							|  |  |  | 		case string: | 
					
						
							|  |  |  | 			err = k.SetStringValue(vk, tv) | 
					
						
							|  |  |  | 		case []string: | 
					
						
							|  |  |  | 			err = k.SetStringsValue(vk, tv) | 
					
						
							|  |  |  | 		case uint32: | 
					
						
							|  |  |  | 			err = k.SetDWordValue(vk, tv) | 
					
						
							|  |  |  | 		case uint64: | 
					
						
							|  |  |  | 			err = k.SetQWordValue(vk, tv) | 
					
						
							|  |  |  | 		case []byte: | 
					
						
							|  |  |  | 			err = k.SetBinaryValue(vk, tv) | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			t.Fatalf("Unknown type") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Fatalf("Error setting %q: %v", vk, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestRegistrySupportInfo(t *testing.T) { | 
					
						
							|  |  |  | 	// Make sure the key doesn't exist yet | 
					
						
							|  |  |  | 	k, err := registry.OpenKey(registry.CURRENT_USER, keyNameTest, registry.READ) | 
					
						
							|  |  |  | 	switch { | 
					
						
							|  |  |  | 	case err == nil: | 
					
						
							|  |  |  | 		k.Close() | 
					
						
							|  |  |  | 		t.Fatalf("Test key already exists") | 
					
						
							|  |  |  | 	case !errors.Is(err, registry.ErrNotExist): | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	func() { | 
					
						
							|  |  |  | 		k, _, err := registry.CreateKey(registry.CURRENT_USER, keyNameTest, registry.WRITE) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Fatalf("Error creating test key: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		defer k.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		setValues(t, k) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		sk, _, err := registry.CreateKey(k, subKeyNameTest, registry.WRITE) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Fatalf("Error creating test subkey: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		defer sk.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		setValues(t, sk) | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t.Cleanup(func() { | 
					
						
							|  |  |  | 		registry.DeleteKey(registry.CURRENT_USER, keyNameTest+"\\"+subKeyNameTest) | 
					
						
							|  |  |  | 		registry.DeleteKey(registry.CURRENT_USER, keyNameTest) | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wantValuesData := maps.Clone(testData) | 
					
						
							|  |  |  | 	wantValuesData["BinaryLong"] = (wantValuesData["BinaryLong"].([]byte))[:maxBinaryValueLen] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wantKeyData := make(map[string]any) | 
					
						
							|  |  |  | 	maps.Copy(wantKeyData, wantValuesData) | 
					
						
							|  |  |  | 	wantSubKeyData := make(map[string]any) | 
					
						
							|  |  |  | 	maps.Copy(wantSubKeyData, wantValuesData) | 
					
						
							|  |  |  | 	wantKeyData[subKeyNameTest] = wantSubKeyData | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wantData := map[string]any{ | 
					
						
							|  |  |  | 		"HKCU\\" + keyNameTest: wantKeyData, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gotData, err := getRegistrySupportInfo(registry.CURRENT_USER, []string{keyNameTest}) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Errorf("getRegistrySupportInfo error: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	want, got := fmt.Sprintf("%#v", wantData), fmt.Sprintf("%#v", gotData) | 
					
						
							|  |  |  | 	if want != got { | 
					
						
							|  |  |  | 		t.Errorf("Compare error: want\n%s,\ngot %s", want, got) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |