package basicnode import ( "fmt" ipld "github.com/ipld/go-ipld-prime" "github.com/ipld/go-ipld-prime/node/mixins" ) var ( _ ipld.Node = &plainMap{} _ ipld.NodeStyle = Style__Map{} _ ipld.NodeBuilder = &plainMap__Builder{} _ ipld.NodeAssembler = &plainMap__Assembler{} ) // plainMap is a concrete type that provides a map-kind ipld.Node. // It can contain any kind of value. // plainMap is also embedded in the 'any' struct and usable from there. type plainMap struct { m map[string]ipld.Node // string key -- even if a runtime schema wrapper is using us for storage, we must have a comparable type here, and string is all we know. t []plainMap__Entry // table for fast iteration, order keeping, and yielding pointers to enable alloc/conv amortization. } type plainMap__Entry struct { k plainString // address of this used when we return keys as nodes, such as in iterators. Need in one place to amortize shifts to heap when ptr'ing for iface. v ipld.Node // identical to map values. keeping them here simplifies iteration. (in codegen'd maps, this position is also part of amortization, but in this implementation, that's less useful.) // note on alternate implementations: 'v' could also use the 'any' type, and thus amortize value allocations. the memory size trade would be large however, so we don't, here. } // -- Node interface methods --> func (plainMap) ReprKind() ipld.ReprKind { return ipld.ReprKind_Map } func (n *plainMap) LookupString(key string) (ipld.Node, error) { v, exists := n.m[key] if !exists { return nil, ipld.ErrNotExists{ipld.PathSegmentOfString(key)} } return v, nil } func (n *plainMap) Lookup(key ipld.Node) (ipld.Node, error) { ks, err := key.AsString() if err != nil { return nil, err } return n.LookupString(ks) } func (plainMap) LookupIndex(idx int) (ipld.Node, error) { return mixins.Map{"map"}.LookupIndex(0) } func (n *plainMap) LookupSegment(seg ipld.PathSegment) (ipld.Node, error) { return n.LookupString(seg.String()) } func (n *plainMap) MapIterator() ipld.MapIterator { return &plainMap_MapIterator{n, 0} } func (plainMap) ListIterator() ipld.ListIterator { return nil } func (n *plainMap) Length() int { return len(n.t) } func (plainMap) IsUndefined() bool { return false } func (plainMap) IsNull() bool { return false } func (plainMap) AsBool() (bool, error) { return mixins.Map{"map"}.AsBool() } func (plainMap) AsInt() (int, error) { return mixins.Map{"map"}.AsInt() } func (plainMap) AsFloat() (float64, error) { return mixins.Map{"map"}.AsFloat() } func (plainMap) AsString() (string, error) { return mixins.Map{"map"}.AsString() } func (plainMap) AsBytes() ([]byte, error) { return mixins.Map{"map"}.AsBytes() } func (plainMap) AsLink() (ipld.Link, error) { return mixins.Map{"map"}.AsLink() } func (plainMap) Style() ipld.NodeStyle { return Style__Map{} } type plainMap_MapIterator struct { n *plainMap idx int } func (itr *plainMap_MapIterator) Next() (k ipld.Node, v ipld.Node, _ error) { if itr.Done() { return nil, nil, ipld.ErrIteratorOverread{} } k = &itr.n.t[itr.idx].k v = itr.n.t[itr.idx].v itr.idx++ return } func (itr *plainMap_MapIterator) Done() bool { return itr.idx >= len(itr.n.t) } // -- NodeStyle --> type Style__Map struct{} func (Style__Map) NewBuilder() ipld.NodeBuilder { return &plainMap__Builder{plainMap__Assembler{w: &plainMap{}}} } // -- NodeBuilder --> type plainMap__Builder struct { plainMap__Assembler } func (nb *plainMap__Builder) Build() ipld.Node { if nb.state != maState_finished { panic("invalid state: assembler must be 'finished' before Build can be called!") } return nb.w } func (nb *plainMap__Builder) Reset() { *nb = plainMap__Builder{} nb.w = &plainMap{} } // -- NodeAssembler --> type plainMap__Assembler struct { w *plainMap ka plainMap__KeyAssembler va plainMap__ValueAssembler state maState } type plainMap__KeyAssembler struct { ma *plainMap__Assembler } type plainMap__ValueAssembler struct { ma *plainMap__Assembler } // maState is an enum of the state machine for a map assembler. // (this might be something to export reusably, but it's also very much an impl detail that need not be seen, so, dubious.) type maState uint8 const ( maState_initial maState = iota // also the 'expect key or finish' state maState_midKey // waiting for a 'finished' state in the KeyAssembler. maState_expectValue // 'AssembleValue' is the only valid next step maState_midValue // waiting for a 'finished' state in the ValueAssembler. maState_finished // 'w' will also be nil, but this is a politer statement ) func (na *plainMap__Assembler) BeginMap(sizeHint int) (ipld.MapAssembler, error) { if sizeHint < 0 { sizeHint = 0 } // Allocate storage space. na.w.t = make([]plainMap__Entry, 0, sizeHint) na.w.m = make(map[string]ipld.Node, sizeHint) // That's it; return self as the MapAssembler. We already have all the right methods on this structure. return na, nil } func (plainMap__Assembler) BeginList(sizeHint int) (ipld.ListAssembler, error) { return mixins.MapAssembler{"map"}.BeginList(0) } func (plainMap__Assembler) AssignNull() error { return mixins.MapAssembler{"map"}.AssignNull() } func (plainMap__Assembler) AssignBool(bool) error { return mixins.MapAssembler{"map"}.AssignBool(false) } func (plainMap__Assembler) AssignInt(int) error { return mixins.MapAssembler{"map"}.AssignInt(0) } func (plainMap__Assembler) AssignFloat(float64) error { return mixins.MapAssembler{"map"}.AssignFloat(0) } func (plainMap__Assembler) AssignString(string) error { return mixins.MapAssembler{"map"}.AssignString("") } func (plainMap__Assembler) AssignBytes([]byte) error { return mixins.MapAssembler{"map"}.AssignBytes(nil) } func (plainMap__Assembler) AssignLink(ipld.Link) error { return mixins.MapAssembler{"map"}.AssignLink(nil) } func (na *plainMap__Assembler) AssignNode(v ipld.Node) error { // todo: apply a generic 'copy' function. // todo: probably can also shortcut to copying na.t and na.m if it's our same concrete type? // (can't quite just `na.w = v`, because we don't have 'freeze' features, and we don't wanna open door to mutation of 'v'.) // (wait... actually, probably we can? 'Assign' is a "finish" method. we can&should invalidate the wip pointer here.) panic("later") } func (plainMap__Assembler) Style() ipld.NodeStyle { return Style__Map{} } // -- MapAssembler --> // AssembleDirectly is part of conforming to MapAssembler, which we do on // plainMap__Assembler so that BeginMap can just return a retyped pointer rather than new object. func (ma *plainMap__Assembler) AssembleDirectly(k string) (ipld.NodeAssembler, error) { // Sanity check, then update, assembler state. if ma.state != maState_initial { panic("misuse") } ma.state = maState_midValue // Check for dup keys; error if so. _, exists := ma.w.m[k] if exists { return nil, ipld.ErrRepeatedMapKey{plainString(k)} } ma.w.t = append(ma.w.t, plainMap__Entry{k: plainString(k)}) // Make value assembler valid by giving it pointer back to whole 'ma'; yield it. ma.va.ma = ma return &ma.va, nil } // AssembleKey is part of conforming to MapAssembler, which we do on // plainMap__Assembler so that BeginMap can just return a retyped pointer rather than new object. func (ma *plainMap__Assembler) AssembleKey() ipld.NodeAssembler { // Sanity check, then update, assembler state. if ma.state != maState_initial { panic("misuse") } ma.state = maState_midKey // Extend entry table. ma.w.t = append(ma.w.t, plainMap__Entry{}) // Make key assembler valid by giving it pointer back to whole 'ma'; yield it. ma.ka.ma = ma return &ma.ka } // AssembleValue is part of conforming to MapAssembler, which we do on // plainMap__Assembler so that BeginMap can just return a retyped pointer rather than new object. func (ma *plainMap__Assembler) AssembleValue() ipld.NodeAssembler { // Sanity check, then update, assembler state. if ma.state != maState_expectValue { panic("misuse") } ma.state = maState_midValue // Make value assembler valid by giving it pointer back to whole 'ma'; yield it. ma.va.ma = ma return &ma.va } // Finish is part of conforming to MapAssembler, which we do on // plainMap__Assembler so that BeginMap can just return a retyped pointer rather than new object. func (ma *plainMap__Assembler) Finish() error { // Sanity check, then update, assembler state. if ma.state != maState_initial { panic("misuse") } ma.state = maState_finished // validators could run and report errors promptly, if this type had any. return nil } func (plainMap__Assembler) KeyStyle() ipld.NodeStyle { return Style__String{} } func (plainMap__Assembler) ValueStyle(_ string) ipld.NodeStyle { return Style__Any{} } // -- MapAssembler.KeyAssembler --> func (plainMap__KeyAssembler) BeginMap(sizeHint int) (ipld.MapAssembler, error) { return mixins.StringAssembler{"string"}.BeginMap(0) } func (plainMap__KeyAssembler) BeginList(sizeHint int) (ipld.ListAssembler, error) { return mixins.StringAssembler{"string"}.BeginList(0) } func (plainMap__KeyAssembler) AssignNull() error { return mixins.StringAssembler{"string"}.AssignNull() } func (plainMap__KeyAssembler) AssignBool(bool) error { return mixins.StringAssembler{"string"}.AssignBool(false) } func (plainMap__KeyAssembler) AssignInt(int) error { return mixins.StringAssembler{"string"}.AssignInt(0) } func (plainMap__KeyAssembler) AssignFloat(float64) error { return mixins.StringAssembler{"string"}.AssignFloat(0) } func (mka *plainMap__KeyAssembler) AssignString(v string) error { // Check for dup keys; error if so. _, exists := mka.ma.w.m[v] if exists { return ipld.ErrRepeatedMapKey{plainString(v)} } // Assign the key into the end of the entry table; // we'll be doing map insertions after we get the value in hand. // (There's no need to delegate to another assembler for the key type, // because we're just at Data Model level here, which only regards plain strings.) mka.ma.w.t[len(mka.ma.w.t)-1].k = plainString(v) // Update parent assembler state: clear to proceed. mka.ma.state = maState_expectValue mka.ma = nil // invalidate self to prevent further incorrect use. return nil } func (plainMap__KeyAssembler) AssignBytes([]byte) error { return mixins.StringAssembler{"string"}.AssignBytes(nil) } func (plainMap__KeyAssembler) AssignLink(ipld.Link) error { return mixins.StringAssembler{"string"}.AssignLink(nil) } func (mka *plainMap__KeyAssembler) AssignNode(v ipld.Node) error { vs, err := v.AsString() if err != nil { return fmt.Errorf("cannot assign non-string node into map key assembler") // FIXME:errors: this doesn't quite fit in ErrWrongKind cleanly; new error type? } return mka.AssignString(vs) } func (plainMap__KeyAssembler) Style() ipld.NodeStyle { return Style__String{} } // -- MapAssembler.ValueAssembler --> func (mva *plainMap__ValueAssembler) BeginMap(sizeHint int) (ipld.MapAssembler, error) { ma := plainMap__ValueAssemblerMap{} ma.ca.w = &plainMap{} ma.p = mva.ma _, err := ma.ca.BeginMap(sizeHint) return &ma, err } func (mva *plainMap__ValueAssembler) BeginList(sizeHint int) (ipld.ListAssembler, error) { la := plainMap__ValueAssemblerList{} la.ca.w = &plainList{} la.p = mva.ma _, err := la.ca.BeginList(sizeHint) return &la, err } func (mva *plainMap__ValueAssembler) AssignNull() error { return mva.AssignNode(ipld.Null) } func (mva *plainMap__ValueAssembler) AssignBool(v bool) error { vb := plainBool(v) return mva.AssignNode(&vb) } func (mva *plainMap__ValueAssembler) AssignInt(v int) error { vb := plainInt(v) return mva.AssignNode(&vb) } func (mva *plainMap__ValueAssembler) AssignFloat(v float64) error { vb := plainFloat(v) return mva.AssignNode(&vb) } func (mva *plainMap__ValueAssembler) AssignString(v string) error { vb := plainString(v) return mva.AssignNode(&vb) } func (mva *plainMap__ValueAssembler) AssignBytes(v []byte) error { vb := plainBytes(v) return mva.AssignNode(&vb) } func (mva *plainMap__ValueAssembler) AssignLink(v ipld.Link) error { vb := plainLink{v} return mva.AssignNode(&vb) } func (mva *plainMap__ValueAssembler) AssignNode(v ipld.Node) error { l := len(mva.ma.w.t) - 1 mva.ma.w.t[l].v = v mva.ma.w.m[string(mva.ma.w.t[l].k)] = v mva.ma.state = maState_initial mva.ma = nil // invalidate self to prevent further incorrect use. return nil } func (plainMap__ValueAssembler) Style() ipld.NodeStyle { return Style__Any{} } type plainMap__ValueAssemblerMap struct { ca plainMap__Assembler p *plainMap__Assembler // pointer back to parent, for final insert and state bump } // we briefly state only the methods we need to delegate here. // just embedding plainMap__Assembler also behaves correctly, // but causes a lot of unnecessary autogenerated functions in the final binary. func (ma *plainMap__ValueAssemblerMap) AssembleDirectly(k string) (ipld.NodeAssembler, error) { return ma.ca.AssembleDirectly(k) } func (ma *plainMap__ValueAssemblerMap) AssembleKey() ipld.NodeAssembler { return ma.ca.AssembleKey() } func (ma *plainMap__ValueAssemblerMap) AssembleValue() ipld.NodeAssembler { return ma.ca.AssembleValue() } func (plainMap__ValueAssemblerMap) KeyStyle() ipld.NodeStyle { return Style__String{} } func (plainMap__ValueAssemblerMap) ValueStyle(_ string) ipld.NodeStyle { return Style__Any{} } func (ma *plainMap__ValueAssemblerMap) Finish() error { if err := ma.ca.Finish(); err != nil { return err } w := ma.ca.w ma.ca.w = nil return ma.p.va.AssignNode(w) } type plainMap__ValueAssemblerList struct { ca plainList__Assembler p *plainMap__Assembler // pointer back to parent, for final insert and state bump } // we briefly state only the methods we need to delegate here. // just embedding plainList__Assembler also behaves correctly, // but causes a lot of unnecessary autogenerated functions in the final binary. func (la *plainMap__ValueAssemblerList) AssembleValue() ipld.NodeAssembler { return la.ca.AssembleValue() } func (plainMap__ValueAssemblerList) ValueStyle(_ int) ipld.NodeStyle { return Style__Any{} } func (la *plainMap__ValueAssemblerList) Finish() error { if err := la.ca.Finish(); err != nil { return err } w := la.ca.w la.ca.w = nil return la.p.va.AssignNode(w) }