You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
161 lines
4.5 KiB
161 lines
4.5 KiB
// Copyright (c) 2016 Uber Technologies, Inc. |
|
// |
|
// Permission is hereby granted, free of charge, to any person obtaining a copy |
|
// of this software and associated documentation files (the "Software"), to deal |
|
// in the Software without restriction, including without limitation the rights |
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
|
// copies of the Software, and to permit persons to whom the Software is |
|
// furnished to do so, subject to the following conditions: |
|
// |
|
// The above copyright notice and this permission notice shall be included in |
|
// all copies or substantial portions of the Software. |
|
// |
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN |
|
// THE SOFTWARE. |
|
|
|
package zap |
|
|
|
import ( |
|
"errors" |
|
"fmt" |
|
"io" |
|
"net/url" |
|
"os" |
|
"strings" |
|
"sync" |
|
|
|
"go.uber.org/zap/zapcore" |
|
) |
|
|
|
const schemeFile = "file" |
|
|
|
var ( |
|
_sinkMutex sync.RWMutex |
|
_sinkFactories map[string]func(*url.URL) (Sink, error) // keyed by scheme |
|
) |
|
|
|
func init() { |
|
resetSinkRegistry() |
|
} |
|
|
|
func resetSinkRegistry() { |
|
_sinkMutex.Lock() |
|
defer _sinkMutex.Unlock() |
|
|
|
_sinkFactories = map[string]func(*url.URL) (Sink, error){ |
|
schemeFile: newFileSink, |
|
} |
|
} |
|
|
|
// Sink defines the interface to write to and close logger destinations. |
|
type Sink interface { |
|
zapcore.WriteSyncer |
|
io.Closer |
|
} |
|
|
|
type nopCloserSink struct{ zapcore.WriteSyncer } |
|
|
|
func (nopCloserSink) Close() error { return nil } |
|
|
|
type errSinkNotFound struct { |
|
scheme string |
|
} |
|
|
|
func (e *errSinkNotFound) Error() string { |
|
return fmt.Sprintf("no sink found for scheme %q", e.scheme) |
|
} |
|
|
|
// RegisterSink registers a user-supplied factory for all sinks with a |
|
// particular scheme. |
|
// |
|
// All schemes must be ASCII, valid under section 3.1 of RFC 3986 |
|
// (https://tools.ietf.org/html/rfc3986#section-3.1), and must not already |
|
// have a factory registered. Zap automatically registers a factory for the |
|
// "file" scheme. |
|
func RegisterSink(scheme string, factory func(*url.URL) (Sink, error)) error { |
|
_sinkMutex.Lock() |
|
defer _sinkMutex.Unlock() |
|
|
|
if scheme == "" { |
|
return errors.New("can't register a sink factory for empty string") |
|
} |
|
normalized, err := normalizeScheme(scheme) |
|
if err != nil { |
|
return fmt.Errorf("%q is not a valid scheme: %v", scheme, err) |
|
} |
|
if _, ok := _sinkFactories[normalized]; ok { |
|
return fmt.Errorf("sink factory already registered for scheme %q", normalized) |
|
} |
|
_sinkFactories[normalized] = factory |
|
return nil |
|
} |
|
|
|
func newSink(rawURL string) (Sink, error) { |
|
u, err := url.Parse(rawURL) |
|
if err != nil { |
|
return nil, fmt.Errorf("can't parse %q as a URL: %v", rawURL, err) |
|
} |
|
if u.Scheme == "" { |
|
u.Scheme = schemeFile |
|
} |
|
|
|
_sinkMutex.RLock() |
|
factory, ok := _sinkFactories[u.Scheme] |
|
_sinkMutex.RUnlock() |
|
if !ok { |
|
return nil, &errSinkNotFound{u.Scheme} |
|
} |
|
return factory(u) |
|
} |
|
|
|
func newFileSink(u *url.URL) (Sink, error) { |
|
if u.User != nil { |
|
return nil, fmt.Errorf("user and password not allowed with file URLs: got %v", u) |
|
} |
|
if u.Fragment != "" { |
|
return nil, fmt.Errorf("fragments not allowed with file URLs: got %v", u) |
|
} |
|
if u.RawQuery != "" { |
|
return nil, fmt.Errorf("query parameters not allowed with file URLs: got %v", u) |
|
} |
|
// Error messages are better if we check hostname and port separately. |
|
if u.Port() != "" { |
|
return nil, fmt.Errorf("ports not allowed with file URLs: got %v", u) |
|
} |
|
if hn := u.Hostname(); hn != "" && hn != "localhost" { |
|
return nil, fmt.Errorf("file URLs must leave host empty or use localhost: got %v", u) |
|
} |
|
switch u.Path { |
|
case "stdout": |
|
return nopCloserSink{os.Stdout}, nil |
|
case "stderr": |
|
return nopCloserSink{os.Stderr}, nil |
|
} |
|
return os.OpenFile(u.Path, os.O_WRONLY|os.O_APPEND|os.O_CREATE, 0666) |
|
} |
|
|
|
func normalizeScheme(s string) (string, error) { |
|
// https://tools.ietf.org/html/rfc3986#section-3.1 |
|
s = strings.ToLower(s) |
|
if first := s[0]; 'a' > first || 'z' < first { |
|
return "", errors.New("must start with a letter") |
|
} |
|
for i := 1; i < len(s); i++ { // iterate over bytes, not runes |
|
c := s[i] |
|
switch { |
|
case 'a' <= c && c <= 'z': |
|
continue |
|
case '0' <= c && c <= '9': |
|
continue |
|
case c == '.' || c == '+' || c == '-': |
|
continue |
|
} |
|
return "", fmt.Errorf("may not contain %q", c) |
|
} |
|
return s, nil |
|
}
|
|
|