-
Notifications
You must be signed in to change notification settings - Fork 68
/
scaffoling_test.go
184 lines (156 loc) · 5.37 KB
/
scaffoling_test.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
package godirwalk
import (
"flag"
"fmt"
"io/ioutil"
"os"
"path/filepath"
"testing"
)
// maxName is the tested maximum length of a filename this library will
// handle. Previous attempts to set it to one less than the size of
// syscall.Dirent.Name array resulted in runtime errors trying to create
// a test scaffolding file whose size exceeded 255 bytes. This filename
// is 255 characters long.
const maxName = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"
// scaffolingRoot is the temporary directory root for scaffold directory.
var scaffolingRoot string
func TestMain(m *testing.M) {
flag.Parse()
var code int // program exit code
// All tests use the same directory test scaffolding. Create the directory
// hierarchy, run the tests, then remove the root directory of the test
// scaffolding.
defer func() {
if err := teardown(); err != nil {
fmt.Fprintf(os.Stderr, "godirwalk teardown: %s\n", err)
code = 1
}
os.Exit(code)
}()
// When cannot complete setup, dump the directory so we see what we have,
// then bail.
if err := setup(); err != nil {
fmt.Fprintf(os.Stderr, "godirwalk setup: %s\n", err)
dumpDirectory()
code = 1
return
}
code = m.Run()
// When any test was a failure, then use standard library to walk test
// scaffolding directory and print its contents.
if code != 0 {
dumpDirectory()
}
}
func setup() error {
var err error
scaffolingRoot, err = ioutil.TempDir(os.TempDir(), "godirwalk-")
if err != nil {
return err
}
entries := []Creater{
file{"d0/" + maxName},
file{"d0/f0"}, // will be deleted after symlink for it created
file{"d0/f1"}, //
file{"d0/d1/f2"}, //
file{"d0/skips/d2/f3"}, // node precedes skip
file{"d0/skips/d2/skip"}, // skip is non-directory
file{"d0/skips/d2/z1"}, // node follows skip non-directory: should never be visited
file{"d0/skips/d3/f4"}, // node precedes skip
file{"d0/skips/d3/skip/f5"}, // skip is directory: this node should never be visited
file{"d0/skips/d3/z2"}, // node follows skip directory: should be visited
link{"d0/symlinks/nothing", "../f0"}, // referent will be deleted
link{"d0/symlinks/toF1", "../f1"}, //
link{"d0/symlinks/toD1", "../d1"}, //
link{"d0/symlinks/d4/toSD1", "../toD1"}, // chained symbolic links
link{"d0/symlinks/d4/toSF1", "../toF1"}, // chained symbolic links
}
for _, entry := range entries {
if err := entry.Create(); err != nil {
return fmt.Errorf("cannot create scaffolding entry: %s", err)
}
}
oldname, err := filepath.Abs(filepath.Join(scaffolingRoot, "d0/f1"))
if err != nil {
return fmt.Errorf("cannot create scaffolding entry: %s", err)
}
if err := (link{"d0/symlinks/toAbs", oldname}).Create(); err != nil {
return fmt.Errorf("cannot create scaffolding entry: %s", err)
}
if err := os.Remove(filepath.Join(scaffolingRoot, "d0/f0")); err != nil {
return fmt.Errorf("cannot remove file from test scaffolding: %s", err)
}
return nil
}
func teardown() error {
if scaffolingRoot == "" {
return nil // if we do not even have a test root directory then exit
}
if err := os.RemoveAll(scaffolingRoot); err != nil {
return err
}
return nil
}
func dumpDirectory() {
trim := len(scaffolingRoot) // trim rootDir from prefix of strings
err := filepath.Walk(scaffolingRoot, func(osPathname string, info os.FileInfo, err error) error {
if err != nil {
// we have no info, so get it
info, err2 := os.Lstat(osPathname)
if err2 != nil {
fmt.Fprintf(os.Stderr, "?--------- %s: %s\n", osPathname[trim:], err2)
} else {
fmt.Fprintf(os.Stderr, "%s %s: %s\n", info.Mode(), osPathname[trim:], err)
}
return nil
}
var suffix string
if info.Mode()&os.ModeSymlink != 0 {
referent, err := os.Readlink(osPathname)
if err != nil {
suffix = fmt.Sprintf(": cannot read symlink: %s", err)
err = nil
} else {
suffix = fmt.Sprintf(" -> %s", referent)
}
}
fmt.Fprintf(os.Stderr, "%s %s%s\n", info.Mode(), osPathname[trim:], suffix)
return nil
})
if err != nil {
fmt.Fprintf(os.Stderr, "cannot walk test directory: %s\n", err)
}
}
////////////////////////////////////////
// helpers to create file system entries for test scaffolding
type Creater interface {
Create() error
}
type file struct {
name string
}
func (f file) Create() error {
newname := filepath.Join(scaffolingRoot, filepath.FromSlash(f.name))
if err := os.MkdirAll(filepath.Dir(newname), os.ModePerm); err != nil {
return fmt.Errorf("cannot create directory for test scaffolding: %s", err)
}
if err := ioutil.WriteFile(newname, []byte(newname+"\n"), os.ModePerm); err != nil {
return fmt.Errorf("cannot create file for test scaffolding: %s", err)
}
return nil
}
type link struct {
name, referent string
}
func (s link) Create() error {
newname := filepath.Join(scaffolingRoot, filepath.FromSlash(s.name))
if err := os.MkdirAll(filepath.Dir(newname), os.ModePerm); err != nil {
return fmt.Errorf("cannot create directory for test scaffolding: %s", err)
}
oldname := filepath.FromSlash(s.referent)
if err := os.Symlink(oldname, newname); err != nil {
return fmt.Errorf("cannot create symbolic link for test scaffolding: %s", err)
}
return nil
}