mirror of https://github.com/k3s-io/k3s
138 lines
3.8 KiB
Go
138 lines
3.8 KiB
Go
// +build freebsd linux darwin
|
|
|
|
/*
|
|
Copyright 2017 The Kubernetes Authors.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package util
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"net"
|
|
"net/url"
|
|
"os"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
"golang.org/x/sys/unix"
|
|
"k8s.io/klog"
|
|
)
|
|
|
|
const (
|
|
// unixProtocol is the network protocol of unix socket.
|
|
unixProtocol = "unix"
|
|
)
|
|
|
|
// CreateListener creates a listener on the specified endpoint.
|
|
func CreateListener(endpoint string) (net.Listener, error) {
|
|
protocol, addr, err := parseEndpointWithFallbackProtocol(endpoint, unixProtocol)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if protocol != unixProtocol {
|
|
return nil, fmt.Errorf("only support unix socket endpoint")
|
|
}
|
|
|
|
// Unlink to cleanup the previous socket file.
|
|
err = unix.Unlink(addr)
|
|
if err != nil && !os.IsNotExist(err) {
|
|
return nil, fmt.Errorf("failed to unlink socket file %q: %v", addr, err)
|
|
}
|
|
|
|
if err := os.MkdirAll(filepath.Dir(addr), 0750); err != nil {
|
|
return nil, fmt.Errorf("error creating socket directory %q: %v", filepath.Dir(addr), err)
|
|
}
|
|
|
|
// Create the socket on a tempfile and move it to the destination socket to handle improprer cleanup
|
|
file, err := ioutil.TempFile(filepath.Dir(addr), "")
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to create temporary file: %v", err)
|
|
}
|
|
|
|
if err := os.Remove(file.Name()); err != nil {
|
|
return nil, fmt.Errorf("failed to remove temporary file: %v", err)
|
|
}
|
|
|
|
l, err := net.Listen(protocol, file.Name())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err = os.Rename(file.Name(), addr); err != nil {
|
|
return nil, fmt.Errorf("failed to move temporary file to addr %q: %v", addr, err)
|
|
}
|
|
|
|
return l, nil
|
|
}
|
|
|
|
// GetAddressAndDialer returns the address parsed from the given endpoint and a dialer.
|
|
func GetAddressAndDialer(endpoint string) (string, func(addr string, timeout time.Duration) (net.Conn, error), error) {
|
|
protocol, addr, err := parseEndpointWithFallbackProtocol(endpoint, unixProtocol)
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
if protocol != unixProtocol {
|
|
return "", nil, fmt.Errorf("only support unix socket endpoint")
|
|
}
|
|
|
|
return addr, dial, nil
|
|
}
|
|
|
|
func dial(addr string, timeout time.Duration) (net.Conn, error) {
|
|
return net.DialTimeout(unixProtocol, addr, timeout)
|
|
}
|
|
|
|
func parseEndpointWithFallbackProtocol(endpoint string, fallbackProtocol string) (protocol string, addr string, err error) {
|
|
if protocol, addr, err = parseEndpoint(endpoint); err != nil && protocol == "" {
|
|
fallbackEndpoint := fallbackProtocol + "://" + endpoint
|
|
protocol, addr, err = parseEndpoint(fallbackEndpoint)
|
|
if err == nil {
|
|
klog.Warningf("Using %q as endpoint is deprecated, please consider using full url format %q.", endpoint, fallbackEndpoint)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func parseEndpoint(endpoint string) (string, string, error) {
|
|
u, err := url.Parse(endpoint)
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
|
|
switch u.Scheme {
|
|
case "tcp":
|
|
return "tcp", u.Host, nil
|
|
|
|
case "unix":
|
|
return "unix", u.Path, nil
|
|
|
|
case "":
|
|
return "", "", fmt.Errorf("Using %q as endpoint is deprecated, please consider using full url format", endpoint)
|
|
|
|
default:
|
|
return u.Scheme, "", fmt.Errorf("protocol %q not supported", u.Scheme)
|
|
}
|
|
}
|
|
|
|
// LocalEndpoint returns the full path to a unix socket at the given endpoint
|
|
func LocalEndpoint(path, file string) (string, error) {
|
|
u := url.URL{
|
|
Scheme: unixProtocol,
|
|
Path: path,
|
|
}
|
|
return filepath.Join(u.String(), file+".sock"), nil
|
|
}
|