| 
									
										
										
										
											2022-03-28 22:23:47 +02:00
										 |  |  | //go:build !windows | 
					
						
							| 
									
										
										
										
											2017-04-06 20:36:09 +02:00
										 |  |  | // +build !windows | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-26 23:03:25 +02:00
										 |  |  | package fs | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2023-07-08 18:18:13 +02:00
										 |  |  | 	"path/filepath" | 
					
						
							| 
									
										
										
										
											2017-04-07 20:37:20 +02:00
										 |  |  | 	"runtime" | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 	"syscall" | 
					
						
							|  |  |  | 	"testing" | 
					
						
							|  |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2023-07-08 18:18:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-26 23:03:25 +02:00
										 |  |  | 	"github.com/restic/restic/internal/restic" | 
					
						
							| 
									
										
										
										
											2023-07-08 18:18:13 +02:00
										 |  |  | 	rtest "github.com/restic/restic/internal/test" | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func stat(t testing.TB, filename string) (fi os.FileInfo, ok bool) { | 
					
						
							|  |  |  | 	fi, err := os.Lstat(filename) | 
					
						
							|  |  |  | 	if err != nil && os.IsNotExist(err) { | 
					
						
							|  |  |  | 		return fi, false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return fi, true | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-26 23:03:25 +02:00
										 |  |  | func checkFile(t testing.TB, stat *syscall.Stat_t, node *restic.Node) { | 
					
						
							| 
									
										
										
										
											2023-07-08 18:18:13 +02:00
										 |  |  | 	t.Helper() | 
					
						
							| 
									
										
										
										
											2017-04-06 20:36:09 +02:00
										 |  |  | 	if uint32(node.Mode.Perm()) != uint32(stat.Mode&0777) { | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 		t.Errorf("Mode does not match, want %v, got %v", stat.Mode&0777, node.Mode) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 20:36:09 +02:00
										 |  |  | 	if node.Inode != uint64(stat.Ino) { | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 		t.Errorf("Inode does not match, want %v, got %v", stat.Ino, node.Inode) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 20:36:09 +02:00
										 |  |  | 	if node.DeviceID != uint64(stat.Dev) { | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 		t.Errorf("Dev does not match, want %v, got %v", stat.Dev, node.DeviceID) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-08 18:18:13 +02:00
										 |  |  | 	if node.Size != uint64(stat.Size) && node.Type != "symlink" { | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 		t.Errorf("Size does not match, want %v, got %v", stat.Size, node.Size) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 20:36:09 +02:00
										 |  |  | 	if node.Links != uint64(stat.Nlink) { | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 		t.Errorf("Links does not match, want %v, got %v", stat.Nlink, node.Links) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 20:36:09 +02:00
										 |  |  | 	if node.UID != stat.Uid { | 
					
						
							|  |  |  | 		t.Errorf("UID does not match, want %v, got %v", stat.Uid, node.UID) | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 20:36:09 +02:00
										 |  |  | 	if node.GID != stat.Gid { | 
					
						
							|  |  |  | 		t.Errorf("UID does not match, want %v, got %v", stat.Gid, node.GID) | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 20:36:09 +02:00
										 |  |  | 	// use the os dependent function to compare the timestamps | 
					
						
							|  |  |  | 	s, ok := toStatT(stat) | 
					
						
							|  |  |  | 	if !ok { | 
					
						
							|  |  |  | 		return | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 20:36:09 +02:00
										 |  |  | 	mtime := s.mtim() | 
					
						
							|  |  |  | 	if node.ModTime != time.Unix(mtime.Unix()) { | 
					
						
							|  |  |  | 		t.Errorf("ModTime does not match, want %v, got %v", time.Unix(mtime.Unix()), node.ModTime) | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 20:36:09 +02:00
										 |  |  | 	ctime := s.ctim() | 
					
						
							|  |  |  | 	if node.ChangeTime != time.Unix(ctime.Unix()) { | 
					
						
							|  |  |  | 		t.Errorf("ChangeTime does not match, want %v, got %v", time.Unix(ctime.Unix()), node.ChangeTime) | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-04-06 20:36:09 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	atime := s.atim() | 
					
						
							|  |  |  | 	if node.AccessTime != time.Unix(atime.Unix()) { | 
					
						
							|  |  |  | 		t.Errorf("AccessTime does not match, want %v, got %v", time.Unix(atime.Unix()), node.AccessTime) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-26 23:03:25 +02:00
										 |  |  | func checkDevice(t testing.TB, stat *syscall.Stat_t, node *restic.Node) { | 
					
						
							| 
									
										
										
										
											2017-04-06 20:36:09 +02:00
										 |  |  | 	if node.Device != uint64(stat.Rdev) { | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 		t.Errorf("Rdev does not match, want %v, got %v", stat.Rdev, node.Device) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestNodeFromFileInfo(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2023-07-08 18:18:13 +02:00
										 |  |  | 	tmp := t.TempDir() | 
					
						
							|  |  |  | 	symlink := filepath.Join(tmp, "symlink") | 
					
						
							|  |  |  | 	rtest.OK(t, os.Symlink("target", symlink)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-07 20:37:20 +02:00
										 |  |  | 	type Test struct { | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 		filename string | 
					
						
							|  |  |  | 		canSkip  bool | 
					
						
							| 
									
										
										
										
											2017-04-07 20:37:20 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	var tests = []Test{ | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 		{"node_test.go", false}, | 
					
						
							|  |  |  | 		{"/dev/sda", true}, | 
					
						
							| 
									
										
										
										
											2023-07-08 18:18:13 +02:00
										 |  |  | 		{symlink, false}, | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-07 20:37:20 +02:00
										 |  |  | 	// on darwin, users are not permitted to list the extended attributes of | 
					
						
							|  |  |  | 	// /dev/null, therefore skip it. | 
					
						
							| 
									
										
										
										
											2022-01-25 18:05:56 -08:00
										 |  |  | 	// on solaris, /dev/null is a symlink to a device node in /devices | 
					
						
							|  |  |  | 	// which does not support extended attributes, therefore skip it. | 
					
						
							|  |  |  | 	if runtime.GOOS != "darwin" && runtime.GOOS != "solaris" { | 
					
						
							| 
									
										
										
										
											2017-04-07 20:37:20 +02:00
										 |  |  | 		tests = append(tests, Test{"/dev/null", true}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 	for _, test := range tests { | 
					
						
							|  |  |  | 		t.Run("", func(t *testing.T) { | 
					
						
							|  |  |  | 			fi, found := stat(t, test.filename) | 
					
						
							|  |  |  | 			if !found && test.canSkip { | 
					
						
							| 
									
										
										
										
											2018-01-17 23:14:37 +01:00
										 |  |  | 				t.Skipf("%v not found in filesystem", test.filename) | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 				return | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if fi.Sys() == nil { | 
					
						
							|  |  |  | 				t.Skip("fi.Sys() is nil") | 
					
						
							|  |  |  | 				return | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			s, ok := fi.Sys().(*syscall.Stat_t) | 
					
						
							|  |  |  | 			if !ok { | 
					
						
							| 
									
										
										
										
											2018-07-14 11:51:34 +02:00
										 |  |  | 				t.Skipf("fi type is %T, not stat_t", fi.Sys()) | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 				return | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-31 20:48:03 +01:00
										 |  |  | 			node, err := NodeFromFileInfo(test.filename, fi, false) | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				t.Fatal(err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			switch node.Type { | 
					
						
							| 
									
										
										
										
											2023-07-08 18:18:13 +02:00
										 |  |  | 			case "file", "symlink": | 
					
						
							| 
									
										
										
										
											2017-04-05 20:51:26 +02:00
										 |  |  | 				checkFile(t, s, node) | 
					
						
							|  |  |  | 			case "dev", "chardev": | 
					
						
							|  |  |  | 				checkFile(t, s, node) | 
					
						
							|  |  |  | 				checkDevice(t, s, node) | 
					
						
							|  |  |  | 			default: | 
					
						
							|  |  |  | 				t.Fatalf("invalid node type %q", node.Type) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |