summaryrefslogtreecommitdiff
path: root/pkg/netstring/netstring.go
blob: 00722a8152a3c88aa6828a0a1db975de9cbd7f1f (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
package netstring

import (
	"errors"
	"fmt"
	"io"
)

type Encoder struct {
	w io.Writer
}

type Decoder struct {
	r io.Reader
}

var errBadFormat = errors.New("netstring: bad format")

/*
func Encode(s string) string {
	return fmt.Sprintf("%d:%s,", len(s), s)
}
*/

func NewEncoder(w io.Writer) *Encoder {
	return &Encoder{w: w}
}

func NewDecoder(r io.Reader) *Decoder {
	return &Decoder{r: r}
}

func (e *Encoder) Encode(s string) error {
	_, err := fmt.Fprintf(e.w, "%d:%s,", len(s), s)
	return err
}

func (d *Decoder) Decode() (out string, err error) {
	var n int

	_, err = fmt.Fscanf(d.r, "%d:", &n)
	if err != nil {
		return "", fmt.Errorf("netstring: %w", err)
	}

	buf := make([]byte, n+1)

	_, err = io.ReadFull(d.r, buf)
	if err != nil {
		if err == io.EOF {
			err = io.ErrUnexpectedEOF
		}
		return "", fmt.Errorf("netstring: %w", err)
	}

	if buf[n] != ',' {
		return "", errBadFormat
	}

	return string(buf[:n]), nil
}

func Encode(w io.Writer, args ...string) error {
	e := Encoder{w: w}
	for _, s := range args {
		if err := e.Encode(s); err != nil {
			return err
		}
	}
	return nil
}

func Decode(r io.Reader, args ...*string) error {
	d := Decoder{r: r}
	for _, s := range args {
		t, err := d.Decode()
		if err != nil {
			return err
		}
		*s = t
	}
	return nil
}