在Go中获取终端大小

问题描述 投票:18回答:4

如何使用Golang获得tty大小?我正在尝试执行stty size命令,但我无法正确编写代码。

package main

import (
  "os/exec"
  "fmt"
  "log"
)

func main() {
  out, err := exec.Command("stty", "size").Output()
  fmt.Printf("out: %#v\n", out)
  fmt.Printf("err: %#v\n", err)
  if err != nil {
    log.Fatal(err)
  }
}

输出:

out: []byte{}
err: &exec.ExitError{ProcessState:(*os.ProcessState)(0xc200066520)}
2013/05/16 02:35:57 exit status 1
exit status 1

我认为这是因为Go产生了一个与当前tty无关的进程,它正在使用它。如何将命令与当前终端关联以获得其大小?

go tty
4个回答
17
投票

如果您让子进程访问父级的stdin,它会起作用:

package main

import (
  "os/exec"
  "fmt"
  "log"
  "os"
)

func main() {
  cmd := exec.Command("stty", "size")
  cmd.Stdin = os.Stdin
  out, err := cmd.Output()
  fmt.Printf("out: %#v\n", string(out))
  fmt.Printf("err: %#v\n", err)
  if err != nil {
    log.Fatal(err)
  }
}

产量:

out: "36 118\n"
err: <nil>

21
投票

我遇到了类似的问题。这就是我最终的结果。

它不使用子进程,因此在某些情况下可能需要。

import (
    "syscall"
    "unsafe"
)

type winsize struct {
    Row    uint16
    Col    uint16
    Xpixel uint16
    Ypixel uint16
}

func getWidth() uint {
    ws := &winsize{}
    retCode, _, errno := syscall.Syscall(syscall.SYS_IOCTL,
        uintptr(syscall.Stdin),
        uintptr(syscall.TIOCGWINSZ),
        uintptr(unsafe.Pointer(ws)))

    if int(retCode) == -1 {
        panic(errno)
    }
    return uint(ws.Col)
}

12
投票

我刚才想添加一个新答案,因为我最近遇到了这个问题。有一个终端包,住在官方ssh包https://godoc.org/golang.org/x/crypto/ssh/terminal内。

该软件包提供了一种轻松获取终端大小的方法。

width, height, err := terminal.GetSize(0)

0将是您想要的大小的终端的文件描述符。要获得fd或您当前的终端,您可以随时做int(os.Stdin.Fd())

在封面下,它使用系统调用来获取给定fd的终端大小。


5
投票

如果有人有兴趣我做了一个包,让这更容易。

https://github.com/wayneashleyberry/terminal-dimensions

package main

import (
    "fmt"

    terminal "github.com/wayneashleyberry/terminal-dimensions"
)

func main() {
    x, _ := terminal.Width()
    y, _ := terminal.Height()
    fmt.Printf("Terminal is %d wide and %d high", x, y)
}
© www.soinside.com 2019 - 2024. All rights reserved.