Skip to content

Commit 19a5d8a

Browse files
prashantvabhinav
andauthored
Open: expose error cause, test for fs.ErrNotExist (#1149)
This is a prefactor for Windows support (#621). TestOpen currently relies on a hardcoded error "no such file or directory" if a file is not found. However, this error message is OS-specific. We can now rely on `errors.Is(err, fs.ErrNotExist)` if we wrap errors using %w instead of %v. Co-authored-by: Abhinav Gupta <[email protected]>
1 parent bdd673d commit 19a5d8a

File tree

7 files changed

+139
-41
lines changed

7 files changed

+139
-41
lines changed

benchmarks/go.sum

+4-1
Original file line numberDiff line numberDiff line change
@@ -60,11 +60,14 @@ github.com/smartystreets/assertions v1.0.0/go.mod h1:kHHU4qYBaI3q23Pp3VPrmWhuIUr
6060
github.com/smartystreets/go-aws-auth v0.0.0-20180515143844-0c1422d1fdb9/go.mod h1:SnhjPscd9TpLiy1LpzGSKh3bXCfxxXuqd9xmQJy3slM=
6161
github.com/smartystreets/gunit v1.0.0/go.mod h1:qwPWnhz6pn0NnRBP++URONOVyNkPyr4SauJk4cUOwJs=
6262
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
63+
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
6364
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
6465
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
6566
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
66-
github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY=
6767
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
68+
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
69+
github.com/stretchr/testify v1.8.0 h1:pSgiaMZlXftHpm5L7V1+rVB+AZJydKsMxsQBIJw4PKk=
70+
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
6871
github.com/tj/assert v0.0.0-20171129193455-018094318fb0/go.mod h1:mZ9/Rh9oLWpLLDRpvE+3b7gP/C2YyLFYxNmcLnPTMe0=
6972
github.com/tj/assert v0.0.3 h1:Df/BlaZ20mq6kuai7f5z2TvPFiwC3xaWJSDQNiIS3Rk=
7073
github.com/tj/assert v0.0.3/go.mod h1:Ne6X72Q+TB1AteidzQncjw9PabbMp4PBMZ1k+vd1Pvk=

go.mod

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ go 1.18
55
require (
66
github.com/benbjohnson/clock v1.1.0
77
github.com/pkg/errors v0.8.1
8-
github.com/stretchr/testify v1.7.0
8+
github.com/stretchr/testify v1.8.0
99
go.uber.org/atomic v1.7.0
1010
go.uber.org/goleak v1.1.11
1111
go.uber.org/multierr v1.6.0

go.sum

+4-1
Original file line numberDiff line numberDiff line change
@@ -13,9 +13,12 @@ github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINE
1313
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
1414
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
1515
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
16+
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
1617
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
17-
github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY=
1818
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
19+
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
20+
github.com/stretchr/testify v1.8.0 h1:pSgiaMZlXftHpm5L7V1+rVB+AZJydKsMxsQBIJw4PKk=
21+
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
1922
github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k=
2023
go.uber.org/atomic v1.7.0 h1:ADUqmZGgLDDfbSL9ZmPxKTybcoEYHgpYfELNoN+7hsw=
2124
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=

writer.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
// Copyright (c) 2016 Uber Technologies, Inc.
1+
// Copyright (c) 2016-2022 Uber Technologies, Inc.
22
//
33
// Permission is hereby granted, free of charge, to any person obtaining a copy
44
// of this software and associated documentation files (the "Software"), to deal
@@ -70,7 +70,7 @@ func open(paths []string) ([]zapcore.WriteSyncer, func(), error) {
7070
for _, path := range paths {
7171
sink, err := newSink(path)
7272
if err != nil {
73-
openErr = multierr.Append(openErr, fmt.Errorf("couldn't open sink %q: %v", path, err))
73+
openErr = multierr.Append(openErr, fmt.Errorf("open sink %q: %w", path, err))
7474
continue
7575
}
7676
writers = append(writers, sink)

writer_test.go

+123-34
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
// Copyright (c) 2016 Uber Technologies, Inc.
1+
// Copyright (c) 2016-2022 Uber Technologies, Inc.
22
//
33
// Permission is hereby granted, free of charge, to any person obtaining a copy
44
// of this software and associated documentation files (the "Software"), to deal
@@ -23,14 +23,15 @@ package zap
2323
import (
2424
"errors"
2525
"io"
26+
"io/fs"
2627
"net/url"
2728
"os"
2829
"path/filepath"
29-
"strings"
3030
"testing"
3131

3232
"github.com/stretchr/testify/assert"
3333
"github.com/stretchr/testify/require"
34+
"go.uber.org/multierr"
3435
"go.uber.org/zap/zapcore"
3536
)
3637

@@ -50,55 +51,95 @@ func TestOpenNoPaths(t *testing.T) {
5051
func TestOpen(t *testing.T) {
5152
tempName := filepath.Join(t.TempDir(), "test.log")
5253
assert.False(t, fileExists(tempName))
53-
require.True(t, strings.HasPrefix(tempName, "/"), "Expected absolute temp file path.")
54+
require.True(t, filepath.IsAbs(tempName), "Expected absolute temp file path.")
5455

5556
tests := []struct {
57+
msg string
5658
paths []string
57-
errs []string
5859
}{
59-
{[]string{"stdout"}, nil},
60-
{[]string{"stderr"}, nil},
61-
{[]string{tempName}, nil},
62-
{[]string{"file://" + tempName}, nil},
63-
{[]string{"file://localhost" + tempName}, nil},
64-
{[]string{"/foo/bar/baz"}, []string{"open /foo/bar/baz: no such file or directory"}},
65-
{[]string{"file://localhost/foo/bar/baz"}, []string{"open /foo/bar/baz: no such file or directory"}},
6660
{
67-
paths: []string{"stdout", "/foo/bar/baz", tempName, "file:///baz/quux"},
68-
errs: []string{
69-
"open /foo/bar/baz: no such file or directory",
70-
"open /baz/quux: no such file or directory",
71-
},
61+
msg: "stdout",
62+
paths: []string{"stdout"},
63+
},
64+
{
65+
msg: "stderr",
66+
paths: []string{"stderr"},
67+
},
68+
{
69+
msg: "temp file path only",
70+
paths: []string{tempName},
71+
},
72+
{
73+
msg: "temp file file scheme",
74+
paths: []string{"file://" + tempName},
75+
},
76+
{
77+
msg: "temp file with file scheme and host localhost",
78+
paths: []string{"file://localhost" + tempName},
7279
},
73-
{[]string{"file:///stderr"}, []string{"open /stderr:"}},
74-
{[]string{"file:///stdout"}, []string{"open /stdout:"}},
75-
{[]string{"file://host01.test.com" + tempName}, []string{"empty or use localhost"}},
76-
{[]string{"file://rms@localhost" + tempName}, []string{"user and password not allowed"}},
77-
{[]string{"file://localhost" + tempName + "#foo"}, []string{"fragments not allowed"}},
78-
{[]string{"file://localhost" + tempName + "?foo=bar"}, []string{"query parameters not allowed"}},
79-
{[]string{"file://localhost:8080" + tempName}, []string{"ports not allowed"}},
8080
}
8181

8282
for _, tt := range tests {
83-
_, cleanup, err := Open(tt.paths...)
84-
if err == nil {
85-
defer cleanup()
86-
}
83+
t.Run(tt.msg, func(t *testing.T) {
84+
_, cleanup, err := Open(tt.paths...)
85+
if err == nil {
86+
defer cleanup()
87+
}
8788

88-
if len(tt.errs) == 0 {
8989
assert.NoError(t, err, "Unexpected error opening paths %v.", tt.paths)
90-
} else {
91-
msg := err.Error()
92-
for _, expect := range tt.errs {
93-
assert.Contains(t, msg, expect, "Unexpected error opening paths %v.", tt.paths)
94-
}
95-
}
90+
})
9691
}
9792

9893
assert.True(t, fileExists(tempName))
9994
os.Remove(tempName)
10095
}
10196

97+
func TestOpenPathsNotFound(t *testing.T) {
98+
tempName := filepath.Join(t.TempDir(), "test.log")
99+
100+
tests := []struct {
101+
msg string
102+
paths []string
103+
wantNotFoundPaths []string
104+
}{
105+
{
106+
msg: "missing path",
107+
paths: []string{"/foo/bar/baz"},
108+
wantNotFoundPaths: []string{"/foo/bar/baz"},
109+
},
110+
{
111+
msg: "missing file scheme url with host localhost",
112+
paths: []string{"file://localhost/foo/bar/baz"},
113+
wantNotFoundPaths: []string{"/foo/bar/baz"},
114+
},
115+
{
116+
msg: "multiple paths",
117+
paths: []string{"stdout", "/foo/bar/baz", tempName, "file:///baz/quux"},
118+
wantNotFoundPaths: []string{
119+
"/foo/bar/baz",
120+
"/baz/quux",
121+
},
122+
},
123+
}
124+
125+
for _, tt := range tests {
126+
t.Run(tt.msg, func(t *testing.T) {
127+
_, cleanup, err := Open(tt.paths...)
128+
if !assert.Error(t, err, "Open must fail.") {
129+
cleanup()
130+
return
131+
}
132+
133+
errs := multierr.Errors(err)
134+
require.Len(t, errs, len(tt.wantNotFoundPaths))
135+
for i, err := range errs {
136+
assert.ErrorIs(t, err, fs.ErrNotExist)
137+
assert.ErrorContains(t, err, tt.wantNotFoundPaths[i], "missing path in error")
138+
}
139+
})
140+
}
141+
}
142+
102143
func TestOpenRelativePath(t *testing.T) {
103144
const name = "test-relative-path.txt"
104145

@@ -136,6 +177,54 @@ func TestOpenFails(t *testing.T) {
136177
}
137178
}
138179

180+
func TestOpenOtherErrors(t *testing.T) {
181+
tempName := filepath.Join(t.TempDir(), "test.log")
182+
183+
tests := []struct {
184+
msg string
185+
paths []string
186+
wantErr string
187+
}{
188+
{
189+
msg: "file with unexpected host",
190+
paths: []string{"file://host01.test.com" + tempName},
191+
wantErr: "empty or use localhost",
192+
},
193+
{
194+
msg: "file with user on localhost",
195+
paths: []string{"file://rms@localhost" + tempName},
196+
wantErr: "user and password not allowed",
197+
},
198+
{
199+
msg: "file url with fragment",
200+
paths: []string{"file://localhost" + tempName + "#foo"},
201+
wantErr: "fragments not allowed",
202+
},
203+
{
204+
msg: "file url with query",
205+
paths: []string{"file://localhost" + tempName + "?foo=bar"},
206+
wantErr: "query parameters not allowed",
207+
},
208+
{
209+
msg: "file with port",
210+
paths: []string{"file://localhost:8080" + tempName},
211+
wantErr: "ports not allowed",
212+
},
213+
}
214+
215+
for _, tt := range tests {
216+
t.Run(tt.msg, func(t *testing.T) {
217+
_, cleanup, err := Open(tt.paths...)
218+
if !assert.Error(t, err, "Open must fail.") {
219+
cleanup()
220+
return
221+
}
222+
223+
assert.ErrorContains(t, err, tt.wantErr, "Unexpected error opening paths %v.", tt.paths)
224+
})
225+
}
226+
}
227+
139228
type testWriter struct {
140229
expected string
141230
t testing.TB

zapgrpc/internal/test/go.mod

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ module go.uber.org/zap/zapgrpc/internal/test
33
go 1.17
44

55
require (
6-
github.com/stretchr/testify v1.7.0
6+
github.com/stretchr/testify v1.8.0
77
go.uber.org/zap v1.16.0
88
google.golang.org/grpc v1.42.0
99
)

zapgrpc/internal/test/go.sum

+4-1
Original file line numberDiff line numberDiff line change
@@ -65,10 +65,13 @@ github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTE
6565
github.com/rogpeppe/go-internal v1.8.0 h1:FCbCCtXNOY3UtUuHUYaghJg4y7Fd14rXifAYUAtL9R8=
6666
github.com/rogpeppe/go-internal v1.8.0/go.mod h1:WmiCO8CzOY8rg0OYDC4/i/2WRWAB6poM+XZ2dLUbcbE=
6767
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
68+
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
6869
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
6970
github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA=
70-
github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY=
7171
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
72+
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
73+
github.com/stretchr/testify v1.8.0 h1:pSgiaMZlXftHpm5L7V1+rVB+AZJydKsMxsQBIJw4PKk=
74+
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
7275
github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k=
7376
go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI=
7477
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=

0 commit comments

Comments
 (0)