Problem:
In Go gibt der setns-Aufruf EINVAL zurück (ungültiges Argument) beim Versuch, den Mount-Namespace für einen Container einzugeben.
Erklärung:
Um einen Namespace mit setns einzugeben, muss der Aufruf zuvor aus einem Single-Threaded-Kontext erfolgen Die Go-Laufzeitthreads werden gestartet.
Lösung:
Es gibt zwei Ansätze zur Behebung dieses Problems:
1. Konstruktor-Trick verwenden:
<code class="go">/* #include <sched.h> #include <stdio.h> #include <fcntl.h> __attribute__((constructor)) void enter_namespace(void) { setns(open("/proc/<PID>/ns/mnt", O_RDONLY, 0644), 0); } */ import "C"</code>
2. Mit syscall.RawSyscall:
<code class="go">package main import ( "fmt" "os" "path/filepath" "syscall" "runtime" ) func main() { if syscall.Geteuid() != 0 { fmt.Println("abort: you want to run this as root") os.Exit(1) } if len(os.Args) != 2 { fmt.Println("abort: you must provide a PID as the sole argument") os.Exit(2) } // Lock the main thread to the OS thread runtime.LockOSThread() namespaces := []string{"ipc", "uts", "net", "pid", "mnt"} for i := range namespaces { fd, _ := syscall.Open(filepath.Join("/proc", os.Args[1], "ns", namespaces[i]), syscall.O_RDONLY, 0644) err, _, msg := syscall.RawSyscall(308, uintptr(fd), 0, 0) // 308 == setns if err != 0 { fmt.Println("setns on", namespaces[i], "namespace failed:", msg) } else { fmt.Println("setns on", namespaces[i], "namespace succeeded") } } }</code>
Hinweis:
Das obige ist der detaillierte Inhalt vonWie behebe ich den Fehler „EINVAL' bei der Eingabe des Mount-Namespace in Go?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!