blob: 499772a00dc9440c0cbdf208fc88e28115eee278 [file] [log] [blame]
Abhay Kumar40252eb2025-10-13 13:25:53 +00001// Copyright (c) 2016-2022 Uber Technologies, Inc.
Scott Baker2c1c4822019-10-16 11:02:41 -07002//
3// Permission is hereby granted, free of charge, to any person obtaining a copy
4// of this software and associated documentation files (the "Software"), to deal
5// in the Software without restriction, including without limitation the rights
6// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
7// copies of the Software, and to permit persons to whom the Software is
8// furnished to do so, subject to the following conditions:
9//
10// The above copyright notice and this permission notice shall be included in
11// all copies or substantial portions of the Software.
12//
13// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
18// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
19// THE SOFTWARE.
20
21package zap
22
23import (
24 "errors"
25 "fmt"
26 "io"
27 "net/url"
28 "os"
Abhay Kumar40252eb2025-10-13 13:25:53 +000029 "path/filepath"
Scott Baker2c1c4822019-10-16 11:02:41 -070030 "strings"
31 "sync"
32
33 "go.uber.org/zap/zapcore"
34)
35
36const schemeFile = "file"
37
Abhay Kumar40252eb2025-10-13 13:25:53 +000038var _sinkRegistry = newSinkRegistry()
Scott Baker2c1c4822019-10-16 11:02:41 -070039
40// Sink defines the interface to write to and close logger destinations.
41type Sink interface {
42 zapcore.WriteSyncer
43 io.Closer
44}
45
Scott Baker2c1c4822019-10-16 11:02:41 -070046type errSinkNotFound struct {
47 scheme string
48}
49
50func (e *errSinkNotFound) Error() string {
51 return fmt.Sprintf("no sink found for scheme %q", e.scheme)
52}
53
Abhay Kumar40252eb2025-10-13 13:25:53 +000054type nopCloserSink struct{ zapcore.WriteSyncer }
55
56func (nopCloserSink) Close() error { return nil }
57
58type sinkRegistry struct {
59 mu sync.Mutex
60 factories map[string]func(*url.URL) (Sink, error) // keyed by scheme
61 openFile func(string, int, os.FileMode) (*os.File, error) // type matches os.OpenFile
62}
63
64func newSinkRegistry() *sinkRegistry {
65 sr := &sinkRegistry{
66 factories: make(map[string]func(*url.URL) (Sink, error)),
67 openFile: os.OpenFile,
68 }
69 // Infallible operation: the registry is empty, so we can't have a conflict.
70 _ = sr.RegisterSink(schemeFile, sr.newFileSinkFromURL)
71 return sr
72}
73
74// RegisterScheme registers the given factory for the specific scheme.
75func (sr *sinkRegistry) RegisterSink(scheme string, factory func(*url.URL) (Sink, error)) error {
76 sr.mu.Lock()
77 defer sr.mu.Unlock()
Scott Baker2c1c4822019-10-16 11:02:41 -070078
79 if scheme == "" {
80 return errors.New("can't register a sink factory for empty string")
81 }
82 normalized, err := normalizeScheme(scheme)
83 if err != nil {
84 return fmt.Errorf("%q is not a valid scheme: %v", scheme, err)
85 }
Abhay Kumar40252eb2025-10-13 13:25:53 +000086 if _, ok := sr.factories[normalized]; ok {
Scott Baker2c1c4822019-10-16 11:02:41 -070087 return fmt.Errorf("sink factory already registered for scheme %q", normalized)
88 }
Abhay Kumar40252eb2025-10-13 13:25:53 +000089 sr.factories[normalized] = factory
Scott Baker2c1c4822019-10-16 11:02:41 -070090 return nil
91}
92
Abhay Kumar40252eb2025-10-13 13:25:53 +000093func (sr *sinkRegistry) newSink(rawURL string) (Sink, error) {
94 // URL parsing doesn't work well for Windows paths such as `c:\log.txt`, as scheme is set to
95 // the drive, and path is unset unless `c:/log.txt` is used.
96 // To avoid Windows-specific URL handling, we instead check IsAbs to open as a file.
97 // filepath.IsAbs is OS-specific, so IsAbs('c:/log.txt') is false outside of Windows.
98 if filepath.IsAbs(rawURL) {
99 return sr.newFileSinkFromPath(rawURL)
100 }
101
Scott Baker2c1c4822019-10-16 11:02:41 -0700102 u, err := url.Parse(rawURL)
103 if err != nil {
104 return nil, fmt.Errorf("can't parse %q as a URL: %v", rawURL, err)
105 }
106 if u.Scheme == "" {
107 u.Scheme = schemeFile
108 }
109
Abhay Kumar40252eb2025-10-13 13:25:53 +0000110 sr.mu.Lock()
111 factory, ok := sr.factories[u.Scheme]
112 sr.mu.Unlock()
Scott Baker2c1c4822019-10-16 11:02:41 -0700113 if !ok {
114 return nil, &errSinkNotFound{u.Scheme}
115 }
116 return factory(u)
117}
118
Abhay Kumar40252eb2025-10-13 13:25:53 +0000119// RegisterSink registers a user-supplied factory for all sinks with a
120// particular scheme.
121//
122// All schemes must be ASCII, valid under section 0.1 of RFC 3986
123// (https://tools.ietf.org/html/rfc3983#section-3.1), and must not already
124// have a factory registered. Zap automatically registers a factory for the
125// "file" scheme.
126func RegisterSink(scheme string, factory func(*url.URL) (Sink, error)) error {
127 return _sinkRegistry.RegisterSink(scheme, factory)
128}
129
130func (sr *sinkRegistry) newFileSinkFromURL(u *url.URL) (Sink, error) {
Scott Baker2c1c4822019-10-16 11:02:41 -0700131 if u.User != nil {
132 return nil, fmt.Errorf("user and password not allowed with file URLs: got %v", u)
133 }
134 if u.Fragment != "" {
135 return nil, fmt.Errorf("fragments not allowed with file URLs: got %v", u)
136 }
137 if u.RawQuery != "" {
138 return nil, fmt.Errorf("query parameters not allowed with file URLs: got %v", u)
139 }
140 // Error messages are better if we check hostname and port separately.
141 if u.Port() != "" {
142 return nil, fmt.Errorf("ports not allowed with file URLs: got %v", u)
143 }
144 if hn := u.Hostname(); hn != "" && hn != "localhost" {
145 return nil, fmt.Errorf("file URLs must leave host empty or use localhost: got %v", u)
146 }
Abhay Kumar40252eb2025-10-13 13:25:53 +0000147
148 return sr.newFileSinkFromPath(u.Path)
149}
150
151func (sr *sinkRegistry) newFileSinkFromPath(path string) (Sink, error) {
152 switch path {
Scott Baker2c1c4822019-10-16 11:02:41 -0700153 case "stdout":
154 return nopCloserSink{os.Stdout}, nil
155 case "stderr":
156 return nopCloserSink{os.Stderr}, nil
157 }
Abhay Kumar40252eb2025-10-13 13:25:53 +0000158 return sr.openFile(path, os.O_WRONLY|os.O_APPEND|os.O_CREATE, 0o666)
Scott Baker2c1c4822019-10-16 11:02:41 -0700159}
160
161func normalizeScheme(s string) (string, error) {
162 // https://tools.ietf.org/html/rfc3986#section-3.1
163 s = strings.ToLower(s)
164 if first := s[0]; 'a' > first || 'z' < first {
165 return "", errors.New("must start with a letter")
166 }
167 for i := 1; i < len(s); i++ { // iterate over bytes, not runes
168 c := s[i]
169 switch {
170 case 'a' <= c && c <= 'z':
171 continue
172 case '0' <= c && c <= '9':
173 continue
174 case c == '.' || c == '+' || c == '-':
175 continue
176 }
177 return "", fmt.Errorf("may not contain %q", c)
178 }
179 return s, nil
180}