mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-06 19:12:19 +00:00
231e468e19
git-subtree-dir: erigon-lib git-subtree-mainline:3c8cbda809
git-subtree-split:93d9c9d9fe
307 lines
8.4 KiB
Go
307 lines
8.4 KiB
Go
// Copyright 2010 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package downloader
|
|
|
|
import (
|
|
"strings"
|
|
"syscall"
|
|
)
|
|
|
|
func isSlash(c uint8) bool {
|
|
return c == '\\' || c == '/'
|
|
}
|
|
|
|
func toUpper(c byte) byte {
|
|
if 'a' <= c && c <= 'z' {
|
|
return c - ('a' - 'A')
|
|
}
|
|
return c
|
|
}
|
|
|
|
// isReservedName reports if name is a Windows reserved device name or a console handle.
|
|
// It does not detect names with an extension, which are also reserved on some Windows versions.
|
|
//
|
|
// For details, search for PRN in
|
|
// https://docs.microsoft.com/en-us/windows/desktop/fileio/naming-a-file.
|
|
func isReservedName(name string) bool {
|
|
if 3 <= len(name) && len(name) <= 4 {
|
|
switch string([]byte{toUpper(name[0]), toUpper(name[1]), toUpper(name[2])}) {
|
|
case "CON", "PRN", "AUX", "NUL":
|
|
return len(name) == 3
|
|
case "COM", "LPT":
|
|
return len(name) == 4 && '1' <= name[3] && name[3] <= '9'
|
|
}
|
|
}
|
|
// Passing CONIN$ or CONOUT$ to CreateFile opens a console handle.
|
|
// https://learn.microsoft.com/en-us/windows/win32/api/fileapi/nf-fileapi-createfilea#consoles
|
|
//
|
|
// While CONIN$ and CONOUT$ aren't documented as being files,
|
|
// they behave the same as CON. For example, ./CONIN$ also opens the console input.
|
|
if len(name) == 6 && name[5] == '$' && strings.EqualFold(name, "CONIN$") {
|
|
return true
|
|
}
|
|
if len(name) == 7 && name[6] == '$' && strings.EqualFold(name, "CONOUT$") {
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func isLocal(path string) bool {
|
|
if path == "" {
|
|
return false
|
|
}
|
|
if isSlash(path[0]) {
|
|
// Path rooted in the current drive.
|
|
return false
|
|
}
|
|
if strings.IndexByte(path, ':') >= 0 {
|
|
// Colons are only valid when marking a drive letter ("C:foo").
|
|
// Rejecting any path with a colon is conservative but safe.
|
|
return false
|
|
}
|
|
hasDots := false // contains . or .. path elements
|
|
for p := path; p != ""; {
|
|
var part string
|
|
part, p, _ = cutPath(p)
|
|
if part == "." || part == ".." {
|
|
hasDots = true
|
|
}
|
|
// Trim the extension and look for a reserved name.
|
|
base, _, hasExt := strings.Cut(part, ".")
|
|
if isReservedName(base) {
|
|
if !hasExt {
|
|
return false
|
|
}
|
|
// The path element is a reserved name with an extension. Some Windows
|
|
// versions consider this a reserved name, while others do not. Use
|
|
// FullPath to see if the name is reserved.
|
|
//
|
|
// FullPath will convert references to reserved device names to their
|
|
// canonical form: \\.\${DEVICE_NAME}
|
|
//
|
|
// FullPath does not perform this conversion for paths which contain
|
|
// a reserved device name anywhere other than in the last element,
|
|
// so check the part rather than the full path.
|
|
if p, _ := syscall.FullPath(part); len(p) >= 4 && p[:4] == `\\.\` {
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
if hasDots {
|
|
path = Clean(path)
|
|
}
|
|
if path == ".." || strings.HasPrefix(path, `..\`) {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
// IsAbs reports whether the path is absolute.
|
|
func IsAbs(path string) (b bool) {
|
|
l := volumeNameLen(path)
|
|
if l == 0 {
|
|
return false
|
|
}
|
|
// If the volume name starts with a double slash, this is an absolute path.
|
|
if isSlash(path[0]) && isSlash(path[1]) {
|
|
return true
|
|
}
|
|
path = path[l:]
|
|
if path == "" {
|
|
return false
|
|
}
|
|
return isSlash(path[0])
|
|
}
|
|
|
|
// volumeNameLen returns length of the leading volume name on Windows.
|
|
// It returns 0 elsewhere.
|
|
//
|
|
// See: https://learn.microsoft.com/en-us/dotnet/standard/io/file-path-formats
|
|
func volumeNameLen(path string) int {
|
|
if len(path) < 2 {
|
|
return 0
|
|
}
|
|
// with drive letter
|
|
c := path[0]
|
|
if path[1] == ':' && ('a' <= c && c <= 'z' || 'A' <= c && c <= 'Z') {
|
|
return 2
|
|
}
|
|
// UNC and DOS device paths start with two slashes.
|
|
if !isSlash(path[0]) || !isSlash(path[1]) {
|
|
return 0
|
|
}
|
|
rest := path[2:]
|
|
p1, rest, _ := cutPath(rest)
|
|
p2, rest, ok := cutPath(rest)
|
|
if !ok {
|
|
return len(path)
|
|
}
|
|
if p1 != "." && p1 != "?" {
|
|
// This is a UNC path: \\${HOST}\${SHARE}\
|
|
return len(path) - len(rest) - 1
|
|
}
|
|
// This is a DOS device path.
|
|
if len(p2) == 3 && toUpper(p2[0]) == 'U' && toUpper(p2[1]) == 'N' && toUpper(p2[2]) == 'C' {
|
|
// This is a DOS device path that links to a UNC: \\.\UNC\${HOST}\${SHARE}\
|
|
_, rest, _ = cutPath(rest) // host
|
|
_, rest, ok = cutPath(rest) // share
|
|
if !ok {
|
|
return len(path)
|
|
}
|
|
}
|
|
return len(path) - len(rest) - 1
|
|
}
|
|
|
|
// cutPath slices path around the first path separator.
|
|
func cutPath(path string) (before, after string, found bool) {
|
|
for i := range path {
|
|
if isSlash(path[i]) {
|
|
return path[:i], path[i+1:], true
|
|
}
|
|
}
|
|
return path, "", false
|
|
}
|
|
|
|
// HasPrefix exists for historical compatibility and should not be used.
|
|
//
|
|
// Deprecated: HasPrefix does not respect path boundaries and
|
|
// does not ignore case when required.
|
|
func HasPrefix(p, prefix string) bool {
|
|
if strings.HasPrefix(p, prefix) {
|
|
return true
|
|
}
|
|
return strings.HasPrefix(strings.ToLower(p), strings.ToLower(prefix))
|
|
}
|
|
|
|
func splitList(path string) []string {
|
|
// The same implementation is used in LookPath in os/exec;
|
|
// consider changing os/exec when changing this.
|
|
|
|
if path == "" {
|
|
return []string{}
|
|
}
|
|
|
|
// Split path, respecting but preserving quotes.
|
|
list := []string{}
|
|
start := 0
|
|
quo := false
|
|
for i := 0; i < len(path); i++ {
|
|
switch c := path[i]; {
|
|
case c == '"':
|
|
quo = !quo
|
|
case c == ListSeparator && !quo:
|
|
list = append(list, path[start:i])
|
|
start = i + 1
|
|
}
|
|
}
|
|
list = append(list, path[start:])
|
|
|
|
// Remove quotes.
|
|
for i, s := range list {
|
|
list[i] = strings.ReplaceAll(s, `"`, ``)
|
|
}
|
|
|
|
return list
|
|
}
|
|
|
|
func abs(path string) (string, error) {
|
|
if path == "" {
|
|
// syscall.FullPath returns an error on empty path, because it's not a valid path.
|
|
// To implement Abs behavior of returning working directory on empty string input,
|
|
// special-case empty path by changing it to "." path. See golang.org/issue/24441.
|
|
path = "."
|
|
}
|
|
fullPath, err := syscall.FullPath(path)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return Clean(fullPath), nil
|
|
}
|
|
|
|
func join(elem []string) string {
|
|
var b strings.Builder
|
|
var lastChar byte
|
|
for _, e := range elem {
|
|
switch {
|
|
case b.Len() == 0:
|
|
// Add the first non-empty path element unchanged.
|
|
case isSlash(lastChar):
|
|
// If the path ends in a slash, strip any leading slashes from the next
|
|
// path element to avoid creating a UNC path (any path starting with "\\")
|
|
// from non-UNC elements.
|
|
//
|
|
// The correct behavior for Join when the first element is an incomplete UNC
|
|
// path (for example, "\\") is underspecified. We currently join subsequent
|
|
// elements so Join("\\", "host", "share") produces "\\host\share".
|
|
for len(e) > 0 && isSlash(e[0]) {
|
|
e = e[1:]
|
|
}
|
|
case lastChar == ':':
|
|
// If the path ends in a colon, keep the path relative to the current directory
|
|
// on a drive and don't add a separator. Preserve leading slashes in the next
|
|
// path element, which may make the path absolute.
|
|
//
|
|
// Join(`C:`, `f`) = `C:f`
|
|
// Join(`C:`, `\f`) = `C:\f`
|
|
default:
|
|
// In all other cases, add a separator between elements.
|
|
b.WriteByte('\\')
|
|
lastChar = '\\'
|
|
}
|
|
if len(e) > 0 {
|
|
b.WriteString(e)
|
|
lastChar = e[len(e)-1]
|
|
}
|
|
}
|
|
if b.Len() == 0 {
|
|
return ""
|
|
}
|
|
return Clean(b.String())
|
|
}
|
|
|
|
// joinNonEmpty is like join, but it assumes that the first element is non-empty.
|
|
func joinNonEmpty(elem []string) string {
|
|
if len(elem[0]) == 2 && elem[0][1] == ':' {
|
|
// First element is drive letter without terminating slash.
|
|
// Keep path relative to current directory on that drive.
|
|
// Skip empty elements.
|
|
i := 1
|
|
for ; i < len(elem); i++ {
|
|
if elem[i] != "" {
|
|
break
|
|
}
|
|
}
|
|
return Clean(elem[0] + strings.Join(elem[i:], string(Separator)))
|
|
}
|
|
// The following logic prevents Join from inadvertently creating a
|
|
// UNC path on Windows. Unless the first element is a UNC path, Join
|
|
// shouldn't create a UNC path. See golang.org/issue/9167.
|
|
p := Clean(strings.Join(elem, string(Separator)))
|
|
if !isUNC(p) {
|
|
return p
|
|
}
|
|
// p == UNC only allowed when the first element is a UNC path.
|
|
head := Clean(elem[0])
|
|
if isUNC(head) {
|
|
return p
|
|
}
|
|
// head + tail == UNC, but joining two non-UNC paths should not result
|
|
// in a UNC path. Undo creation of UNC path.
|
|
tail := Clean(strings.Join(elem[1:], string(Separator)))
|
|
if head[len(head)-1] == Separator {
|
|
return head + tail
|
|
}
|
|
return head + string(Separator) + tail
|
|
}
|
|
|
|
// isUNC reports whether path is a UNC path.
|
|
func isUNC(path string) bool {
|
|
return len(path) > 1 && isSlash(path[0]) && isSlash(path[1])
|
|
}
|
|
|
|
func sameWord(a, b string) bool {
|
|
return strings.EqualFold(a, b)
|
|
}
|