package gengo import ( "io" "github.com/ipld/go-ipld-prime/schema" "github.com/ipld/go-ipld-prime/schema/gen/go/mixins" ) type structGenerator struct { AdjCfg *AdjunctCfg mixins.MapTraits PkgName string Type schema.TypeStruct } // --- native content and specializations ---> func (g structGenerator) EmitNativeType(w io.Writer) { doTemplate(` type _{{ .Type | TypeSymbol }} struct { {{- range $field := .Type.Fields}} {{ $field | FieldSymbolLower }} _{{ $field.Type | TypeSymbol }}{{if $field.IsMaybe }}__Maybe{{end}} {{- end}} } type {{ .Type | TypeSymbol }} = *_{{ .Type | TypeSymbol }} `, w, g.AdjCfg, g) } func (g structGenerator) EmitNativeAccessors(w io.Writer) { doTemplate(` {{- $type := .Type -}} {{- /* ranging modifies dot, unhelpfully */ -}} {{- range $field := .Type.Fields }} func (n _{{ $type | TypeSymbol }}) Field{{ $field | FieldSymbolUpper }}() {{ if $field.IsMaybe }}Maybe{{end}}{{ $field.Type | TypeSymbol }} { return &n.{{ $field | FieldSymbolLower }} } {{- end}} `, w, g.AdjCfg, g) } func (g structGenerator) EmitNativeBuilder(w io.Writer) { // Unclear what, if anything, goes here. } func (g structGenerator) EmitNativeMaybe(w io.Writer) { // TODO maybes need a lot of questions answered } // --- type info ---> func (g structGenerator) EmitTypeConst(w io.Writer) { doTemplate(` // TODO EmitTypeConst `, w, g.AdjCfg, g) } // --- TypedNode interface satisfaction ---> func (g structGenerator) EmitTypedNodeMethodType(w io.Writer) { doTemplate(` func ({{ .Type | TypeSymbol }}) Type() schema.Type { return nil /*TODO:typelit*/ } `, w, g.AdjCfg, g) } func (g structGenerator) EmitTypedNodeMethodRepresentation(w io.Writer) { // Perhaps surprisingly, the way to get the representation node pointer // does not actually depend on what the representation strategy is. // REVIEW: this appears to be standard even across kinds; can we extract it? doTemplate(` func (n {{ .Type | TypeSymbol }}) Representation() ipld.Node { return (*_{{ .Type | TypeSymbol }}__Repr)(n) } `, w, g.AdjCfg, g) } // --- Node interface satisfaction ---> func (g structGenerator) EmitNodeType(w io.Writer) { // No additional types needed. Methods all attach to the native type. // We do, however, want some constants for our fields; // they'll make iterators able to work faster. So let's emit those. doTemplate(` var ( {{- $type := .Type -}} {{- /* ranging modifies dot, unhelpfully */ -}} {{- range $field := .Type.Fields }} fieldName__{{ $type | TypeSymbol }}_{{ $field | FieldSymbolUpper }} = _String{"{{ $field.Name }}"} {{- end }} ) `, w, g.AdjCfg, g) } func (g structGenerator) EmitNodeTypeAssertions(w io.Writer) { doTemplate(` var _ ipld.Node = ({{ .Type | TypeSymbol }})(&_{{ .Type | TypeSymbol }}{}) var _ schema.TypedNode = ({{ .Type | TypeSymbol }})(&_{{ .Type | TypeSymbol }}{}) `, w, g.AdjCfg, g) } func (g structGenerator) EmitNodeMethodLookupString(w io.Writer) { doTemplate(` func (n {{ .Type | TypeSymbol }}) LookupString(key string) (ipld.Node, error) { switch key { {{- range $field := .Type.Fields }} case "{{ $field.Name }}": {{- if $field.IsOptional }} if n.{{ $field | FieldSymbolLower }}.m == schema.Maybe_Absent { return ipld.Undef, nil } {{- end}} {{- if $field.IsNullable }} if n.{{ $field | FieldSymbolLower }}.m == schema.Maybe_Null { return ipld.Null, nil } {{- end}} {{- if $field.IsMaybe }} return {{if not (MaybeUsesPtr $field.Type) }}&{{end}}n.{{ $field | FieldSymbolLower }}.v, nil {{- else}} return &n.{{ $field | FieldSymbolLower }}, nil {{- end}} {{- end}} default: return nil, schema.ErrNoSuchField{Type: nil /*TODO*/, FieldName: key} } } `, w, g.AdjCfg, g) } func (g structGenerator) EmitNodeMethodLookup(w io.Writer) { doTemplate(` func (n {{ .Type | TypeSymbol }}) Lookup(key ipld.Node) (ipld.Node, error) { ks, err := key.AsString() if err != nil { return nil, err } return n.LookupString(ks) } `, w, g.AdjCfg, g) } func (g structGenerator) EmitNodeMethodMapIterator(w io.Writer) { // Note that the typed iterator will report absent fields. // The representation iterator (if has one) however will skip those. doTemplate(` func (n {{ .Type | TypeSymbol }}) MapIterator() ipld.MapIterator { return &_{{ .Type | TypeSymbol }}__MapItr{n, 0} } type _{{ .Type | TypeSymbol }}__MapItr struct { n {{ .Type | TypeSymbol }} idx int } func (itr *_{{ .Type | TypeSymbol }}__MapItr) Next() (k ipld.Node, v ipld.Node, _ error) { if itr.idx >= {{ len .Type.Fields }} { return nil, nil, ipld.ErrIteratorOverread{} } switch itr.idx { {{- $type := .Type -}} {{- /* ranging modifies dot, unhelpfully */ -}} {{- range $i, $field := .Type.Fields }} case {{ $i }}: k = &fieldName__{{ $type | TypeSymbol }}_{{ $field | FieldSymbolUpper }} {{- if $field.IsOptional }} if itr.n.{{ $field | FieldSymbolLower }}.m == schema.Maybe_Absent { v = ipld.Undef break } {{- end}} {{- if $field.IsNullable }} if itr.n.{{ $field | FieldSymbolLower }}.m == schema.Maybe_Null { v = ipld.Null break } {{- end}} {{- if $field.IsMaybe }} v = {{if not (MaybeUsesPtr $field.Type) }}&{{end}}itr.n.{{ $field | FieldSymbolLower}}.v {{- else}} v = &itr.n.{{ $field | FieldSymbolLower}} {{- end}} {{- end}} default: panic("unreachable") } itr.idx++ return } func (itr *_{{ .Type | TypeSymbol }}__MapItr) Done() bool { return itr.idx >= {{ len .Type.Fields }} } `, w, g.AdjCfg, g) } func (g structGenerator) EmitNodeMethodLength(w io.Writer) { doTemplate(` func ({{ .Type | TypeSymbol }}) Length() int { return {{ len .Type.Fields }} } `, w, g.AdjCfg, g) } func (g structGenerator) EmitNodeMethodStyle(w io.Writer) { // REVIEW: this appears to be standard even across kinds; can we extract it? doTemplate(` func ({{ .Type | TypeSymbol }}) Style() ipld.NodeStyle { return _{{ .Type | TypeSymbol }}__Style{} } `, w, g.AdjCfg, g) } func (g structGenerator) EmitNodeStyleType(w io.Writer) { // REVIEW: this appears to be standard even across kinds; can we extract it? doTemplate(` type _{{ .Type | TypeSymbol }}__Style struct{} func (_{{ .Type | TypeSymbol }}__Style) NewBuilder() ipld.NodeBuilder { var nb _{{ .Type | TypeSymbol }}__Builder nb.Reset() return &nb } `, w, g.AdjCfg, g) } // --- NodeBuilder and NodeAssembler ---> func (g structGenerator) GetNodeBuilderGenerator() NodeBuilderGenerator { return structBuilderGenerator{ g.AdjCfg, mixins.MapAssemblerTraits{ g.PkgName, g.TypeName, "_" + g.AdjCfg.TypeSymbol(g.Type) + "__", }, g.PkgName, g.Type, } } type structBuilderGenerator struct { AdjCfg *AdjunctCfg mixins.MapAssemblerTraits PkgName string Type schema.TypeStruct } func (g structBuilderGenerator) EmitNodeBuilderType(w io.Writer) { doTemplate(` type _{{ .Type | TypeSymbol }}__Builder struct { _{{ .Type | TypeSymbol }}__Assembler } `, w, g.AdjCfg, g) } func (g structBuilderGenerator) EmitNodeBuilderMethods(w io.Writer) { doTemplate(` func (nb *_{{ .Type | TypeSymbol }}__Builder) Build() ipld.Node { if nb.state != maState_finished { panic("invalid state: assembler for {{ .PkgName }}.{{ .Type.Name }} must be 'finished' before Build can be called!") } return nb.w } func (nb *_{{ .Type | TypeSymbol }}__Builder) Reset() { var w _{{ .Type | TypeSymbol }} var m schema.Maybe *nb = _{{ .Type | TypeSymbol }}__Builder{_{{ .Type | TypeSymbol }}__Assembler{w: &w, m: &m, state: maState_initial}} } `, w, g.AdjCfg, g) } func (g structBuilderGenerator) EmitNodeAssemblerType(w io.Writer) { // - 'w' is the "**w**ip" pointer. // - 'm' is the **m**aybe which communicates our completeness to the parent if we're a child assembler. // - 'state' is what it says on the tin. this is used for the map state (the broad transitions between null, start-map, and finish are handled by 'm' for consistency.) // - 's' is a bitfield for what's been **s**et. // - 'f' is the **f**ocused field that will be assembled next. // // - 'cm' is **c**hild **m**aybe and is used for the completion message from children that aren't allowed to be nullable (for those that are, their own maybe.m is used). // ('cm' could be elided for structs where all fields are maybes. trivial but not yet implemented.) // - the 'ca_*' fields embed **c**hild **a**ssemblers -- these are embedded so we can yield pointers to them without causing new allocations. doTemplate(` type _{{ .Type | TypeSymbol }}__Assembler struct { w *_{{ .Type | TypeSymbol }} m *schema.Maybe state maState s int f int cm schema.Maybe {{range $field := .Type.Fields -}} ca_{{ $field | FieldSymbolLower }} _{{ $field.Type | TypeSymbol }}__Assembler {{end -}} } var ( {{- $type := .Type -}} {{- /* ranging modifies dot, unhelpfully */ -}} {{- range $i, $field := .Type.Fields }} fieldBit__{{ $type | TypeSymbol }}_{{ $field | FieldSymbolUpper }} = 1 << {{ $i }} {{- end}} fieldBits__{{ $type | TypeSymbol }}_sufficient = 0 {{- range $i, $field := .Type.Fields }}{{if not $field.IsOptional }} + 1 << {{ $i }}{{end}}{{end}} ) `, w, g.AdjCfg, g) } func (g structBuilderGenerator) EmitNodeAssemblerMethodBeginMap(w io.Writer) { // We currently disregard sizeHint. It's not relevant to us. // We could check it strictly and emit errors; presently, we don't. // This method contains a branch to support MaybeUsesPtr because new memory may need to be allocated. // This allocation only happens if the 'w' ptr is nil, which means we're being used on a Maybe; // otherwise, the 'w' ptr should already be set, and we fill that memory location without allocating, as usual. doTemplate(` func (na *_{{ .Type | TypeSymbol }}__Assembler) BeginMap(int) (ipld.MapAssembler, error) { switch *na.m { case schema.Maybe_Value, schema.Maybe_Null: panic("invalid state: cannot assign into assembler that's already finished") case midvalue: panic("invalid state: it makes no sense to 'begin' twice on the same assembler!") } *na.m = midvalue {{- if .Type | MaybeUsesPtr }} if na.w == nil { na.w = &_{{ .Type | TypeSymbol }}{} } {{- end}} return na, nil } `, w, g.AdjCfg, g) } func (g structBuilderGenerator) EmitNodeAssemblerMethodAssignNull(w io.Writer) { doTemplate(` func (na *_{{ .Type | TypeSymbol }}__Assembler) AssignNull() error { switch *na.m { case allowNull: *na.m = schema.Maybe_Null return nil case schema.Maybe_Absent: return mixins.MapAssembler{"{{ .PkgName }}.{{ .TypeName }}"}.AssignNull() case schema.Maybe_Value, schema.Maybe_Null: panic("invalid state: cannot assign into assembler that's already finished") case midvalue: panic("invalid state: cannot assign null into an assembler that's already begun working on recursive structures!") } panic("unreachable") } `, w, g.AdjCfg, g) } func (g structBuilderGenerator) EmitNodeAssemblerMethodAssignNode(w io.Writer) { // AssignNode goes through three phases: // 1. is it null? Jump over to AssignNull (which may or may not reject it). // 2. is it our own type? Handle specially -- we might be able to do efficient things. // 3. is it the right kind to morph into us? Do so. // // We do not set m=midvalue in phase 3 -- it shouldn't matter unless you're trying to pull off concurrent access, which is wrong and unsafe regardless. doTemplate(` func (na *_{{ .Type | TypeSymbol }}__Assembler) AssignNode(v ipld.Node) error { if v.IsNull() { return na.AssignNull() } if v2, ok := v.(*_{{ .Type | TypeSymbol }}); ok { switch *na.m { case schema.Maybe_Value, schema.Maybe_Null: panic("invalid state: cannot assign into assembler that's already finished") case midvalue: panic("invalid state: cannot assign null into an assembler that's already begun working on recursive structures!") } {{- if .Type | MaybeUsesPtr }} if na.w == nil { na.w = v2 *na.m = schema.Maybe_Value return nil } {{- end}} *na.w = *v2 *na.m = schema.Maybe_Value return nil } if v.ReprKind() != ipld.ReprKind_Map { return ipld.ErrWrongKind{TypeName: "{{ .PkgName }}.{{ .Type.Name }}", MethodName: "AssignNode", AppropriateKind: ipld.ReprKindSet_JustMap, ActualKind: v.ReprKind()} } itr := v.MapIterator() for !itr.Done() { k, v, err := itr.Next() if err != nil { return err } if err := na.AssembleKey().AssignNode(k); err != nil { return err } if err := na.AssembleValue().AssignNode(v); err != nil { return err } } return na.Finish() } `, w, g.AdjCfg, g) } func (g structBuilderGenerator) EmitNodeAssemblerOtherBits(w io.Writer) { g.emitMapAssemblerChildTidyHelper(w) g.emitMapAssemblerMethods(w) g.emitKeyAssembler(w) } func (g structBuilderGenerator) emitMapAssemblerChildTidyHelper(w io.Writer) { // This function attempts to clean up the state machine to acknolwedge child assembly finish. // If the child was finished and we just collected it, return true and update state to maState_initial. // Otherwise, if it wasn't done, return false; // and the caller is almost certain to emit an error momentarily. // The function will only be called when the current state is maState_midValue. // (In general, the idea is that if the user is doing things correctly, // this function will only be called when the child is in fact finished.) // Most of the logic here is about nullables and not optionals, // because if you're an optional that's absent, you never got to value assembly. // There's still one branch for optionals, though, because they have a different residence for 'm' just as nullables do. // Child assemblers are expected to control their own state machines; // for values that have maybes, we never change their maybe state again, so the usual logic should hold; // for values that don't have maybes (and thus share 'cm')... // actually, we just let that fly. If the user does this, it's possible to mutate a subsequent field // with the assembler held from an earlier one, and while that's surely going to look bizarre, it's not unsafe per se. doTemplate(` func (ma *_{{ .Type | TypeSymbol }}__Assembler) valueFinishTidy() bool { switch ma.f { {{- range $i, $field := .Type.Fields }} case {{ $i }}: {{- if $field.IsNullable }} switch ma.w.{{ $field | FieldSymbolLower }}.m { case schema.Maybe_Null: ma.state = maState_initial return true case schema.Maybe_Value: {{- if (MaybeUsesPtr $field.Type) }} ma.w.{{ $field | FieldSymbolLower }}.v = ma.ca_{{ $field | FieldSymbolLower }}.w {{- end}} ma.state = maState_initial return true default: return false } {{- else if $field.IsOptional }} switch ma.w.{{ $field | FieldSymbolLower }}.m { case schema.Maybe_Value: {{- if (MaybeUsesPtr $field.Type) }} ma.w.{{ $field | FieldSymbolLower }}.v = ma.ca_{{ $field | FieldSymbolLower }}.w {{- end}} ma.state = maState_initial return true default: return false } {{- else}} switch ma.cm { case schema.Maybe_Value: {{- /* while defense in depth here might avoid some 'wat' outcomes, it's not strictly necessary for safety */ -}} {{- /* ma.ca_{{ $field | FieldSymbolLower }}.w = nil */ -}} {{- /* ma.ca_{{ $field | FieldSymbolLower }}.m = nil */ -}} ma.cm = schema.Maybe_Absent ma.state = maState_initial return true default: return false } {{- end}} {{- end}} default: panic("unreachable") } } `, w, g.AdjCfg, g) } func (g structBuilderGenerator) emitMapAssemblerMethods(w io.Writer) { // FUTURE: some of the setup of the child assemblers could probably be DRY'd up. doTemplate(` func (ma *_{{ .Type | TypeSymbol }}__Assembler) AssembleEntry(k string) (ipld.NodeAssembler, error) { switch ma.state { case maState_initial: // carry on case maState_midKey: panic("invalid state: AssembleEntry cannot be called when in the middle of assembling another key") case maState_expectValue: panic("invalid state: AssembleEntry cannot be called when expecting start of value assembly") case maState_midValue: if !ma.valueFinishTidy() { panic("invalid state: AssembleEntry cannot be called when in the middle of assembling a value") } // if tidy success: carry on case maState_finished: panic("invalid state: AssembleEntry cannot be called on an assembler that's already finished") } switch k { {{- $type := .Type -}} {{- /* ranging modifies dot, unhelpfully */ -}} {{- range $i, $field := .Type.Fields }} case "{{ $field.Name }}": if ma.s & fieldBit__{{ $type | TypeSymbol }}_{{ $field | FieldSymbolUpper }} != 0 { return nil, ipld.ErrRepeatedMapKey{&fieldName__{{ $type | TypeSymbol }}_{{ $field | FieldSymbolUpper }}} } ma.s += fieldBit__{{ $type | TypeSymbol }}_{{ $field | FieldSymbolUpper }} ma.state = maState_midValue {{- if $field.IsMaybe }} ma.ca_{{ $field | FieldSymbolLower }}.w = {{if not (MaybeUsesPtr $field.Type) }}&{{end}}ma.w.{{ $field | FieldSymbolLower }}.v ma.ca_{{ $field | FieldSymbolLower }}.m = &ma.w.{{ $field | FieldSymbolLower }}.m {{if $field.IsNullable }}ma.w.{{ $field | FieldSymbolLower }}.m = allowNull{{end}} {{- else}} ma.ca_{{ $field | FieldSymbolLower }}.w = &ma.w.{{ $field | FieldSymbolLower }} ma.ca_{{ $field | FieldSymbolLower }}.m = &ma.cm {{- end}} return &ma.ca_{{ $field | FieldSymbolLower }}, nil {{- end}} default: return nil, ipld.ErrInvalidKey{TypeName:"{{ .PkgName }}.{{ .Type.Name }}", Key:&_String{k}} } } func (ma *_{{ .Type | TypeSymbol }}__Assembler) AssembleKey() ipld.NodeAssembler { switch ma.state { case maState_initial: // carry on case maState_midKey: panic("invalid state: AssembleKey cannot be called when in the middle of assembling another key") case maState_expectValue: panic("invalid state: AssembleKey cannot be called when expecting start of value assembly") case maState_midValue: if !ma.valueFinishTidy() { panic("invalid state: AssembleKey cannot be called when in the middle of assembling a value") } // if tidy success: carry on case maState_finished: panic("invalid state: AssembleKey cannot be called on an assembler that's already finished") } ma.state = maState_midKey return (*_{{ .Type | TypeSymbol }}__KeyAssembler)(ma) } func (ma *_{{ .Type | TypeSymbol }}__Assembler) AssembleValue() ipld.NodeAssembler { switch ma.state { case maState_initial: panic("invalid state: AssembleValue cannot be called when no key is primed") case maState_midKey: panic("invalid state: AssembleValue cannot be called when in the middle of assembling a key") case maState_expectValue: // carry on case maState_midValue: panic("invalid state: AssembleValue cannot be called when in the middle of assembling another value") case maState_finished: panic("invalid state: AssembleValue cannot be called on an assembler that's already finished") } ma.state = maState_midValue switch ma.f { {{- range $i, $field := .Type.Fields }} case {{ $i }}: {{- if $field.IsMaybe }} ma.ca_{{ $field | FieldSymbolLower }}.w = {{if not (MaybeUsesPtr $field.Type) }}&{{end}}ma.w.{{ $field | FieldSymbolLower }}.v ma.ca_{{ $field | FieldSymbolLower }}.m = &ma.w.{{ $field | FieldSymbolLower }}.m {{if $field.IsNullable }}ma.w.{{ $field | FieldSymbolLower }}.m = allowNull{{end}} {{- else}} ma.ca_{{ $field | FieldSymbolLower }}.w = &ma.w.{{ $field | FieldSymbolLower }} ma.ca_{{ $field | FieldSymbolLower }}.m = &ma.cm {{- end}} return &ma.ca_{{ $field | FieldSymbolLower }} {{- end}} default: panic("unreachable") } } func (ma *_{{ .Type | TypeSymbol }}__Assembler) Finish() error { switch ma.state { case maState_initial: // carry on case maState_midKey: panic("invalid state: Finish cannot be called when in the middle of assembling a key") case maState_expectValue: panic("invalid state: Finish cannot be called when expecting start of value assembly") case maState_midValue: if !ma.valueFinishTidy() { panic("invalid state: Finish cannot be called when in the middle of assembling a value") } // if tidy success: carry on case maState_finished: panic("invalid state: Finish cannot be called on an assembler that's already finished") } //FIXME check if all required fields are set ma.state = maState_finished *ma.m = schema.Maybe_Value return nil } func (ma *_{{ .Type | TypeSymbol }}__Assembler) KeyStyle() ipld.NodeStyle { return _String__Style{} } func (ma *_{{ .Type | TypeSymbol }}__Assembler) ValueStyle(k string) ipld.NodeStyle { panic("todo structbuilder mapassembler valuestyle") } `, w, g.AdjCfg, g) } func (g structBuilderGenerator) emitKeyAssembler(w io.Writer) { doTemplate(` type _{{ .Type | TypeSymbol }}__KeyAssembler _{{ .Type | TypeSymbol }}__Assembler `, w, g.AdjCfg, g) stubs := mixins.StringAssemblerTraits{ g.PkgName, g.TypeName + ".KeyAssembler", "_" + g.AdjCfg.TypeSymbol(g.Type) + "__Key", } // This key assembler can disregard any idea of complex keys because it's a struct! // Struct field names must be strings (and quite simple ones at that). stubs.EmitNodeAssemblerMethodBeginMap(w) stubs.EmitNodeAssemblerMethodBeginList(w) stubs.EmitNodeAssemblerMethodAssignNull(w) stubs.EmitNodeAssemblerMethodAssignBool(w) stubs.EmitNodeAssemblerMethodAssignInt(w) stubs.EmitNodeAssemblerMethodAssignFloat(w) doTemplate(` func (ka *_{{ .Type | TypeSymbol }}__KeyAssembler) AssignString(k string) error { if ka.state != maState_midKey { panic("misuse: KeyAssembler held beyond its valid lifetime") } switch k { {{- $type := .Type -}} {{- /* ranging modifies dot, unhelpfully */ -}} {{- range $i, $field := .Type.Fields }} case "{{ $field.Name }}": if ka.s & fieldBit__{{ $type | TypeSymbol }}_{{ $field | FieldSymbolUpper }} != 0 { return ipld.ErrRepeatedMapKey{&fieldName__{{ $type | TypeSymbol }}_{{ $field | FieldSymbolUpper }}} } ka.s += fieldBit__{{ $type | TypeSymbol }}_{{ $field | FieldSymbolUpper }} ka.state = maState_expectValue ka.f = {{ $i }} {{- end}} default: return ipld.ErrInvalidKey{TypeName:"{{ .PkgName }}.{{ .Type.Name }}", Key:&_String{k}} } return nil } `, w, g.AdjCfg, g) stubs.EmitNodeAssemblerMethodAssignBytes(w) stubs.EmitNodeAssemblerMethodAssignLink(w) doTemplate(` func (ka *_{{ .Type | TypeSymbol }}__KeyAssembler) AssignNode(v ipld.Node) error { if v2, err := v.AsString(); err != nil { return err } else { return ka.AssignString(v2) } } func (_{{ .Type | TypeSymbol }}__KeyAssembler) Style() ipld.NodeStyle { return _String__Style{} } `, w, g.AdjCfg, g) }