X-Git-Url: https://ruderich.org/simon/gitweb/?p=nsscash%2Fnsscash.git;a=blobdiff_plain;f=group.go;h=66da8f00551e32de8272a5b02b2fdb7c4006039f;hp=214f91438d78740a331c487cbfe5ecab1355718c;hb=526f6f1db39bde8ca1f7684225a3983634bddafe;hpb=c6b898ea9ab372c58c59658c2acf17ea831d71db diff --git a/group.go b/group.go index 214f914..66da8f0 100644 --- a/group.go +++ b/group.go @@ -23,6 +23,7 @@ import ( "encoding/binary" "fmt" "io" + "math" "sort" "strconv" "strings" @@ -67,6 +68,11 @@ func ParseGroups(r io.Reader) ([]Group, error) { t, err := s.ReadString('\n') if err != nil { if err == io.EOF { + if t != "" { + return nil, fmt.Errorf( + "no newline in last line: %q", + t) + } break } return nil, err @@ -102,7 +108,7 @@ func ParseGroups(r io.Reader) ([]Group, error) { return res, nil } -func SerializeGroup(g Group) []byte { +func SerializeGroup(g Group) ([]byte, error) { le := binary.LittleEndian // Concatenate all (NUL-terminated) strings and store the offsets @@ -130,6 +136,11 @@ func SerializeGroup(g Group) []byte { } // And the group members concatenated as above data.Write(mems.Bytes()) + // Ensure the offsets can fit the length of this entry + if data.Len() > math.MaxUint16 { + return nil, fmt.Errorf("group too large to serialize: %v, %v", + data.Len(), g) + } size := uint16(data.Len()) var res bytes.Buffer // serialized result @@ -158,7 +169,7 @@ func SerializeGroup(g Group) []byte { // struct are 8 byte aligned alignBufferTo(&res, 8) - return res.Bytes() + return res.Bytes(), nil } func SerializeGroups(w io.Writer, grs []Group) error { @@ -168,7 +179,11 @@ func SerializeGroups(w io.Writer, grs []Group) error { for _, g := range grs { // TODO: warn about duplicate entries offsets[toKey(g)] = uint64(data.Len()) - data.Write(SerializeGroup(g)) + x, err := SerializeGroup(g) + if err != nil { + return err + } + data.Write(x) } // Copy to prevent sorting from modifying the argument