X-Git-Url: http://www.git.cypherpunks.ru/?p=ucspi.git;a=blobdiff_plain;f=conn.go;h=0b1168affec69bb3f0fc91a7cee2f901df06c0fb;hp=ad9c697251a7758754aee4dc90bdd481f4d50e2e;hb=f519c4e470d63240c045c27951df3ed9de0471e9;hpb=c39958cb57c7a598f668a15a3d793a2ab708b193 diff --git a/conn.go b/conn.go index ad9c697..0b1168a 100644 --- a/conn.go +++ b/conn.go @@ -1,6 +1,6 @@ /* ucspi -- UCSPI-related utilities -Copyright (C) 2021 Sergey Matveev +Copyright (C) 2021-2022 Sergey Matveev This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by @@ -18,7 +18,7 @@ along with this program. If not, see . package ucspi import ( - "io" + "errors" "net" "os" "time" @@ -26,55 +26,51 @@ import ( var aLongTimeAgo = time.Unix(1, 0) -type UCSPIAddr struct { +type Addr struct { ip string port string } -func (addr *UCSPIAddr) Network() string { return "tcp" } +func (addr *Addr) Network() string { return "tcp" } -func (addr *UCSPIAddr) String() string { return addr.ip + ":" + addr.port } +func (addr *Addr) String() string { return addr.ip + ":" + addr.port } type Conn struct { - R *os.File - W *os.File - eof chan struct{} + R *os.File + W *os.File } -type ReadResult struct { - n int - err error +func NewConn(r, w *os.File) (*Conn, error) { + if r == nil { + return nil, errors.New("no R file descriptor") + } + if w == nil { + return nil, errors.New("no W file descriptor") + } + return &Conn{R: r, W: w}, nil } func (conn *Conn) Read(b []byte) (int, error) { - c := make(chan ReadResult) - go func() { - n, err := conn.R.Read(b) - c <- ReadResult{n, err} - }() - select { - case res := <-c: - return res.n, res.err - case <-conn.eof: - return 0, io.EOF - } + return conn.R.Read(b) } func (conn *Conn) Write(b []byte) (int, error) { return conn.W.Write(b) } func (conn *Conn) Close() error { - if err := conn.R.Close(); err != nil { - return err + errR := conn.R.Close() + errW := conn.W.Close() + if errR != nil { + return errR } - return os.Stdin.Close() + return errW } func (conn *Conn) LocalAddr() net.Addr { - return &UCSPIAddr{ip: os.Getenv("TCPLOCALIP"), port: os.Getenv("TCPLOCALPORT")} + return &Addr{ip: os.Getenv("TCPLOCALIP"), port: os.Getenv("TCPLOCALPORT")} } func (conn *Conn) RemoteAddr() net.Addr { - return &UCSPIAddr{ip: os.Getenv("TCPREMOTEIP"), port: os.Getenv("TCPREMOTEPORT")} + return &Addr{ip: os.Getenv("TCPREMOTEIP"), port: os.Getenv("TCPREMOTEPORT")} } func (conn *Conn) SetDeadline(t time.Time) error { @@ -85,12 +81,6 @@ func (conn *Conn) SetDeadline(t time.Time) error { } func (conn *Conn) SetReadDeadline(t time.Time) error { - // An ugly hack to forcefully terminate pending read. - // net/http calls SetReadDeadline(aLongTimeAgo), but file - // descriptors are not capable to exit immediately that way. - if t.Equal(aLongTimeAgo) { - conn.eof <- struct{}{} - } return conn.R.SetReadDeadline(t) }