github.com/sbinet/go@v0.0.0-20160827155028-54d7de7dd62b/src/os/os_test.go (about) 1 // Copyright 2009 The Go Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 package os_test 6 7 import ( 8 "bytes" 9 "errors" 10 "flag" 11 "fmt" 12 "internal/testenv" 13 "io" 14 "io/ioutil" 15 . "os" 16 osexec "os/exec" 17 "path/filepath" 18 "reflect" 19 "runtime" 20 "sort" 21 "strings" 22 "sync" 23 "syscall" 24 "testing" 25 "time" 26 ) 27 28 var dot = []string{ 29 "dir_unix.go", 30 "env.go", 31 "error.go", 32 "file.go", 33 "os_test.go", 34 "types.go", 35 "stat_darwin.go", 36 "stat_linux.go", 37 } 38 39 type sysDir struct { 40 name string 41 files []string 42 } 43 44 var sysdir = func() *sysDir { 45 switch runtime.GOOS { 46 case "android": 47 return &sysDir{ 48 "/system/lib", 49 []string{ 50 "libmedia.so", 51 "libpowermanager.so", 52 }, 53 } 54 case "darwin": 55 switch runtime.GOARCH { 56 case "arm", "arm64": 57 wd, err := syscall.Getwd() 58 if err != nil { 59 wd = err.Error() 60 } 61 return &sysDir{ 62 filepath.Join(wd, "..", ".."), 63 []string{ 64 "ResourceRules.plist", 65 "Info.plist", 66 }, 67 } 68 } 69 case "windows": 70 return &sysDir{ 71 Getenv("SystemRoot") + "\\system32\\drivers\\etc", 72 []string{ 73 "networks", 74 "protocol", 75 "services", 76 }, 77 } 78 case "plan9": 79 return &sysDir{ 80 "/lib/ndb", 81 []string{ 82 "common", 83 "local", 84 }, 85 } 86 } 87 return &sysDir{ 88 "/etc", 89 []string{ 90 "group", 91 "hosts", 92 "passwd", 93 }, 94 } 95 }() 96 97 func size(name string, t *testing.T) int64 { 98 file, err := Open(name) 99 if err != nil { 100 t.Fatal("open failed:", err) 101 } 102 defer file.Close() 103 var buf [100]byte 104 len := 0 105 for { 106 n, e := file.Read(buf[0:]) 107 len += n 108 if e == io.EOF { 109 break 110 } 111 if e != nil { 112 t.Fatal("read failed:", err) 113 } 114 } 115 return int64(len) 116 } 117 118 func equal(name1, name2 string) (r bool) { 119 switch runtime.GOOS { 120 case "windows": 121 r = strings.ToLower(name1) == strings.ToLower(name2) 122 default: 123 r = name1 == name2 124 } 125 return 126 } 127 128 // localTmp returns a local temporary directory not on NFS. 129 func localTmp() string { 130 switch runtime.GOOS { 131 case "android", "windows": 132 return TempDir() 133 case "darwin": 134 switch runtime.GOARCH { 135 case "arm", "arm64": 136 return TempDir() 137 } 138 } 139 return "/tmp" 140 } 141 142 func newFile(testName string, t *testing.T) (f *File) { 143 f, err := ioutil.TempFile(localTmp(), "_Go_"+testName) 144 if err != nil { 145 t.Fatalf("TempFile %s: %s", testName, err) 146 } 147 return 148 } 149 150 func newDir(testName string, t *testing.T) (name string) { 151 name, err := ioutil.TempDir(localTmp(), "_Go_"+testName) 152 if err != nil { 153 t.Fatalf("TempDir %s: %s", testName, err) 154 } 155 return 156 } 157 158 var sfdir = sysdir.name 159 var sfname = sysdir.files[0] 160 161 func TestStat(t *testing.T) { 162 path := sfdir + "/" + sfname 163 dir, err := Stat(path) 164 if err != nil { 165 t.Fatal("stat failed:", err) 166 } 167 if !equal(sfname, dir.Name()) { 168 t.Error("name should be ", sfname, "; is", dir.Name()) 169 } 170 filesize := size(path, t) 171 if dir.Size() != filesize { 172 t.Error("size should be", filesize, "; is", dir.Size()) 173 } 174 } 175 176 func TestFstat(t *testing.T) { 177 path := sfdir + "/" + sfname 178 file, err1 := Open(path) 179 if err1 != nil { 180 t.Fatal("open failed:", err1) 181 } 182 defer file.Close() 183 dir, err2 := file.Stat() 184 if err2 != nil { 185 t.Fatal("fstat failed:", err2) 186 } 187 if !equal(sfname, dir.Name()) { 188 t.Error("name should be ", sfname, "; is", dir.Name()) 189 } 190 filesize := size(path, t) 191 if dir.Size() != filesize { 192 t.Error("size should be", filesize, "; is", dir.Size()) 193 } 194 } 195 196 func TestLstat(t *testing.T) { 197 path := sfdir + "/" + sfname 198 dir, err := Lstat(path) 199 if err != nil { 200 t.Fatal("lstat failed:", err) 201 } 202 if !equal(sfname, dir.Name()) { 203 t.Error("name should be ", sfname, "; is", dir.Name()) 204 } 205 filesize := size(path, t) 206 if dir.Size() != filesize { 207 t.Error("size should be", filesize, "; is", dir.Size()) 208 } 209 } 210 211 // Read with length 0 should not return EOF. 212 func TestRead0(t *testing.T) { 213 path := sfdir + "/" + sfname 214 f, err := Open(path) 215 if err != nil { 216 t.Fatal("open failed:", err) 217 } 218 defer f.Close() 219 220 b := make([]byte, 0) 221 n, err := f.Read(b) 222 if n != 0 || err != nil { 223 t.Errorf("Read(0) = %d, %v, want 0, nil", n, err) 224 } 225 b = make([]byte, 100) 226 n, err = f.Read(b) 227 if n <= 0 || err != nil { 228 t.Errorf("Read(100) = %d, %v, want >0, nil", n, err) 229 } 230 } 231 232 func testReaddirnames(dir string, contents []string, t *testing.T) { 233 file, err := Open(dir) 234 if err != nil { 235 t.Fatalf("open %q failed: %v", dir, err) 236 } 237 defer file.Close() 238 s, err2 := file.Readdirnames(-1) 239 if err2 != nil { 240 t.Fatalf("readdirnames %q failed: %v", dir, err2) 241 } 242 for _, m := range contents { 243 found := false 244 for _, n := range s { 245 if n == "." || n == ".." { 246 t.Errorf("got %s in directory", n) 247 } 248 if equal(m, n) { 249 if found { 250 t.Error("present twice:", m) 251 } 252 found = true 253 } 254 } 255 if !found { 256 t.Error("could not find", m) 257 } 258 } 259 } 260 261 func testReaddir(dir string, contents []string, t *testing.T) { 262 file, err := Open(dir) 263 if err != nil { 264 t.Fatalf("open %q failed: %v", dir, err) 265 } 266 defer file.Close() 267 s, err2 := file.Readdir(-1) 268 if err2 != nil { 269 t.Fatalf("readdir %q failed: %v", dir, err2) 270 } 271 for _, m := range contents { 272 found := false 273 for _, n := range s { 274 if equal(m, n.Name()) { 275 if found { 276 t.Error("present twice:", m) 277 } 278 found = true 279 } 280 } 281 if !found { 282 t.Error("could not find", m) 283 } 284 } 285 } 286 287 func TestReaddirnames(t *testing.T) { 288 testReaddirnames(".", dot, t) 289 testReaddirnames(sysdir.name, sysdir.files, t) 290 } 291 292 func TestReaddir(t *testing.T) { 293 testReaddir(".", dot, t) 294 testReaddir(sysdir.name, sysdir.files, t) 295 } 296 297 func benchmarkReaddirname(path string, b *testing.B) { 298 var nentries int 299 for i := 0; i < b.N; i++ { 300 f, err := Open(path) 301 if err != nil { 302 b.Fatalf("open %q failed: %v", path, err) 303 } 304 ns, err := f.Readdirnames(-1) 305 f.Close() 306 if err != nil { 307 b.Fatalf("readdirnames %q failed: %v", path, err) 308 } 309 nentries = len(ns) 310 } 311 b.Logf("benchmarkReaddirname %q: %d entries", path, nentries) 312 } 313 314 func benchmarkReaddir(path string, b *testing.B) { 315 var nentries int 316 for i := 0; i < b.N; i++ { 317 f, err := Open(path) 318 if err != nil { 319 b.Fatalf("open %q failed: %v", path, err) 320 } 321 fs, err := f.Readdir(-1) 322 f.Close() 323 if err != nil { 324 b.Fatalf("readdir %q failed: %v", path, err) 325 } 326 nentries = len(fs) 327 } 328 b.Logf("benchmarkReaddir %q: %d entries", path, nentries) 329 } 330 331 func BenchmarkReaddirname(b *testing.B) { 332 benchmarkReaddirname(".", b) 333 } 334 335 func BenchmarkReaddir(b *testing.B) { 336 benchmarkReaddir(".", b) 337 } 338 339 // Read the directory one entry at a time. 340 func smallReaddirnames(file *File, length int, t *testing.T) []string { 341 names := make([]string, length) 342 count := 0 343 for { 344 d, err := file.Readdirnames(1) 345 if err == io.EOF { 346 break 347 } 348 if err != nil { 349 t.Fatalf("readdirnames %q failed: %v", file.Name(), err) 350 } 351 if len(d) == 0 { 352 t.Fatalf("readdirnames %q returned empty slice and no error", file.Name()) 353 } 354 names[count] = d[0] 355 count++ 356 } 357 return names[0:count] 358 } 359 360 // Check that reading a directory one entry at a time gives the same result 361 // as reading it all at once. 362 func TestReaddirnamesOneAtATime(t *testing.T) { 363 // big directory that doesn't change often. 364 dir := "/usr/bin" 365 switch runtime.GOOS { 366 case "android": 367 dir = "/system/bin" 368 case "darwin": 369 switch runtime.GOARCH { 370 case "arm", "arm64": 371 wd, err := Getwd() 372 if err != nil { 373 t.Fatal(err) 374 } 375 dir = wd 376 } 377 case "plan9": 378 dir = "/bin" 379 case "windows": 380 dir = Getenv("SystemRoot") + "\\system32" 381 } 382 file, err := Open(dir) 383 if err != nil { 384 t.Fatalf("open %q failed: %v", dir, err) 385 } 386 defer file.Close() 387 all, err1 := file.Readdirnames(-1) 388 if err1 != nil { 389 t.Fatalf("readdirnames %q failed: %v", dir, err1) 390 } 391 file1, err2 := Open(dir) 392 if err2 != nil { 393 t.Fatalf("open %q failed: %v", dir, err2) 394 } 395 defer file1.Close() 396 small := smallReaddirnames(file1, len(all)+100, t) // +100 in case we screw up 397 if len(small) < len(all) { 398 t.Fatalf("len(small) is %d, less than %d", len(small), len(all)) 399 } 400 for i, n := range all { 401 if small[i] != n { 402 t.Errorf("small read %q mismatch: %v", small[i], n) 403 } 404 } 405 } 406 407 func TestReaddirNValues(t *testing.T) { 408 if testing.Short() { 409 t.Skip("test.short; skipping") 410 } 411 dir, err := ioutil.TempDir("", "") 412 if err != nil { 413 t.Fatalf("TempDir: %v", err) 414 } 415 defer RemoveAll(dir) 416 for i := 1; i <= 105; i++ { 417 f, err := Create(filepath.Join(dir, fmt.Sprintf("%d", i))) 418 if err != nil { 419 t.Fatalf("Create: %v", err) 420 } 421 f.Write([]byte(strings.Repeat("X", i))) 422 f.Close() 423 } 424 425 var d *File 426 openDir := func() { 427 var err error 428 d, err = Open(dir) 429 if err != nil { 430 t.Fatalf("Open directory: %v", err) 431 } 432 } 433 434 readDirExpect := func(n, want int, wantErr error) { 435 fi, err := d.Readdir(n) 436 if err != wantErr { 437 t.Fatalf("Readdir of %d got error %v, want %v", n, err, wantErr) 438 } 439 if g, e := len(fi), want; g != e { 440 t.Errorf("Readdir of %d got %d files, want %d", n, g, e) 441 } 442 } 443 444 readDirNamesExpect := func(n, want int, wantErr error) { 445 fi, err := d.Readdirnames(n) 446 if err != wantErr { 447 t.Fatalf("Readdirnames of %d got error %v, want %v", n, err, wantErr) 448 } 449 if g, e := len(fi), want; g != e { 450 t.Errorf("Readdirnames of %d got %d files, want %d", n, g, e) 451 } 452 } 453 454 for _, fn := range []func(int, int, error){readDirExpect, readDirNamesExpect} { 455 // Test the slurp case 456 openDir() 457 fn(0, 105, nil) 458 fn(0, 0, nil) 459 d.Close() 460 461 // Slurp with -1 instead 462 openDir() 463 fn(-1, 105, nil) 464 fn(-2, 0, nil) 465 fn(0, 0, nil) 466 d.Close() 467 468 // Test the bounded case 469 openDir() 470 fn(1, 1, nil) 471 fn(2, 2, nil) 472 fn(105, 102, nil) // and tests buffer >100 case 473 fn(3, 0, io.EOF) 474 d.Close() 475 } 476 } 477 478 func touch(t *testing.T, name string) { 479 f, err := Create(name) 480 if err != nil { 481 t.Fatal(err) 482 } 483 if err := f.Close(); err != nil { 484 t.Fatal(err) 485 } 486 } 487 488 func TestReaddirStatFailures(t *testing.T) { 489 switch runtime.GOOS { 490 case "windows", "plan9": 491 // Windows and Plan 9 already do this correctly, 492 // but are structured with different syscalls such 493 // that they don't use Lstat, so the hook below for 494 // testing it wouldn't work. 495 t.Skipf("skipping test on %v", runtime.GOOS) 496 } 497 dir, err := ioutil.TempDir("", "") 498 if err != nil { 499 t.Fatalf("TempDir: %v", err) 500 } 501 defer RemoveAll(dir) 502 touch(t, filepath.Join(dir, "good1")) 503 touch(t, filepath.Join(dir, "x")) // will disappear or have an error 504 touch(t, filepath.Join(dir, "good2")) 505 defer func() { 506 *LstatP = Lstat 507 }() 508 var xerr error // error to return for x 509 *LstatP = func(path string) (FileInfo, error) { 510 if xerr != nil && strings.HasSuffix(path, "x") { 511 return nil, xerr 512 } 513 return Lstat(path) 514 } 515 readDir := func() ([]FileInfo, error) { 516 d, err := Open(dir) 517 if err != nil { 518 t.Fatal(err) 519 } 520 defer d.Close() 521 return d.Readdir(-1) 522 } 523 mustReadDir := func(testName string) []FileInfo { 524 fis, err := readDir() 525 if err != nil { 526 t.Fatalf("%s: Readdir: %v", testName, err) 527 } 528 return fis 529 } 530 names := func(fis []FileInfo) []string { 531 s := make([]string, len(fis)) 532 for i, fi := range fis { 533 s[i] = fi.Name() 534 } 535 sort.Strings(s) 536 return s 537 } 538 539 if got, want := names(mustReadDir("initial readdir")), 540 []string{"good1", "good2", "x"}; !reflect.DeepEqual(got, want) { 541 t.Errorf("initial readdir got %q; want %q", got, want) 542 } 543 544 xerr = ErrNotExist 545 if got, want := names(mustReadDir("with x disappearing")), 546 []string{"good1", "good2"}; !reflect.DeepEqual(got, want) { 547 t.Errorf("with x disappearing, got %q; want %q", got, want) 548 } 549 550 xerr = errors.New("some real error") 551 if _, err := readDir(); err != xerr { 552 t.Errorf("with a non-ErrNotExist error, got error %v; want %v", err, xerr) 553 } 554 } 555 556 // Readdir on a regular file should fail. 557 func TestReaddirOfFile(t *testing.T) { 558 f, err := ioutil.TempFile("", "_Go_ReaddirOfFile") 559 if err != nil { 560 t.Fatal(err) 561 } 562 defer Remove(f.Name()) 563 f.Write([]byte("foo")) 564 f.Close() 565 reg, err := Open(f.Name()) 566 if err != nil { 567 t.Fatal(err) 568 } 569 defer reg.Close() 570 571 names, err := reg.Readdirnames(-1) 572 if err == nil { 573 t.Error("Readdirnames succeeded; want non-nil error") 574 } 575 if len(names) > 0 { 576 t.Errorf("unexpected dir names in regular file: %q", names) 577 } 578 } 579 580 func TestHardLink(t *testing.T) { 581 if runtime.GOOS == "plan9" { 582 t.Skip("skipping on plan9, hardlinks not supported") 583 } 584 // From Android release M (Marshmallow), hard linking files is blocked 585 // and an attempt to call link() on a file will return EACCES. 586 // - https://code.google.com/p/android-developer-preview/issues/detail?id=3150 587 if runtime.GOOS == "android" { 588 t.Skip("skipping on android, hardlinks not supported") 589 } 590 defer chtmpdir(t)() 591 from, to := "hardlinktestfrom", "hardlinktestto" 592 Remove(from) // Just in case. 593 file, err := Create(to) 594 if err != nil { 595 t.Fatalf("open %q failed: %v", to, err) 596 } 597 defer Remove(to) 598 if err = file.Close(); err != nil { 599 t.Errorf("close %q failed: %v", to, err) 600 } 601 err = Link(to, from) 602 if err != nil { 603 t.Fatalf("link %q, %q failed: %v", to, from, err) 604 } 605 606 none := "hardlinktestnone" 607 err = Link(none, none) 608 // Check the returned error is well-formed. 609 if lerr, ok := err.(*LinkError); !ok || lerr.Error() == "" { 610 t.Errorf("link %q, %q failed to return a valid error", none, none) 611 } 612 613 defer Remove(from) 614 tostat, err := Stat(to) 615 if err != nil { 616 t.Fatalf("stat %q failed: %v", to, err) 617 } 618 fromstat, err := Stat(from) 619 if err != nil { 620 t.Fatalf("stat %q failed: %v", from, err) 621 } 622 if !SameFile(tostat, fromstat) { 623 t.Errorf("link %q, %q did not create hard link", to, from) 624 } 625 } 626 627 // chtmpdir changes the working directory to a new temporary directory and 628 // provides a cleanup function. Used when PWD is read-only. 629 func chtmpdir(t *testing.T) func() { 630 if runtime.GOOS != "darwin" || (runtime.GOARCH != "arm" && runtime.GOARCH != "arm64") { 631 return func() {} // only needed on darwin/arm{,64} 632 } 633 oldwd, err := Getwd() 634 if err != nil { 635 t.Fatalf("chtmpdir: %v", err) 636 } 637 d, err := ioutil.TempDir("", "test") 638 if err != nil { 639 t.Fatalf("chtmpdir: %v", err) 640 } 641 if err := Chdir(d); err != nil { 642 t.Fatalf("chtmpdir: %v", err) 643 } 644 return func() { 645 if err := Chdir(oldwd); err != nil { 646 t.Fatalf("chtmpdir: %v", err) 647 } 648 RemoveAll(d) 649 } 650 } 651 652 func TestSymlink(t *testing.T) { 653 testenv.MustHaveSymlink(t) 654 655 defer chtmpdir(t)() 656 from, to := "symlinktestfrom", "symlinktestto" 657 Remove(from) // Just in case. 658 file, err := Create(to) 659 if err != nil { 660 t.Fatalf("open %q failed: %v", to, err) 661 } 662 defer Remove(to) 663 if err = file.Close(); err != nil { 664 t.Errorf("close %q failed: %v", to, err) 665 } 666 err = Symlink(to, from) 667 if err != nil { 668 t.Fatalf("symlink %q, %q failed: %v", to, from, err) 669 } 670 defer Remove(from) 671 tostat, err := Lstat(to) 672 if err != nil { 673 t.Fatalf("stat %q failed: %v", to, err) 674 } 675 if tostat.Mode()&ModeSymlink != 0 { 676 t.Fatalf("stat %q claims to have found a symlink", to) 677 } 678 fromstat, err := Stat(from) 679 if err != nil { 680 t.Fatalf("stat %q failed: %v", from, err) 681 } 682 if !SameFile(tostat, fromstat) { 683 t.Errorf("symlink %q, %q did not create symlink", to, from) 684 } 685 fromstat, err = Lstat(from) 686 if err != nil { 687 t.Fatalf("lstat %q failed: %v", from, err) 688 } 689 if fromstat.Mode()&ModeSymlink == 0 { 690 t.Fatalf("symlink %q, %q did not create symlink", to, from) 691 } 692 fromstat, err = Stat(from) 693 if err != nil { 694 t.Fatalf("stat %q failed: %v", from, err) 695 } 696 if fromstat.Mode()&ModeSymlink != 0 { 697 t.Fatalf("stat %q did not follow symlink", from) 698 } 699 s, err := Readlink(from) 700 if err != nil { 701 t.Fatalf("readlink %q failed: %v", from, err) 702 } 703 if s != to { 704 t.Fatalf("after symlink %q != %q", s, to) 705 } 706 file, err = Open(from) 707 if err != nil { 708 t.Fatalf("open %q failed: %v", from, err) 709 } 710 file.Close() 711 } 712 713 func TestLongSymlink(t *testing.T) { 714 testenv.MustHaveSymlink(t) 715 716 defer chtmpdir(t)() 717 s := "0123456789abcdef" 718 // Long, but not too long: a common limit is 255. 719 s = s + s + s + s + s + s + s + s + s + s + s + s + s + s + s 720 from := "longsymlinktestfrom" 721 Remove(from) // Just in case. 722 err := Symlink(s, from) 723 if err != nil { 724 t.Fatalf("symlink %q, %q failed: %v", s, from, err) 725 } 726 defer Remove(from) 727 r, err := Readlink(from) 728 if err != nil { 729 t.Fatalf("readlink %q failed: %v", from, err) 730 } 731 if r != s { 732 t.Fatalf("after symlink %q != %q", r, s) 733 } 734 } 735 736 func TestRename(t *testing.T) { 737 defer chtmpdir(t)() 738 from, to := "renamefrom", "renameto" 739 // Ensure we are not testing the overwrite case here. 740 Remove(from) 741 Remove(to) 742 743 file, err := Create(from) 744 if err != nil { 745 t.Fatalf("open %q failed: %v", from, err) 746 } 747 if err = file.Close(); err != nil { 748 t.Errorf("close %q failed: %v", from, err) 749 } 750 err = Rename(from, to) 751 if err != nil { 752 t.Fatalf("rename %q, %q failed: %v", to, from, err) 753 } 754 defer Remove(to) 755 _, err = Stat(to) 756 if err != nil { 757 t.Errorf("stat %q failed: %v", to, err) 758 } 759 } 760 761 func TestRenameOverwriteDest(t *testing.T) { 762 defer chtmpdir(t)() 763 from, to := "renamefrom", "renameto" 764 // Just in case. 765 Remove(from) 766 Remove(to) 767 768 toData := []byte("to") 769 fromData := []byte("from") 770 771 err := ioutil.WriteFile(to, toData, 0777) 772 if err != nil { 773 t.Fatalf("write file %q failed: %v", to, err) 774 } 775 776 err = ioutil.WriteFile(from, fromData, 0777) 777 if err != nil { 778 t.Fatalf("write file %q failed: %v", from, err) 779 } 780 err = Rename(from, to) 781 if err != nil { 782 t.Fatalf("rename %q, %q failed: %v", to, from, err) 783 } 784 defer Remove(to) 785 786 _, err = Stat(from) 787 if err == nil { 788 t.Errorf("from file %q still exists", from) 789 } 790 if err != nil && !IsNotExist(err) { 791 t.Fatalf("stat from: %v", err) 792 } 793 toFi, err := Stat(to) 794 if err != nil { 795 t.Fatalf("stat %q failed: %v", to, err) 796 } 797 if toFi.Size() != int64(len(fromData)) { 798 t.Errorf(`"to" size = %d; want %d (old "from" size)`, toFi.Size(), len(fromData)) 799 } 800 } 801 802 func TestRenameFailed(t *testing.T) { 803 defer chtmpdir(t)() 804 from, to := "renamefrom", "renameto" 805 // Ensure we are not testing the overwrite case here. 806 Remove(from) 807 Remove(to) 808 809 err := Rename(from, to) 810 switch err := err.(type) { 811 case *LinkError: 812 if err.Op != "rename" { 813 t.Errorf("rename %q, %q: err.Op: want %q, got %q", from, to, "rename", err.Op) 814 } 815 if err.Old != from { 816 t.Errorf("rename %q, %q: err.Old: want %q, got %q", from, to, from, err.Old) 817 } 818 if err.New != to { 819 t.Errorf("rename %q, %q: err.New: want %q, got %q", from, to, to, err.New) 820 } 821 case nil: 822 t.Errorf("rename %q, %q: expected error, got nil", from, to) 823 824 // cleanup whatever was placed in "renameto" 825 Remove(to) 826 default: 827 t.Errorf("rename %q, %q: expected %T, got %T %v", from, to, new(LinkError), err, err) 828 } 829 } 830 831 func exec(t *testing.T, dir, cmd string, args []string, expect string) { 832 r, w, err := Pipe() 833 if err != nil { 834 t.Fatalf("Pipe: %v", err) 835 } 836 defer r.Close() 837 attr := &ProcAttr{Dir: dir, Files: []*File{nil, w, Stderr}} 838 p, err := StartProcess(cmd, args, attr) 839 if err != nil { 840 t.Fatalf("StartProcess: %v", err) 841 } 842 w.Close() 843 844 var b bytes.Buffer 845 io.Copy(&b, r) 846 output := b.String() 847 848 fi1, _ := Stat(strings.TrimSpace(output)) 849 fi2, _ := Stat(expect) 850 if !SameFile(fi1, fi2) { 851 t.Errorf("exec %q returned %q wanted %q", 852 strings.Join(append([]string{cmd}, args...), " "), output, expect) 853 } 854 p.Wait() 855 } 856 857 func TestStartProcess(t *testing.T) { 858 testenv.MustHaveExec(t) 859 860 var dir, cmd string 861 var args []string 862 switch runtime.GOOS { 863 case "android": 864 t.Skip("android doesn't have /bin/pwd") 865 case "windows": 866 cmd = Getenv("COMSPEC") 867 dir = Getenv("SystemRoot") 868 args = []string{"/c", "cd"} 869 default: 870 cmd = "/bin/pwd" 871 dir = "/" 872 args = []string{} 873 } 874 cmddir, cmdbase := filepath.Split(cmd) 875 args = append([]string{cmdbase}, args...) 876 // Test absolute executable path. 877 exec(t, dir, cmd, args, dir) 878 // Test relative executable path. 879 exec(t, cmddir, cmdbase, args, cmddir) 880 } 881 882 func checkMode(t *testing.T, path string, mode FileMode) { 883 dir, err := Stat(path) 884 if err != nil { 885 t.Fatalf("Stat %q (looking for mode %#o): %s", path, mode, err) 886 } 887 if dir.Mode()&0777 != mode { 888 t.Errorf("Stat %q: mode %#o want %#o", path, dir.Mode(), mode) 889 } 890 } 891 892 func TestChmod(t *testing.T) { 893 // Chmod is not supported under windows. 894 if runtime.GOOS == "windows" { 895 return 896 } 897 f := newFile("TestChmod", t) 898 defer Remove(f.Name()) 899 defer f.Close() 900 901 if err := Chmod(f.Name(), 0456); err != nil { 902 t.Fatalf("chmod %s 0456: %s", f.Name(), err) 903 } 904 checkMode(t, f.Name(), 0456) 905 906 if err := f.Chmod(0123); err != nil { 907 t.Fatalf("chmod %s 0123: %s", f.Name(), err) 908 } 909 checkMode(t, f.Name(), 0123) 910 } 911 912 func checkSize(t *testing.T, f *File, size int64) { 913 dir, err := f.Stat() 914 if err != nil { 915 t.Fatalf("Stat %q (looking for size %d): %s", f.Name(), size, err) 916 } 917 if dir.Size() != size { 918 t.Errorf("Stat %q: size %d want %d", f.Name(), dir.Size(), size) 919 } 920 } 921 922 func TestFTruncate(t *testing.T) { 923 f := newFile("TestFTruncate", t) 924 defer Remove(f.Name()) 925 defer f.Close() 926 927 checkSize(t, f, 0) 928 f.Write([]byte("hello, world\n")) 929 checkSize(t, f, 13) 930 f.Truncate(10) 931 checkSize(t, f, 10) 932 f.Truncate(1024) 933 checkSize(t, f, 1024) 934 f.Truncate(0) 935 checkSize(t, f, 0) 936 _, err := f.Write([]byte("surprise!")) 937 if err == nil { 938 checkSize(t, f, 13+9) // wrote at offset past where hello, world was. 939 } 940 } 941 942 func TestTruncate(t *testing.T) { 943 f := newFile("TestTruncate", t) 944 defer Remove(f.Name()) 945 defer f.Close() 946 947 checkSize(t, f, 0) 948 f.Write([]byte("hello, world\n")) 949 checkSize(t, f, 13) 950 Truncate(f.Name(), 10) 951 checkSize(t, f, 10) 952 Truncate(f.Name(), 1024) 953 checkSize(t, f, 1024) 954 Truncate(f.Name(), 0) 955 checkSize(t, f, 0) 956 _, err := f.Write([]byte("surprise!")) 957 if err == nil { 958 checkSize(t, f, 13+9) // wrote at offset past where hello, world was. 959 } 960 } 961 962 // Use TempDir (via newFile) to make sure we're on a local file system, 963 // so that timings are not distorted by latency and caching. 964 // On NFS, timings can be off due to caching of meta-data on 965 // NFS servers (Issue 848). 966 func TestChtimes(t *testing.T) { 967 f := newFile("TestChtimes", t) 968 defer Remove(f.Name()) 969 970 f.Write([]byte("hello, world\n")) 971 f.Close() 972 973 testChtimes(t, f.Name()) 974 } 975 976 // Use TempDir (via newDir) to make sure we're on a local file system, 977 // so that timings are not distorted by latency and caching. 978 // On NFS, timings can be off due to caching of meta-data on 979 // NFS servers (Issue 848). 980 func TestChtimesDir(t *testing.T) { 981 name := newDir("TestChtimes", t) 982 defer RemoveAll(name) 983 984 testChtimes(t, name) 985 } 986 987 func testChtimes(t *testing.T, name string) { 988 st, err := Stat(name) 989 if err != nil { 990 t.Fatalf("Stat %s: %s", name, err) 991 } 992 preStat := st 993 994 // Move access and modification time back a second 995 at := Atime(preStat) 996 mt := preStat.ModTime() 997 err = Chtimes(name, at.Add(-time.Second), mt.Add(-time.Second)) 998 if err != nil { 999 t.Fatalf("Chtimes %s: %s", name, err) 1000 } 1001 1002 st, err = Stat(name) 1003 if err != nil { 1004 t.Fatalf("second Stat %s: %s", name, err) 1005 } 1006 postStat := st 1007 1008 /* Plan 9, NaCl: 1009 Mtime is the time of the last change of content. Similarly, atime is set whenever the 1010 contents are accessed; also, it is set whenever mtime is set. 1011 */ 1012 pat := Atime(postStat) 1013 pmt := postStat.ModTime() 1014 if !pat.Before(at) && runtime.GOOS != "plan9" && runtime.GOOS != "nacl" { 1015 t.Errorf("AccessTime didn't go backwards; was=%d, after=%d", at, pat) 1016 } 1017 1018 if !pmt.Before(mt) { 1019 t.Errorf("ModTime didn't go backwards; was=%d, after=%d", mt, pmt) 1020 } 1021 } 1022 1023 func TestChdirAndGetwd(t *testing.T) { 1024 // TODO(brainman): file.Chdir() is not implemented on windows. 1025 if runtime.GOOS == "windows" { 1026 return 1027 } 1028 fd, err := Open(".") 1029 if err != nil { 1030 t.Fatalf("Open .: %s", err) 1031 } 1032 // These are chosen carefully not to be symlinks on a Mac 1033 // (unlike, say, /var, /etc), except /tmp, which we handle below. 1034 dirs := []string{"/", "/usr/bin", "/tmp"} 1035 // /usr/bin does not usually exist on Plan 9 or Android. 1036 switch runtime.GOOS { 1037 case "android": 1038 dirs = []string{"/", "/system/bin"} 1039 case "plan9": 1040 dirs = []string{"/", "/usr"} 1041 case "darwin": 1042 switch runtime.GOARCH { 1043 case "arm", "arm64": 1044 d1, err := ioutil.TempDir("", "d1") 1045 if err != nil { 1046 t.Fatalf("TempDir: %v", err) 1047 } 1048 d2, err := ioutil.TempDir("", "d2") 1049 if err != nil { 1050 t.Fatalf("TempDir: %v", err) 1051 } 1052 dirs = []string{d1, d2} 1053 } 1054 } 1055 oldwd := Getenv("PWD") 1056 for mode := 0; mode < 2; mode++ { 1057 for _, d := range dirs { 1058 if mode == 0 { 1059 err = Chdir(d) 1060 } else { 1061 fd1, err := Open(d) 1062 if err != nil { 1063 t.Errorf("Open %s: %s", d, err) 1064 continue 1065 } 1066 err = fd1.Chdir() 1067 fd1.Close() 1068 } 1069 if d == "/tmp" { 1070 Setenv("PWD", "/tmp") 1071 } 1072 pwd, err1 := Getwd() 1073 Setenv("PWD", oldwd) 1074 err2 := fd.Chdir() 1075 if err2 != nil { 1076 // We changed the current directory and cannot go back. 1077 // Don't let the tests continue; they'll scribble 1078 // all over some other directory. 1079 fmt.Fprintf(Stderr, "fchdir back to dot failed: %s\n", err2) 1080 Exit(1) 1081 } 1082 if err != nil { 1083 fd.Close() 1084 t.Fatalf("Chdir %s: %s", d, err) 1085 } 1086 if err1 != nil { 1087 fd.Close() 1088 t.Fatalf("Getwd in %s: %s", d, err1) 1089 } 1090 if pwd != d { 1091 fd.Close() 1092 t.Fatalf("Getwd returned %q want %q", pwd, d) 1093 } 1094 } 1095 } 1096 fd.Close() 1097 } 1098 1099 // Test that Chdir+Getwd is program-wide. 1100 func TestProgWideChdir(t *testing.T) { 1101 const N = 10 1102 c := make(chan bool) 1103 cpwd := make(chan string) 1104 for i := 0; i < N; i++ { 1105 go func(i int) { 1106 // Lock half the goroutines in their own operating system 1107 // thread to exercise more scheduler possibilities. 1108 if i%2 == 1 { 1109 // On Plan 9, after calling LockOSThread, the goroutines 1110 // run on different processes which don't share the working 1111 // directory. This used to be an issue because Go expects 1112 // the working directory to be program-wide. 1113 // See issue 9428. 1114 runtime.LockOSThread() 1115 } 1116 <-c 1117 pwd, err := Getwd() 1118 if err != nil { 1119 t.Errorf("Getwd on goroutine %d: %v", i, err) 1120 return 1121 } 1122 cpwd <- pwd 1123 }(i) 1124 } 1125 oldwd, err := Getwd() 1126 if err != nil { 1127 t.Fatalf("Getwd: %v", err) 1128 } 1129 d, err := ioutil.TempDir("", "test") 1130 if err != nil { 1131 t.Fatalf("TempDir: %v", err) 1132 } 1133 defer func() { 1134 if err := Chdir(oldwd); err != nil { 1135 t.Fatalf("Chdir: %v", err) 1136 } 1137 RemoveAll(d) 1138 }() 1139 if err := Chdir(d); err != nil { 1140 t.Fatalf("Chdir: %v", err) 1141 } 1142 // OS X sets TMPDIR to a symbolic link. 1143 // So we resolve our working directory again before the test. 1144 d, err = Getwd() 1145 if err != nil { 1146 t.Fatalf("Getwd: %v", err) 1147 } 1148 close(c) 1149 for i := 0; i < N; i++ { 1150 pwd := <-cpwd 1151 if pwd != d { 1152 t.Errorf("Getwd returned %q; want %q", pwd, d) 1153 } 1154 } 1155 } 1156 1157 func TestSeek(t *testing.T) { 1158 f := newFile("TestSeek", t) 1159 defer Remove(f.Name()) 1160 defer f.Close() 1161 1162 const data = "hello, world\n" 1163 io.WriteString(f, data) 1164 1165 type test struct { 1166 in int64 1167 whence int 1168 out int64 1169 } 1170 var tests = []test{ 1171 {0, io.SeekCurrent, int64(len(data))}, 1172 {0, io.SeekStart, 0}, 1173 {5, io.SeekStart, 5}, 1174 {0, io.SeekEnd, int64(len(data))}, 1175 {0, io.SeekStart, 0}, 1176 {-1, io.SeekEnd, int64(len(data)) - 1}, 1177 {1 << 33, io.SeekStart, 1 << 33}, 1178 {1 << 33, io.SeekEnd, 1<<33 + int64(len(data))}, 1179 } 1180 for i, tt := range tests { 1181 off, err := f.Seek(tt.in, tt.whence) 1182 if off != tt.out || err != nil { 1183 if e, ok := err.(*PathError); ok && e.Err == syscall.EINVAL && tt.out > 1<<32 { 1184 // Reiserfs rejects the big seeks. 1185 // https://golang.org/issue/91 1186 break 1187 } 1188 t.Errorf("#%d: Seek(%v, %v) = %v, %v want %v, nil", i, tt.in, tt.whence, off, err, tt.out) 1189 } 1190 } 1191 } 1192 1193 type openErrorTest struct { 1194 path string 1195 mode int 1196 error error 1197 } 1198 1199 var openErrorTests = []openErrorTest{ 1200 { 1201 sfdir + "/no-such-file", 1202 O_RDONLY, 1203 syscall.ENOENT, 1204 }, 1205 { 1206 sfdir, 1207 O_WRONLY, 1208 syscall.EISDIR, 1209 }, 1210 { 1211 sfdir + "/" + sfname + "/no-such-file", 1212 O_WRONLY, 1213 syscall.ENOTDIR, 1214 }, 1215 } 1216 1217 func TestOpenError(t *testing.T) { 1218 for _, tt := range openErrorTests { 1219 f, err := OpenFile(tt.path, tt.mode, 0) 1220 if err == nil { 1221 t.Errorf("Open(%q, %d) succeeded", tt.path, tt.mode) 1222 f.Close() 1223 continue 1224 } 1225 perr, ok := err.(*PathError) 1226 if !ok { 1227 t.Errorf("Open(%q, %d) returns error of %T type; want *PathError", tt.path, tt.mode, err) 1228 } 1229 if perr.Err != tt.error { 1230 if runtime.GOOS == "plan9" { 1231 syscallErrStr := perr.Err.Error() 1232 expectedErrStr := strings.Replace(tt.error.Error(), "file ", "", 1) 1233 if !strings.HasSuffix(syscallErrStr, expectedErrStr) { 1234 // Some Plan 9 file servers incorrectly return 1235 // EACCES rather than EISDIR when a directory is 1236 // opened for write. 1237 if tt.error == syscall.EISDIR && strings.HasSuffix(syscallErrStr, syscall.EACCES.Error()) { 1238 continue 1239 } 1240 t.Errorf("Open(%q, %d) = _, %q; want suffix %q", tt.path, tt.mode, syscallErrStr, expectedErrStr) 1241 } 1242 continue 1243 } 1244 if runtime.GOOS == "dragonfly" { 1245 // DragonFly incorrectly returns EACCES rather 1246 // EISDIR when a directory is opened for write. 1247 if tt.error == syscall.EISDIR && perr.Err == syscall.EACCES { 1248 continue 1249 } 1250 } 1251 t.Errorf("Open(%q, %d) = _, %q; want %q", tt.path, tt.mode, perr.Err.Error(), tt.error.Error()) 1252 } 1253 } 1254 } 1255 1256 func TestOpenNoName(t *testing.T) { 1257 f, err := Open("") 1258 if err == nil { 1259 t.Fatal(`Open("") succeeded`) 1260 f.Close() 1261 } 1262 } 1263 1264 func runBinHostname(t *testing.T) string { 1265 // Run /bin/hostname and collect output. 1266 r, w, err := Pipe() 1267 if err != nil { 1268 t.Fatal(err) 1269 } 1270 defer r.Close() 1271 const path = "/bin/hostname" 1272 p, err := StartProcess(path, []string{"hostname"}, &ProcAttr{Files: []*File{nil, w, Stderr}}) 1273 if err != nil { 1274 if _, err := Stat(path); IsNotExist(err) { 1275 t.Skipf("skipping test; test requires %s but it does not exist", path) 1276 } 1277 t.Fatal(err) 1278 } 1279 w.Close() 1280 1281 var b bytes.Buffer 1282 io.Copy(&b, r) 1283 _, err = p.Wait() 1284 if err != nil { 1285 t.Fatalf("run hostname Wait: %v", err) 1286 } 1287 err = p.Kill() 1288 if err == nil { 1289 t.Errorf("expected an error from Kill running 'hostname'") 1290 } 1291 output := b.String() 1292 if n := len(output); n > 0 && output[n-1] == '\n' { 1293 output = output[0 : n-1] 1294 } 1295 if output == "" { 1296 t.Fatalf("/bin/hostname produced no output") 1297 } 1298 1299 return output 1300 } 1301 1302 func testWindowsHostname(t *testing.T) { 1303 hostname, err := Hostname() 1304 if err != nil { 1305 t.Fatal(err) 1306 } 1307 cmd := osexec.Command("hostname") 1308 out, err := cmd.CombinedOutput() 1309 if err != nil { 1310 t.Fatalf("Failed to execute hostname command: %v %s", err, out) 1311 } 1312 want := strings.Trim(string(out), "\r\n") 1313 if hostname != want { 1314 t.Fatalf("Hostname() = %q, want %q", hostname, want) 1315 } 1316 } 1317 1318 func TestHostname(t *testing.T) { 1319 // There is no other way to fetch hostname on windows, but via winapi. 1320 // On Plan 9 it can be taken from #c/sysname as Hostname() does. 1321 switch runtime.GOOS { 1322 case "android", "plan9": 1323 t.Skipf("%s doesn't have /bin/hostname", runtime.GOOS) 1324 case "windows": 1325 testWindowsHostname(t) 1326 return 1327 } 1328 1329 testenv.MustHaveExec(t) 1330 1331 // Check internal Hostname() against the output of /bin/hostname. 1332 // Allow that the internal Hostname returns a Fully Qualified Domain Name 1333 // and the /bin/hostname only returns the first component 1334 hostname, err := Hostname() 1335 if err != nil { 1336 t.Fatalf("%v", err) 1337 } 1338 want := runBinHostname(t) 1339 if hostname != want { 1340 i := strings.Index(hostname, ".") 1341 if i < 0 || hostname[0:i] != want { 1342 t.Errorf("Hostname() = %q, want %q", hostname, want) 1343 } 1344 } 1345 } 1346 1347 func TestReadAt(t *testing.T) { 1348 f := newFile("TestReadAt", t) 1349 defer Remove(f.Name()) 1350 defer f.Close() 1351 1352 const data = "hello, world\n" 1353 io.WriteString(f, data) 1354 1355 b := make([]byte, 5) 1356 n, err := f.ReadAt(b, 7) 1357 if err != nil || n != len(b) { 1358 t.Fatalf("ReadAt 7: %d, %v", n, err) 1359 } 1360 if string(b) != "world" { 1361 t.Fatalf("ReadAt 7: have %q want %q", string(b), "world") 1362 } 1363 } 1364 1365 // Verify that ReadAt doesn't affect seek offset. 1366 // In the Plan 9 kernel, there used to be a bug in the implementation of 1367 // the pread syscall, where the channel offset was erroneously updated after 1368 // calling pread on a file. 1369 func TestReadAtOffset(t *testing.T) { 1370 f := newFile("TestReadAtOffset", t) 1371 defer Remove(f.Name()) 1372 defer f.Close() 1373 1374 const data = "hello, world\n" 1375 io.WriteString(f, data) 1376 1377 f.Seek(0, 0) 1378 b := make([]byte, 5) 1379 1380 n, err := f.ReadAt(b, 7) 1381 if err != nil || n != len(b) { 1382 t.Fatalf("ReadAt 7: %d, %v", n, err) 1383 } 1384 if string(b) != "world" { 1385 t.Fatalf("ReadAt 7: have %q want %q", string(b), "world") 1386 } 1387 1388 n, err = f.Read(b) 1389 if err != nil || n != len(b) { 1390 t.Fatalf("Read: %d, %v", n, err) 1391 } 1392 if string(b) != "hello" { 1393 t.Fatalf("Read: have %q want %q", string(b), "hello") 1394 } 1395 } 1396 1397 func TestWriteAt(t *testing.T) { 1398 f := newFile("TestWriteAt", t) 1399 defer Remove(f.Name()) 1400 defer f.Close() 1401 1402 const data = "hello, world\n" 1403 io.WriteString(f, data) 1404 1405 n, err := f.WriteAt([]byte("WORLD"), 7) 1406 if err != nil || n != 5 { 1407 t.Fatalf("WriteAt 7: %d, %v", n, err) 1408 } 1409 1410 b, err := ioutil.ReadFile(f.Name()) 1411 if err != nil { 1412 t.Fatalf("ReadFile %s: %v", f.Name(), err) 1413 } 1414 if string(b) != "hello, WORLD\n" { 1415 t.Fatalf("after write: have %q want %q", string(b), "hello, WORLD\n") 1416 } 1417 } 1418 1419 func writeFile(t *testing.T, fname string, flag int, text string) string { 1420 f, err := OpenFile(fname, flag, 0666) 1421 if err != nil { 1422 t.Fatalf("Open: %v", err) 1423 } 1424 n, err := io.WriteString(f, text) 1425 if err != nil { 1426 t.Fatalf("WriteString: %d, %v", n, err) 1427 } 1428 f.Close() 1429 data, err := ioutil.ReadFile(fname) 1430 if err != nil { 1431 t.Fatalf("ReadFile: %v", err) 1432 } 1433 return string(data) 1434 } 1435 1436 func TestAppend(t *testing.T) { 1437 defer chtmpdir(t)() 1438 const f = "append.txt" 1439 defer Remove(f) 1440 s := writeFile(t, f, O_CREATE|O_TRUNC|O_RDWR, "new") 1441 if s != "new" { 1442 t.Fatalf("writeFile: have %q want %q", s, "new") 1443 } 1444 s = writeFile(t, f, O_APPEND|O_RDWR, "|append") 1445 if s != "new|append" { 1446 t.Fatalf("writeFile: have %q want %q", s, "new|append") 1447 } 1448 s = writeFile(t, f, O_CREATE|O_APPEND|O_RDWR, "|append") 1449 if s != "new|append|append" { 1450 t.Fatalf("writeFile: have %q want %q", s, "new|append|append") 1451 } 1452 err := Remove(f) 1453 if err != nil { 1454 t.Fatalf("Remove: %v", err) 1455 } 1456 s = writeFile(t, f, O_CREATE|O_APPEND|O_RDWR, "new&append") 1457 if s != "new&append" { 1458 t.Fatalf("writeFile: after append have %q want %q", s, "new&append") 1459 } 1460 s = writeFile(t, f, O_CREATE|O_RDWR, "old") 1461 if s != "old&append" { 1462 t.Fatalf("writeFile: after create have %q want %q", s, "old&append") 1463 } 1464 s = writeFile(t, f, O_CREATE|O_TRUNC|O_RDWR, "new") 1465 if s != "new" { 1466 t.Fatalf("writeFile: after truncate have %q want %q", s, "new") 1467 } 1468 } 1469 1470 func TestStatDirWithTrailingSlash(t *testing.T) { 1471 // Create new temporary directory and arrange to clean it up. 1472 path, err := ioutil.TempDir("", "/_TestStatDirWithSlash_") 1473 if err != nil { 1474 t.Fatalf("TempDir: %s", err) 1475 } 1476 defer RemoveAll(path) 1477 1478 // Stat of path should succeed. 1479 _, err = Stat(path) 1480 if err != nil { 1481 t.Fatalf("stat %s failed: %s", path, err) 1482 } 1483 1484 // Stat of path+"/" should succeed too. 1485 path += "/" 1486 _, err = Stat(path) 1487 if err != nil { 1488 t.Fatalf("stat %s failed: %s", path, err) 1489 } 1490 } 1491 1492 func TestNilProcessStateString(t *testing.T) { 1493 var ps *ProcessState 1494 s := ps.String() 1495 if s != "<nil>" { 1496 t.Errorf("(*ProcessState)(nil).String() = %q, want %q", s, "<nil>") 1497 } 1498 } 1499 1500 func TestSameFile(t *testing.T) { 1501 defer chtmpdir(t)() 1502 fa, err := Create("a") 1503 if err != nil { 1504 t.Fatalf("Create(a): %v", err) 1505 } 1506 defer Remove(fa.Name()) 1507 fa.Close() 1508 fb, err := Create("b") 1509 if err != nil { 1510 t.Fatalf("Create(b): %v", err) 1511 } 1512 defer Remove(fb.Name()) 1513 fb.Close() 1514 1515 ia1, err := Stat("a") 1516 if err != nil { 1517 t.Fatalf("Stat(a): %v", err) 1518 } 1519 ia2, err := Stat("a") 1520 if err != nil { 1521 t.Fatalf("Stat(a): %v", err) 1522 } 1523 if !SameFile(ia1, ia2) { 1524 t.Errorf("files should be same") 1525 } 1526 1527 ib, err := Stat("b") 1528 if err != nil { 1529 t.Fatalf("Stat(b): %v", err) 1530 } 1531 if SameFile(ia1, ib) { 1532 t.Errorf("files should be different") 1533 } 1534 } 1535 1536 func TestDevNullFile(t *testing.T) { 1537 f, err := Open(DevNull) 1538 if err != nil { 1539 t.Fatalf("Open(%s): %v", DevNull, err) 1540 } 1541 defer f.Close() 1542 fi, err := f.Stat() 1543 if err != nil { 1544 t.Fatalf("Stat(%s): %v", DevNull, err) 1545 } 1546 name := filepath.Base(DevNull) 1547 if fi.Name() != name { 1548 t.Fatalf("wrong file name have %v want %v", fi.Name(), name) 1549 } 1550 if fi.Size() != 0 { 1551 t.Fatalf("wrong file size have %d want 0", fi.Size()) 1552 } 1553 } 1554 1555 var testLargeWrite = flag.Bool("large_write", false, "run TestLargeWriteToConsole test that floods console with output") 1556 1557 func TestLargeWriteToConsole(t *testing.T) { 1558 if !*testLargeWrite { 1559 t.Skip("skipping console-flooding test; enable with -large_write") 1560 } 1561 b := make([]byte, 32000) 1562 for i := range b { 1563 b[i] = '.' 1564 } 1565 b[len(b)-1] = '\n' 1566 n, err := Stdout.Write(b) 1567 if err != nil { 1568 t.Fatalf("Write to os.Stdout failed: %v", err) 1569 } 1570 if n != len(b) { 1571 t.Errorf("Write to os.Stdout should return %d; got %d", len(b), n) 1572 } 1573 n, err = Stderr.Write(b) 1574 if err != nil { 1575 t.Fatalf("Write to os.Stderr failed: %v", err) 1576 } 1577 if n != len(b) { 1578 t.Errorf("Write to os.Stderr should return %d; got %d", len(b), n) 1579 } 1580 } 1581 1582 func TestStatDirModeExec(t *testing.T) { 1583 const mode = 0111 1584 1585 path, err := ioutil.TempDir("", "go-build") 1586 if err != nil { 1587 t.Fatalf("Failed to create temp directory: %v", err) 1588 } 1589 defer RemoveAll(path) 1590 1591 if err := Chmod(path, 0777); err != nil { 1592 t.Fatalf("Chmod %q 0777: %v", path, err) 1593 } 1594 1595 dir, err := Stat(path) 1596 if err != nil { 1597 t.Fatalf("Stat %q (looking for mode %#o): %s", path, mode, err) 1598 } 1599 if dir.Mode()&mode != mode { 1600 t.Errorf("Stat %q: mode %#o want %#o", path, dir.Mode()&mode, mode) 1601 } 1602 } 1603 1604 func TestStatStdin(t *testing.T) { 1605 switch runtime.GOOS { 1606 case "android", "plan9": 1607 t.Skipf("%s doesn't have /bin/sh", runtime.GOOS) 1608 } 1609 1610 testenv.MustHaveExec(t) 1611 1612 if Getenv("GO_WANT_HELPER_PROCESS") == "1" { 1613 st, err := Stdin.Stat() 1614 if err != nil { 1615 t.Fatalf("Stat failed: %v", err) 1616 } 1617 fmt.Println(st.Mode() & ModeNamedPipe) 1618 Exit(0) 1619 } 1620 1621 var cmd *osexec.Cmd 1622 if runtime.GOOS == "windows" { 1623 cmd = osexec.Command("cmd", "/c", "echo output | "+Args[0]+" -test.run=TestStatStdin") 1624 } else { 1625 cmd = osexec.Command("/bin/sh", "-c", "echo output | "+Args[0]+" -test.run=TestStatStdin") 1626 } 1627 cmd.Env = append(Environ(), "GO_WANT_HELPER_PROCESS=1") 1628 1629 output, err := cmd.CombinedOutput() 1630 if err != nil { 1631 t.Fatalf("Failed to spawn child process: %v %q", err, string(output)) 1632 } 1633 1634 // result will be like "prw-rw-rw" 1635 if len(output) < 1 || output[0] != 'p' { 1636 t.Fatalf("Child process reports stdin is not pipe '%v'", string(output)) 1637 } 1638 } 1639 1640 func TestReadAtEOF(t *testing.T) { 1641 f := newFile("TestReadAtEOF", t) 1642 defer Remove(f.Name()) 1643 defer f.Close() 1644 1645 _, err := f.ReadAt(make([]byte, 10), 0) 1646 switch err { 1647 case io.EOF: 1648 // all good 1649 case nil: 1650 t.Fatalf("ReadAt succeeded") 1651 default: 1652 t.Fatalf("ReadAt failed: %s", err) 1653 } 1654 } 1655 1656 func testKillProcess(t *testing.T, processKiller func(p *Process)) { 1657 testenv.MustHaveExec(t) 1658 1659 // Re-exec the test binary itself to emulate "sleep 1". 1660 cmd := osexec.Command(Args[0], "-test.run", "TestSleep") 1661 err := cmd.Start() 1662 if err != nil { 1663 t.Fatalf("Failed to start test process: %v", err) 1664 } 1665 go func() { 1666 time.Sleep(100 * time.Millisecond) 1667 processKiller(cmd.Process) 1668 }() 1669 err = cmd.Wait() 1670 if err == nil { 1671 t.Errorf("Test process succeeded, but expected to fail") 1672 } 1673 } 1674 1675 // TestSleep emulates "sleep 1". It is a helper for testKillProcess, so we 1676 // don't have to rely on an external "sleep" command being available. 1677 func TestSleep(t *testing.T) { 1678 if testing.Short() { 1679 t.Skip("Skipping in short mode") 1680 } 1681 time.Sleep(time.Second) 1682 } 1683 1684 func TestKillStartProcess(t *testing.T) { 1685 testKillProcess(t, func(p *Process) { 1686 err := p.Kill() 1687 if err != nil { 1688 t.Fatalf("Failed to kill test process: %v", err) 1689 } 1690 }) 1691 } 1692 1693 func TestGetppid(t *testing.T) { 1694 if runtime.GOOS == "plan9" { 1695 // TODO: golang.org/issue/8206 1696 t.Skipf("skipping test on plan9; see issue 8206") 1697 } 1698 1699 testenv.MustHaveExec(t) 1700 1701 if Getenv("GO_WANT_HELPER_PROCESS") == "1" { 1702 fmt.Print(Getppid()) 1703 Exit(0) 1704 } 1705 1706 cmd := osexec.Command(Args[0], "-test.run=TestGetppid") 1707 cmd.Env = append(Environ(), "GO_WANT_HELPER_PROCESS=1") 1708 1709 // verify that Getppid() from the forked process reports our process id 1710 output, err := cmd.CombinedOutput() 1711 if err != nil { 1712 t.Fatalf("Failed to spawn child process: %v %q", err, string(output)) 1713 } 1714 1715 childPpid := string(output) 1716 ourPid := fmt.Sprintf("%d", Getpid()) 1717 if childPpid != ourPid { 1718 t.Fatalf("Child process reports parent process id '%v', expected '%v'", childPpid, ourPid) 1719 } 1720 } 1721 1722 func TestKillFindProcess(t *testing.T) { 1723 testKillProcess(t, func(p *Process) { 1724 p2, err := FindProcess(p.Pid) 1725 if err != nil { 1726 t.Fatalf("Failed to find test process: %v", err) 1727 } 1728 err = p2.Kill() 1729 if err != nil { 1730 t.Fatalf("Failed to kill test process: %v", err) 1731 } 1732 }) 1733 } 1734 1735 var nilFileMethodTests = []struct { 1736 name string 1737 f func(*File) error 1738 }{ 1739 {"Chdir", func(f *File) error { return f.Chdir() }}, 1740 {"Close", func(f *File) error { return f.Close() }}, 1741 {"Chmod", func(f *File) error { return f.Chmod(0) }}, 1742 {"Chown", func(f *File) error { return f.Chown(0, 0) }}, 1743 {"Read", func(f *File) error { _, err := f.Read(make([]byte, 0)); return err }}, 1744 {"ReadAt", func(f *File) error { _, err := f.ReadAt(make([]byte, 0), 0); return err }}, 1745 {"Readdir", func(f *File) error { _, err := f.Readdir(1); return err }}, 1746 {"Readdirnames", func(f *File) error { _, err := f.Readdirnames(1); return err }}, 1747 {"Seek", func(f *File) error { _, err := f.Seek(0, io.SeekStart); return err }}, 1748 {"Stat", func(f *File) error { _, err := f.Stat(); return err }}, 1749 {"Sync", func(f *File) error { return f.Sync() }}, 1750 {"Truncate", func(f *File) error { return f.Truncate(0) }}, 1751 {"Write", func(f *File) error { _, err := f.Write(make([]byte, 0)); return err }}, 1752 {"WriteAt", func(f *File) error { _, err := f.WriteAt(make([]byte, 0), 0); return err }}, 1753 {"WriteString", func(f *File) error { _, err := f.WriteString(""); return err }}, 1754 } 1755 1756 // Test that all File methods give ErrInvalid if the receiver is nil. 1757 func TestNilFileMethods(t *testing.T) { 1758 for _, tt := range nilFileMethodTests { 1759 var file *File 1760 got := tt.f(file) 1761 if got != ErrInvalid { 1762 t.Errorf("%v should fail when f is nil; got %v", tt.name, got) 1763 } 1764 } 1765 } 1766 1767 func mkdirTree(t *testing.T, root string, level, max int) { 1768 if level >= max { 1769 return 1770 } 1771 level++ 1772 for i := 'a'; i < 'c'; i++ { 1773 dir := filepath.Join(root, string(i)) 1774 if err := Mkdir(dir, 0700); err != nil { 1775 t.Fatal(err) 1776 } 1777 mkdirTree(t, dir, level, max) 1778 } 1779 } 1780 1781 // Test that simultaneous RemoveAll do not report an error. 1782 // As long as it gets removed, we should be happy. 1783 func TestRemoveAllRace(t *testing.T) { 1784 if runtime.GOOS == "windows" { 1785 // Windows has very strict rules about things like 1786 // removing directories while someone else has 1787 // them open. The racing doesn't work out nicely 1788 // like it does on Unix. 1789 t.Skip("skipping on windows") 1790 } 1791 1792 n := runtime.GOMAXPROCS(16) 1793 defer runtime.GOMAXPROCS(n) 1794 root, err := ioutil.TempDir("", "issue") 1795 if err != nil { 1796 t.Fatal(err) 1797 } 1798 mkdirTree(t, root, 1, 6) 1799 hold := make(chan struct{}) 1800 var wg sync.WaitGroup 1801 for i := 0; i < 4; i++ { 1802 wg.Add(1) 1803 go func() { 1804 defer wg.Done() 1805 <-hold 1806 err := RemoveAll(root) 1807 if err != nil { 1808 t.Errorf("unexpected error: %T, %q", err, err) 1809 } 1810 }() 1811 } 1812 close(hold) // let workers race to remove root 1813 wg.Wait() 1814 }