summaryrefslogtreecommitdiff
path: root/lib/jsonutil
diff options
context:
space:
mode:
Diffstat (limited to 'lib/jsonutil')
-rw-r--r--lib/jsonutil/binstruct.go48
-rw-r--r--lib/jsonutil/hex_decoder.go61
-rw-r--r--lib/jsonutil/hex_string.go86
3 files changed, 195 insertions, 0 deletions
diff --git a/lib/jsonutil/binstruct.go b/lib/jsonutil/binstruct.go
new file mode 100644
index 0000000..7f4bd3f
--- /dev/null
+++ b/lib/jsonutil/binstruct.go
@@ -0,0 +1,48 @@
+// Copyright (C) 2023 Luke Shumaker <lukeshu@lukeshu.com>
+//
+// SPDX-License-Identifier: GPL-2.0-or-later
+
+package jsonutil
+
+import (
+ "bytes"
+ "fmt"
+ "io"
+
+ "git.lukeshu.com/go/lowmemjson"
+
+ "git.lukeshu.com/btrfs-progs-ng/lib/binstruct"
+ "git.lukeshu.com/btrfs-progs-ng/lib/textui"
+)
+
+type Binary[T any] struct {
+ Val T
+}
+
+var (
+ _ lowmemjson.Encodable = Binary[int]{}
+ _ lowmemjson.Decodable = (*Binary[int])(nil)
+)
+
+func (o Binary[T]) EncodeJSON(w io.Writer) error {
+ bs, err := binstruct.Marshal(o.Val)
+ if err != nil {
+ return err
+ }
+ return EncodeSplitHexString(w, bs, textui.Tunable(80))
+}
+
+func (o *Binary[T]) DecodeJSON(r io.RuneScanner) error {
+ var buf bytes.Buffer
+ if err := DecodeSplitHexString(r, &buf); err != nil {
+ return err
+ }
+ n, err := binstruct.Unmarshal(buf.Bytes(), &o.Val)
+ if err != nil {
+ return err
+ }
+ if n < buf.Len() {
+ return fmt.Errorf("%d bytes of garbage after value", n-buf.Len())
+ }
+ return nil
+}
diff --git a/lib/jsonutil/hex_decoder.go b/lib/jsonutil/hex_decoder.go
new file mode 100644
index 0000000..e5c84a7
--- /dev/null
+++ b/lib/jsonutil/hex_decoder.go
@@ -0,0 +1,61 @@
+// Copyright (C) 2023 Luke Shumaker <lukeshu@lukeshu.com>
+//
+// SPDX-License-Identifier: GPL-2.0-or-later
+
+package jsonutil
+
+import (
+ "fmt"
+ "io"
+ "math"
+)
+
+type invalidHexRuneError rune
+
+func (e invalidHexRuneError) Error() string {
+ return fmt.Sprintf("jsonutil: invalid hex digit: %q", rune(e))
+}
+
+// hexDecoder is like an encoding/hex.Decoder, but has a "push"
+// interface rather than a "pull" interface.
+type hexDecoder struct {
+ dst io.ByteWriter
+
+ buf byte
+ bufOK bool
+}
+
+func (d *hexDecoder) WriteRune(r rune) (int, error) {
+ if r > math.MaxUint8 {
+ return 0, invalidHexRuneError(r)
+ }
+
+ c := byte(r)
+ var v byte
+ //nolint:gomnd // Hex conversion.
+ switch {
+ case '0' <= c && c <= '9':
+ v = c - '0'
+ case 'a' <= c && c <= 'f':
+ v = c - 'a' + 10
+ case 'A' <= c && c <= 'F':
+ v = c - 'A' + 10
+ default:
+ return 0, invalidHexRuneError(r)
+ }
+
+ if !d.bufOK {
+ d.buf = v
+ d.bufOK = true
+ return 1, nil
+ }
+ d.bufOK = false
+ return 1, d.dst.WriteByte(d.buf<<4 | v)
+}
+
+func (d *hexDecoder) Close() error {
+ if d.bufOK {
+ return io.ErrUnexpectedEOF
+ }
+ return nil
+}
diff --git a/lib/jsonutil/hex_string.go b/lib/jsonutil/hex_string.go
new file mode 100644
index 0000000..3e0b154
--- /dev/null
+++ b/lib/jsonutil/hex_string.go
@@ -0,0 +1,86 @@
+// Copyright (C) 2023 Luke Shumaker <lukeshu@lukeshu.com>
+//
+// SPDX-License-Identifier: GPL-2.0-or-later
+
+// Package jsonutil provides utilities for implementing the interfaces
+// consumed by the "git.lukeshu.com/go/lowmemjson" package.
+package jsonutil
+
+import (
+ "io"
+
+ "git.lukeshu.com/go/lowmemjson"
+)
+
+func EncodeHexString[T ~[]byte | ~string](w io.Writer, str T) error {
+ const hextable = "0123456789abcdef"
+ var buf [2]byte
+ buf[0] = '"'
+ if _, err := w.Write(buf[:1]); err != nil {
+ return err
+ }
+ for i := 0; i < len(str); i++ {
+ buf[0] = hextable[str[i]>>4]
+ buf[1] = hextable[str[i]&0x0f]
+ if _, err := w.Write(buf[:]); err != nil {
+ return err
+ }
+ }
+ buf[0] = '"'
+ if _, err := w.Write(buf[:1]); err != nil {
+ return err
+ }
+ return nil
+}
+
+func DecodeHexString(r io.RuneScanner, dst io.ByteWriter) error {
+ dec := &hexDecoder{dst: dst}
+ if err := lowmemjson.DecodeString(r, dec); err != nil {
+ return err
+ }
+ return dec.Close()
+}
+
+func EncodeSplitHexString[T ~[]byte | ~string](w io.Writer, str T, maxStrLen int) error {
+ if maxStrLen <= 0 || len(str) <= maxStrLen/2 {
+ return EncodeHexString(w, str)
+ }
+ var buf [1]byte
+ buf[0] = '['
+ if _, err := w.Write(buf[:]); err != nil {
+ return err
+ }
+ for len(str) > maxStrLen/2 {
+ if err := EncodeHexString(w, str[:maxStrLen/2]); err != nil {
+ return err
+ }
+ str = str[maxStrLen/2:]
+ if len(str) > 0 {
+ buf[0] = ','
+ if _, err := w.Write(buf[:]); err != nil {
+ return err
+ }
+ }
+ }
+ if len(str) > 0 {
+ if err := EncodeHexString(w, str); err != nil {
+ return err
+ }
+ }
+ buf[0] = ']'
+ if _, err := w.Write(buf[:]); err != nil {
+ return err
+ }
+ return nil
+}
+
+func DecodeSplitHexString(r io.RuneScanner, dst io.ByteWriter) error {
+ c, _, _ := r.ReadRune()
+ _ = r.UnreadRune()
+ if c == '"' {
+ return DecodeHexString(r, dst)
+ }
+ return lowmemjson.DecodeArray(r, func(r io.RuneScanner) error {
+ return DecodeHexString(r, dst)
+ })
+}