mirror of
https://github.com/yusing/godoxy.git
synced 2026-03-20 00:25:02 +01:00
Added new Proxmox journalctl endpoint `/journalctl/:node/:vmid` for viewing all journalctl output without requiring a service name. Made the service parameter optional across both endpoints. Introduced configurable `limit` query parameter (1-1000, default 100) to both proxmox journalctl and docker logs APIs, replacing hardcoded 100-line tail. Added container status check in LXCCommand to prevent command execution on stopped containers, returning a clear status message instead. Refactored route validation to use pre-fetched IPs and improved References() method for proxmox routes with better alias handling.
142 lines
3.8 KiB
Go
142 lines
3.8 KiB
Go
package proxmox
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
|
|
"github.com/gorilla/websocket"
|
|
)
|
|
|
|
var ErrNoSession = fmt.Errorf("no session found, make sure username and password are set")
|
|
|
|
// LXCCommand connects to the Proxmox VNC websocket and streams command output.
|
|
// It returns an io.ReadCloser that streams the command output.
|
|
func (n *Node) LXCCommand(ctx context.Context, vmid int, command string) (io.ReadCloser, error) {
|
|
if !n.client.HasSession() {
|
|
return nil, ErrNoSession
|
|
}
|
|
|
|
node, err := n.client.Node(ctx, n.name)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get node: %w", err)
|
|
}
|
|
|
|
lxc, err := node.Container(ctx, vmid)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get container: %w", err)
|
|
}
|
|
|
|
if lxc.Status != "running" {
|
|
return io.NopCloser(bytes.NewReader(fmt.Appendf(nil, "container %d is not running, status: %s\n", vmid, lxc.Status))), nil
|
|
}
|
|
|
|
term, err := node.TermProxy(ctx)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get term proxy: %w", err)
|
|
}
|
|
|
|
send, recv, errs, close, err := node.TermWebSocket(term)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to connect to term websocket: %w", err)
|
|
}
|
|
|
|
handleSend := func(data []byte) error {
|
|
select {
|
|
case <-ctx.Done():
|
|
return ctx.Err()
|
|
case send <- data:
|
|
return nil
|
|
case err := <-errs:
|
|
return fmt.Errorf("failed to send: %w", err)
|
|
}
|
|
}
|
|
|
|
// Send command: `pct exec <vmid> -- <command>`
|
|
cmd := fmt.Appendf(nil, "pct exec %d -- %s\n", vmid, command)
|
|
if err := handleSend(cmd); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Create a pipe to stream the websocket messages
|
|
pr, pw := io.Pipe()
|
|
|
|
// Command line without trailing newline for matching in output
|
|
cmdLine := cmd[:len(cmd)-1]
|
|
|
|
// Start a goroutine to read from websocket and write to pipe
|
|
go func() {
|
|
defer close()
|
|
defer pw.Close()
|
|
|
|
seenCommand := false
|
|
shouldSkip := true
|
|
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case msg := <-recv:
|
|
// skip the header message like
|
|
|
|
// Linux pve 6.17.4-1-pve #1 SMP PREEMPT_DYNAMIC PMX 6.17.4-1 (2025-12-03T15:42Z) x86_64
|
|
//
|
|
// The programs included with the Debian GNU/Linux system are free software;
|
|
// the exact distribution terms for each program are described in the
|
|
// individual files in /usr/share/doc/*/copyright.
|
|
//
|
|
// Debian GNU/Linux comes with ABSOLUTELY NO WARRANTY, to the extent
|
|
// permitted by applicable law.
|
|
//
|
|
// root@pve:~# pct exec 101 -- journalctl -u "sftpgo" -f
|
|
//
|
|
// send begins after the line above
|
|
if shouldSkip {
|
|
// First, check if this message contains our command echo
|
|
if !seenCommand && bytes.Contains(msg, cmdLine) {
|
|
seenCommand = true
|
|
}
|
|
// Only stop skipping after we've seen the command AND output markers
|
|
if seenCommand {
|
|
if bytes.Contains(msg, []byte("\x1b[H")) || // watch cursor home
|
|
bytes.Contains(msg, []byte("\x1b[?2004l")) { // bracket paste OFF (command ended)
|
|
shouldSkip = false
|
|
}
|
|
}
|
|
continue
|
|
}
|
|
if _, err := pw.Write(msg); err != nil {
|
|
return
|
|
}
|
|
case err := <-errs:
|
|
if err != nil {
|
|
if websocket.IsUnexpectedCloseError(err, websocket.CloseGoingAway, websocket.CloseAbnormalClosure) {
|
|
_ = pw.Close()
|
|
return
|
|
}
|
|
_ = pw.CloseWithError(err)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
}()
|
|
|
|
return pr, nil
|
|
}
|
|
|
|
// LXCJournalctl streams journalctl output for the given service.
|
|
//
|
|
// If service is not empty, it will be used to filter the output by service.
|
|
// If limit is greater than 0, it will be used to limit the number of lines of output.
|
|
func (n *Node) LXCJournalctl(ctx context.Context, vmid int, service string, limit int) (io.ReadCloser, error) {
|
|
command := "journalctl -f"
|
|
if service != "" {
|
|
command = fmt.Sprintf("journalctl -u %q -f", service)
|
|
}
|
|
if limit > 0 {
|
|
command = fmt.Sprintf("%s -n %d", command, limit)
|
|
}
|
|
return n.LXCCommand(ctx, vmid, command)
|
|
}
|