mirror of
https://github.com/v2fly/v2ray-core.git
synced 2025-01-03 07:56:42 -05:00
refine ctlcmd
This commit is contained in:
parent
48e825567a
commit
f751bb610c
@ -3,7 +3,6 @@ package buf
|
|||||||
import (
|
import (
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"os"
|
|
||||||
|
|
||||||
"v2ray.com/core/common"
|
"v2ray.com/core/common"
|
||||||
"v2ray.com/core/common/errors"
|
"v2ray.com/core/common/errors"
|
||||||
@ -14,22 +13,12 @@ import (
|
|||||||
func ReadAllToMultiBuffer(reader io.Reader) (MultiBuffer, error) {
|
func ReadAllToMultiBuffer(reader io.Reader) (MultiBuffer, error) {
|
||||||
mb := NewMultiBufferCap(128)
|
mb := NewMultiBufferCap(128)
|
||||||
|
|
||||||
for {
|
if _, err := mb.ReadFrom(reader); err != nil {
|
||||||
b := New()
|
|
||||||
err := b.Reset(ReadFrom(reader))
|
|
||||||
if b.IsEmpty() {
|
|
||||||
b.Release()
|
|
||||||
} else {
|
|
||||||
mb.Append(b)
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
if errors.Cause(err) == io.EOF || errors.Cause(err) == os.ErrClosed {
|
|
||||||
return mb, nil
|
|
||||||
}
|
|
||||||
mb.Release()
|
mb.Release()
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
return mb, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// ReadSizeToMultiBuffer reads specific number of bytes from reader into a MultiBuffer.
|
// ReadSizeToMultiBuffer reads specific number of bytes from reader into a MultiBuffer.
|
||||||
@ -102,6 +91,28 @@ func (mb MultiBuffer) Copy(b []byte) int {
|
|||||||
return total
|
return total
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ReadFrom implements io.ReaderFrom.
|
||||||
|
func (mb *MultiBuffer) ReadFrom(reader io.Reader) (int64, error) {
|
||||||
|
totalBytes := int64(0)
|
||||||
|
|
||||||
|
for {
|
||||||
|
b := New()
|
||||||
|
err := b.Reset(ReadFrom(reader))
|
||||||
|
if b.IsEmpty() {
|
||||||
|
b.Release()
|
||||||
|
} else {
|
||||||
|
mb.Append(b)
|
||||||
|
}
|
||||||
|
totalBytes += int64(b.Len())
|
||||||
|
if err != nil {
|
||||||
|
if errors.Cause(err) == io.EOF {
|
||||||
|
return totalBytes, nil
|
||||||
|
}
|
||||||
|
return totalBytes, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Read implements io.Reader.
|
// Read implements io.Reader.
|
||||||
func (mb *MultiBuffer) Read(b []byte) (int, error) {
|
func (mb *MultiBuffer) Read(b []byte) (int, error) {
|
||||||
if mb.Len() == 0 {
|
if mb.Len() == 0 {
|
||||||
@ -125,6 +136,22 @@ func (mb *MultiBuffer) Read(b []byte) (int, error) {
|
|||||||
return totalBytes, nil
|
return totalBytes, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// WriteTo implements io.WriterTo.
|
||||||
|
func (mb *MultiBuffer) WriteTo(writer io.Writer) (int64, error) {
|
||||||
|
defer mb.Release()
|
||||||
|
|
||||||
|
totalBytes := int64(0)
|
||||||
|
for _, b := range *mb {
|
||||||
|
nBytes, err := writer.Write(b.Bytes())
|
||||||
|
totalBytes += int64(nBytes)
|
||||||
|
if err != nil {
|
||||||
|
return totalBytes, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return totalBytes, nil
|
||||||
|
}
|
||||||
|
|
||||||
// Write implements io.Writer.
|
// Write implements io.Writer.
|
||||||
func (mb *MultiBuffer) Write(b []byte) (int, error) {
|
func (mb *MultiBuffer) Write(b []byte) (int, error) {
|
||||||
totalBytes := len(b)
|
totalBytes := len(b)
|
||||||
|
@ -2,6 +2,7 @@ package buf_test
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"crypto/rand"
|
"crypto/rand"
|
||||||
|
"io"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"v2ray.com/core/common"
|
"v2ray.com/core/common"
|
||||||
@ -48,3 +49,10 @@ func TestMultiBufferSliceBySizeLarge(t *testing.T) {
|
|||||||
mb2 := mb.SliceBySize(4 * 1024)
|
mb2 := mb.SliceBySize(4 * 1024)
|
||||||
assert(mb2.Len(), Equals, int32(4*1024))
|
assert(mb2.Len(), Equals, int32(4*1024))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestInterface(t *testing.T) {
|
||||||
|
assert := With(t)
|
||||||
|
|
||||||
|
assert((*MultiBuffer)(nil), Implements, (*io.WriterTo)(nil))
|
||||||
|
assert((*MultiBuffer)(nil), Implements, (*io.ReaderFrom)(nil))
|
||||||
|
}
|
||||||
|
@ -1,14 +1,12 @@
|
|||||||
package ctlcmd
|
package ctlcmd
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
|
||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
|
|
||||||
"v2ray.com/core/common/buf"
|
"v2ray.com/core/common/buf"
|
||||||
"v2ray.com/core/common/platform"
|
"v2ray.com/core/common/platform"
|
||||||
"v2ray.com/core/common/signal"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
//go:generate go run $GOPATH/src/v2ray.com/core/common/errors/errorgen/main.go -pkg ctlcmd -path Command,Platform,CtlCmd
|
//go:generate go run $GOPATH/src/v2ray.com/core/common/errors/errorgen/main.go -pkg ctlcmd -path Command,Platform,CtlCmd
|
||||||
@ -19,49 +17,30 @@ func Run(args []string, input io.Reader) (buf.MultiBuffer, error) {
|
|||||||
return nil, newError("v2ctl doesn't exist").Base(err)
|
return nil, newError("v2ctl doesn't exist").Base(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
errBuffer := &buf.MultiBuffer{}
|
errBuffer := buf.MultiBuffer{}
|
||||||
|
outBuffer := buf.MultiBuffer{}
|
||||||
|
|
||||||
cmd := exec.Command(v2ctl, args...)
|
cmd := exec.Command(v2ctl, args...)
|
||||||
cmd.Stderr = errBuffer
|
cmd.Stderr = &errBuffer
|
||||||
|
cmd.Stdout = &outBuffer
|
||||||
cmd.SysProcAttr = getSysProcAttr()
|
cmd.SysProcAttr = getSysProcAttr()
|
||||||
if input != nil {
|
if input != nil {
|
||||||
cmd.Stdin = input
|
cmd.Stdin = input
|
||||||
}
|
}
|
||||||
|
|
||||||
stdoutReader, err := cmd.StdoutPipe()
|
|
||||||
if err != nil {
|
|
||||||
return nil, newError("failed to get stdout from v2ctl").Base(err)
|
|
||||||
}
|
|
||||||
defer stdoutReader.Close()
|
|
||||||
|
|
||||||
if err := cmd.Start(); err != nil {
|
if err := cmd.Start(); err != nil {
|
||||||
return nil, newError("failed to start v2ctl").Base(err)
|
return nil, newError("failed to start v2ctl").Base(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
var content buf.MultiBuffer
|
|
||||||
loadTask := func() error {
|
|
||||||
c, err := buf.ReadAllToMultiBuffer(stdoutReader)
|
|
||||||
if err != nil {
|
|
||||||
return newError("failed to read config").Base(err)
|
|
||||||
}
|
|
||||||
content = c
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
waitTask := func() error {
|
|
||||||
if err := cmd.Wait(); err != nil {
|
if err := cmd.Wait(); err != nil {
|
||||||
msg := "failed to execute v2ctl"
|
msg := "failed to execute v2ctl"
|
||||||
if errBuffer.Len() > 0 {
|
if errBuffer.Len() > 0 {
|
||||||
msg += ": " + errBuffer.String()
|
msg += ": " + errBuffer.String()
|
||||||
}
|
}
|
||||||
return newError(msg).Base(err)
|
errBuffer.Release()
|
||||||
}
|
outBuffer.Release()
|
||||||
return nil
|
return nil, newError(msg).Base(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := signal.ExecuteParallel(context.Background(), loadTask, waitTask); err != nil {
|
return outBuffer, nil
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return content, nil
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user