2019-01-12 04:58:27 +00:00
|
|
|
// +build windows
|
|
|
|
|
|
|
|
/*
|
|
|
|
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"
|
|
|
|
"net"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
2019-01-22 20:53:35 +00:00
|
|
|
"syscall"
|
2019-01-12 04:58:27 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/Microsoft/go-winio"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
tcpProtocol = "tcp"
|
|
|
|
npipeProtocol = "npipe"
|
|
|
|
)
|
|
|
|
|
2019-08-30 18:33:25 +00:00
|
|
|
// CreateListener creates a listener on the specified endpoint.
|
2019-01-12 04:58:27 +00:00
|
|
|
func CreateListener(endpoint string) (net.Listener, error) {
|
|
|
|
protocol, addr, err := parseEndpoint(endpoint)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
switch protocol {
|
|
|
|
case tcpProtocol:
|
|
|
|
return net.Listen(tcpProtocol, addr)
|
|
|
|
|
|
|
|
case npipeProtocol:
|
|
|
|
return winio.ListenPipe(addr, nil)
|
|
|
|
|
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("only support tcp and npipe endpoint")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-30 18:33:25 +00:00
|
|
|
// GetAddressAndDialer returns the address parsed from the given endpoint and a dialer.
|
2019-01-12 04:58:27 +00:00
|
|
|
func GetAddressAndDialer(endpoint string) (string, func(addr string, timeout time.Duration) (net.Conn, error), error) {
|
|
|
|
protocol, addr, err := parseEndpoint(endpoint)
|
|
|
|
if err != nil {
|
|
|
|
return "", nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if protocol == tcpProtocol {
|
|
|
|
return addr, tcpDial, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if protocol == npipeProtocol {
|
|
|
|
return addr, npipeDial, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return "", nil, fmt.Errorf("only support tcp and npipe endpoint")
|
|
|
|
}
|
|
|
|
|
|
|
|
func tcpDial(addr string, timeout time.Duration) (net.Conn, error) {
|
|
|
|
return net.DialTimeout(tcpProtocol, addr, timeout)
|
|
|
|
}
|
|
|
|
|
|
|
|
func npipeDial(addr string, timeout time.Duration) (net.Conn, error) {
|
|
|
|
return winio.DialPipe(addr, &timeout)
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseEndpoint(endpoint string) (string, string, error) {
|
|
|
|
// url.Parse doesn't recognize \, so replace with / first.
|
|
|
|
endpoint = strings.Replace(endpoint, "\\", "/", -1)
|
|
|
|
u, err := url.Parse(endpoint)
|
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
if u.Scheme == "tcp" {
|
|
|
|
return "tcp", u.Host, nil
|
|
|
|
} else if u.Scheme == "npipe" {
|
|
|
|
if strings.HasPrefix(u.Path, "//./pipe") {
|
|
|
|
return "npipe", u.Path, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// fallback host if not provided.
|
|
|
|
host := u.Host
|
|
|
|
if host == "" {
|
|
|
|
host = "."
|
|
|
|
}
|
|
|
|
return "npipe", fmt.Sprintf("//%s%s", host, u.Path), nil
|
|
|
|
} else if u.Scheme == "" {
|
|
|
|
return "", "", fmt.Errorf("Using %q as endpoint is deprecated, please consider using full url format", endpoint)
|
|
|
|
} else {
|
|
|
|
return u.Scheme, "", fmt.Errorf("protocol %q not supported", u.Scheme)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-30 18:33:25 +00:00
|
|
|
// LocalEndpoint empty implementation
|
|
|
|
func LocalEndpoint(path, file string) (string, error) {
|
|
|
|
return "", fmt.Errorf("LocalEndpoints are unsupported in this build")
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
2019-01-22 20:53:35 +00:00
|
|
|
|
|
|
|
var tickCount = syscall.NewLazyDLL("kernel32.dll").NewProc("GetTickCount64")
|
|
|
|
|
|
|
|
// GetBootTime returns the time at which the machine was started, truncated to the nearest second
|
|
|
|
func GetBootTime() (time.Time, error) {
|
|
|
|
currentTime := time.Now()
|
|
|
|
output, _, err := tickCount.Call()
|
|
|
|
if errno, ok := err.(syscall.Errno); !ok || errno != 0 {
|
|
|
|
return time.Time{}, err
|
|
|
|
}
|
|
|
|
return currentTime.Add(-time.Duration(output) * time.Millisecond).Truncate(time.Second), nil
|
|
|
|
}
|
2019-09-27 21:51:53 +00:00
|
|
|
|
|
|
|
// IsUnixDomainSocket returns whether a given file is a AF_UNIX socket file
|
|
|
|
func IsUnixDomainSocket(filePath string) (bool, error) {
|
|
|
|
// Due to the absence of golang support for os.ModeSocket in Windows (https://github.com/golang/go/issues/33357)
|
|
|
|
// we need to dial the file and check if we receive an error to determine if a file is Unix Domain Socket file.
|
|
|
|
|
|
|
|
// Note that querrying for the Reparse Points (https://docs.microsoft.com/en-us/windows/win32/fileio/reparse-points)
|
|
|
|
// for the file (using FSCTL_GET_REPARSE_POINT) and checking for reparse tag: reparseTagSocket
|
|
|
|
// does NOT work in 1809 if the socket file is created within a bind mounted directory by a container
|
|
|
|
// and the FSCTL is issued in the host by the kubelet.
|
|
|
|
|
|
|
|
c, err := net.Dial("unix", filePath)
|
|
|
|
if err == nil {
|
|
|
|
c.Close()
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NormalizePath converts FS paths returned by certain go frameworks (like fsnotify)
|
|
|
|
// to native Windows paths that can be passed to Windows specific code
|
|
|
|
func NormalizePath(path string) string {
|
|
|
|
path = strings.ReplaceAll(path, "/", "\\")
|
|
|
|
if strings.HasPrefix(path, "\\") {
|
|
|
|
path = "c:" + path
|
|
|
|
}
|
|
|
|
return path
|
|
|
|
}
|