You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
134 lines
3.4 KiB
134 lines
3.4 KiB
// Package okenvelope is a codec for the OK message, which is an acknowledgement |
|
// for an EVENT eventenvelope.Submission, containing true/false and if false a |
|
// message with a machine readable error type as found in the messages package. |
|
package okenvelope |
|
|
|
import ( |
|
"io" |
|
|
|
"lol.mleku.dev/chk" |
|
"lol.mleku.dev/errorf" |
|
"lol.mleku.dev/log" |
|
"github.com/minio/sha256-simd" |
|
"next.orly.dev/pkg/encoders/envelopes" |
|
"next.orly.dev/pkg/encoders/hex" |
|
"next.orly.dev/pkg/encoders/text" |
|
"next.orly.dev/pkg/interfaces/codec" |
|
"next.orly.dev/pkg/utils/constraints" |
|
) |
|
|
|
// L is the label associated with this type of codec.Envelope. |
|
const L = "OK" |
|
|
|
// T is an OK envelope, used to signal acceptance or rejection, with a reason, |
|
// to an eventenvelope.Submission. |
|
type T struct { |
|
EventID []byte |
|
OK bool |
|
Reason []byte |
|
} |
|
|
|
var _ codec.Envelope = (*T)(nil) |
|
|
|
// New creates a new empty OK T. |
|
func New() *T { return &T{} } |
|
|
|
// NewFrom creates a new okenvelope.T with a string for the subscription.Id and |
|
// the optional reason. |
|
func NewFrom[V constraints.Bytes](eid V, ok bool, msg ...V) *T { |
|
var m []byte |
|
if len(msg) > 0 { |
|
m = []byte(msg[0]) |
|
} |
|
if len(eid) != sha256.Size { |
|
log.W.F( |
|
"event Subscription unexpected length, expect %d got %d", |
|
len(eid), sha256.Size, |
|
) |
|
} |
|
return &T{EventID: []byte(eid), OK: ok, Reason: m} |
|
} |
|
|
|
// Label returns the label of an okenvelope.T. |
|
func (en *T) Label() string { return L } |
|
|
|
// ReasonString returns the Reason in the form of a string. |
|
func (en *T) ReasonString() string { return string(en.Reason) } |
|
|
|
// Write the okenvelope.T to a provided io.Writer. |
|
func (en *T) Write(w io.Writer) (err error) { |
|
_, err = w.Write(en.Marshal(nil)) |
|
return |
|
} |
|
|
|
// Marshal a okenvelope.T from minified JSON, appending to a provided |
|
// destination slice. Note that this ensures correct string escaping on the |
|
// subscription.Id and Reason fields. |
|
func (en *T) Marshal(dst []byte) (b []byte) { |
|
var err error |
|
_ = err |
|
b = dst |
|
b = envelopes.Marshal( |
|
b, L, |
|
func(bst []byte) (o []byte) { |
|
o = bst |
|
o = append(o, '"') |
|
o = hex.EncAppend(o, en.EventID) |
|
o = append(o, '"') |
|
o = append(o, ',') |
|
o = text.MarshalBool(o, en.OK) |
|
o = append(o, ',') |
|
o = append(o, '"') |
|
o = text.NostrEscape(o, en.Reason) |
|
o = append(o, '"') |
|
return |
|
}, |
|
) |
|
return |
|
} |
|
|
|
// Unmarshal a okenvelope.T from minified JSON, returning the remainder after |
|
// the end of the envelope. Note that this ensures the Reason and |
|
// subscription.Id strings are correctly unescaped by NIP-01 escaping rules. |
|
func (en *T) Unmarshal(b []byte) (r []byte, err error) { |
|
r = b |
|
var idBytes []byte |
|
// Parse event id as quoted hex (NIP-20 compliant) |
|
if idBytes, r, err = text.UnmarshalHex(r); err != nil { |
|
return |
|
} |
|
if len(idBytes) != sha256.Size { |
|
err = errorf.E( |
|
"invalid size for Subscription, require %d got %d", |
|
sha256.Size, len(idBytes), |
|
) |
|
return |
|
} |
|
en.EventID = idBytes |
|
if r, err = text.Comma(r); chk.E(err) { |
|
return |
|
} |
|
if r, en.OK, err = text.UnmarshalBool(r); chk.E(err) { |
|
return |
|
} |
|
if r, err = text.Comma(r); chk.E(err) { |
|
return |
|
} |
|
if en.Reason, r, err = text.UnmarshalQuoted(r); chk.E(err) { |
|
return |
|
} |
|
if r, err = envelopes.SkipToTheEnd(r); chk.E(err) { |
|
return |
|
} |
|
return |
|
} |
|
|
|
// Parse reads a OK envelope in minified JSON into a newly allocated |
|
// okenvelope.T. |
|
func Parse(b []byte) (t *T, rem []byte, err error) { |
|
t = New() |
|
if rem, err = t.Unmarshal(b); chk.E(err) { |
|
return |
|
} |
|
return |
|
}
|
|
|