helpers/ssh/stub_ssh_server_windows.go (67 lines of code) (raw):
//go:build windows
package ssh
import (
"fmt"
"os/exec"
"syscall"
"unsafe"
"golang.org/x/sys/windows"
)
func runCmd(cmd *exec.Cmd) error {
cmd.SysProcAttr = &syscall.SysProcAttr{
CreationFlags: syscall.CREATE_NEW_PROCESS_GROUP,
}
jobObject, err := createJobObject()
if err != nil {
return err
}
cmd.Cancel = func() error {
return windows.CloseHandle(windows.Handle(jobObject))
}
if err := cmd.Start(); err != nil {
return err
}
if err := assignProcessToJobObject(cmd.Process.Pid, jobObject); err != nil {
return fmt.Errorf("assigning process to job object: %w", err)
}
return cmd.Wait()
}
func createJobObject() (uintptr, error) {
jobObj, err := windows.CreateJobObject(nil, nil)
if err != nil {
return 0, fmt.Errorf("creating job object: %w", err)
}
info := windows.JOBOBJECT_EXTENDED_LIMIT_INFORMATION{
BasicLimitInformation: windows.JOBOBJECT_BASIC_LIMIT_INFORMATION{
LimitFlags: windows.JOB_OBJECT_LIMIT_KILL_ON_JOB_CLOSE,
},
}
if _, err = windows.SetInformationJobObject(
jobObj,
windows.JobObjectExtendedLimitInformation,
uintptr(unsafe.Pointer(&info)),
uint32(unsafe.Sizeof(info))); err != nil {
return 0, fmt.Errorf("setting job object information: %w", err)
}
return uintptr(jobObj), nil
}
func assignProcessToJobObject(pid int, jobObject uintptr) error {
procHandle, err := findProcessHandleFromPID(pid)
if err != nil {
return fmt.Errorf("failed to retrieve handle for process: %w", err)
}
if err = windows.AssignProcessToJobObject(windows.Handle(jobObject), windows.Handle(procHandle)); err != nil {
return fmt.Errorf("failed to assign process to job: %w", err)
}
return nil
}
func findProcessHandleFromPID(pid int) (uintptr, error) {
const da = windows.PROCESS_TERMINATE | windows.PROCESS_SET_QUOTA
h, err := syscall.OpenProcess(da, false, uint32(pid))
if err != nil {
return 0, fmt.Errorf("calling OpenProcess: %w", err)
}
if uintptr(h) == 0 {
return 0, fmt.Errorf("getting process handle for pid %q", pid)
}
return uintptr(h), nil
}