package main import ( "fmt" "io" "os" "go.wit.com/log" "golang.org/x/text/cases" "golang.org/x/text/language" ) // this file is named poorly. It has more than Sort() func (pb *Files) makeNewSortfile(pf *File) error { wSort, _ := os.OpenFile(pf.Filebase+".sort.pb.go", os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0644) defer wSort.Close() header(wSort, pf) pf.syncLock(wSort) fmt.Fprintf(wSort, "// START SORT\n") fmt.Fprintf(wSort, "\n") // add iterators for all the structs to be used for _, msg := range pf.allMsg() { funcdef := newIter(wSort, msg) log.Printf("Adding %s\n", funcdef) } // log.Printf("START COMPARE\n") for _, msg := range pf.allMsg() { PARENT := msg.Name for _, v := range msg.Vars { if !v.HasSort { continue } if v.IsRepeated { // can't work against slices continue } VARNAME := v.VarName funcdef := newSortType(wSort, PARENT, VARNAME) log.Printf("Adding %s\n", funcdef) } } // add selectAll() functions for all the sort combinations that will be used for _, s := range pf.ToSort { // log.Printf("SORT: %-2d %20s %20s %20s %20s\n", i, s.MsgName, s.VarType, s.VarName, s.Lockname) FRUIT := s.MsgName APPLE := s.VarType APPLES := s.VarName LOCK := s.Lockname msg := pf.findMsg(s.VarType) if msg == nil { return fmt.Errorf("failed to find struct %s", s.VarType) } funcdef := msg.newIterAll(wSort, FRUIT, APPLE, APPLES, LOCK) log.Printf("Adding %s\n", funcdef) } // log.Printf("START SELECT\n") // make the sort iterators selectAll() for _, s := range pf.ToSort { PARENT := s.MsgName CHILD := s.VarType VARNAME := s.VarName LOCK := s.Lockname msg := pf.findMsg(s.VarType) if msg == nil { return fmt.Errorf("failed to find struct %s", s.VarType) } funcdef := msg.addSelectAll(wSort, PARENT, CHILD, VARNAME, LOCK) log.Printf("Adding %s\n", funcdef) } // log.Printf("END SELECT\n") // log.Printf("START SORT\n") // make the SortBy() functions for _, s := range pf.ToSort { // var funcname string PARENT := s.MsgName VARNAME := s.VarName msg := pf.findMsg(s.VarType) if msg == nil { return fmt.Errorf("failed to find struct %s", s.VarType) } for _, v := range msg.Vars { if v.IsRepeated { // can't work against slices continue } if v.HasSort { // funcname := "func (x *" + FUNCTYPE + ") SortBy" + v.VarName + "(" + v.VarType + ") *[]iter" + s.VarType // log.Printf(" %-2s %20s %20s %20s %s %s\n", "", "", "fix", "", "", funcname) // funcdef := "func (x *"+FRUIT+") SortBy"+COLOR+"() *"+APPLE+"Iterator" if v.VarType == "string" { var sortby string // func newSortBy(w io.Writer, STRUCT, ITER, SORTNAME, SORTBY, SELECT string) string { if PARENT == VARNAME { sortby = "SortBy" + v.VarName } else { sortby = "Sort" + VARNAME + "By" + v.VarName } sortname := s.VarType + v.VarName selectName := "selectAll" + VARNAME funcdef := newSortBy(wSort, PARENT, s.VarType, sortname, sortby, selectName, v.VarName) log.Printf("Adding %s\n", funcdef) } else { // funcdef := "func (x *" + FUNCTYPE + ") SortBy" + v.VarName + "(" + v.VarType + ") *[]iter" + s.VarType + " # can not do this yet" } } } } fmt.Fprintf(wSort, "\n") fmt.Fprintf(wSort, "// END SORT\n") // make Len() for _, msg := range pf.allMsg() { PARENT := msg.Name LOCK := msg.Lockname for _, v := range msg.Vars { if v.IsRepeated { VARNAME := v.VarName // funcdef := "func (x *" + PARENT + ") Len" + VARNAME + "() int " + CHILD + " name:" + VARNAME if PARENT == VARNAME { // special case funcdef := addLenFunc(wSort, PARENT, VARNAME, LOCK) // + " " + v.VarType + " name:" + v.VarName funcdef += " # is special struct=varname" log.Printf("Adding %s\n", funcdef) } if PARENT == VARNAME { // FUNCNAME := "Append" msg.simpleAppend(wSort, PARENT, v.VarName, v.VarType) // funcdef := msg.addAppendBy(wSort, PARENT, FUNCNAME, VARNAME, PARENT, s.VarType) // log.Printf(" %-2s %20s %20s %20s %s %s\n", "", "", "", "done", "", funcdef) } } } } log.Printf("\n") // add All() for _, s := range pf.ToSort { PARENT := s.MsgName CHILD := s.VarType VARNAME := s.VarName if PARENT == VARNAME { funcdef := addAllFunc(wSort, PARENT, CHILD, VARNAME) log.Printf("Adding %s\n", funcdef) } /* // todo: figure out how to do more All() functions correction var FUNCTYPE string if PARENT == VARNAME { FUNCTYPE = PARENT } else { FUNCTYPE = VARNAME } if s.VarType+"s" == s.VarName { funcname = "func (x *" + FUNCTYPE + ") All() *[]iter" + s.VarType } else { funcname = "func (x *" + FUNCTYPE + ") all" + s.VarName + "() *[]iter" + s.VarType } log.Printf(" %-2s %20s %20s %20s %s %s\n", "", "", "", "", "", funcname) */ } // add Find() Delete() Append() Insert() for _, s := range pf.ToSort { PARENT := s.MsgName VARNAME := s.VarName msg := pf.findMsg(s.VarType) if msg == nil { return fmt.Errorf("failed to find struct %s", s.VarType) } // find() for _, v := range msg.Vars { if v.IsRepeated { continue } if v.HasUnique { // funcname = "func (x *" + PARENT + ") Find" + VARNAME + "By" + v.VarName + "(a " + v.VarType + ") *" + s.VarType + "(using" + v.VarName + ")" // log.Printf(" %-2s %20s %20s %20s %s %s\n", "", "", "", "", "", funcname) var FUNCNAME string if PARENT == VARNAME { // special case because of the enforced .proto format FUNCNAME = "FindBy" + v.VarName } else { FUNCNAME = "Find" + VARNAME + "By" + v.VarName } funcdef := msg.generateFindBy(wSort, FUNCNAME, PARENT, VARNAME, s.VarType, v.VarName) // func (msg *MsgName) generateFindBy(w io.Writer, FUNCNAME, STRUCT, VARNAME, VARTYPE, COLOR string) string { log.Printf("Adding %s\n", funcdef) } else { if v.VarType == "string" { // probably dumb. move to a database if you need this // funcname = "func (x *" + PARENT + ") Find" + VARNAME + "By" + v.VarName + "(a string) []*" + s.VarType + " ???" // log.Printf(" %-2s %20s %20s %20s %s %s\n", "", "", "", "", "", funcname) } } } // delete() functions for _, v := range msg.Vars { if v.IsRepeated { continue } if !v.HasUnique { continue } // funcname := "func (x *" + PARENT + ") Delete" + VARNAME + "By" + v.VarName + "(" + v.VarType + ") bool" // log.Printf(" %-2s %20s %20s %20s %s %s\n", "", "", "", "", "", funcname) FRUIT := PARENT APPLES := VARNAME APPLE := v.VarName COLOR := v.VarType var FUNCNAME string if PARENT == VARNAME { // special case because of the enforced .proto format FUNCNAME = "DeleteBy" + v.VarName } else { FUNCNAME = "Delete" + VARNAME + "By" + v.VarName } var funcdef string if argv.Delete { funcdef = msg.deleteByWithCopy(wSort, FRUIT, APPLES, APPLE, COLOR, FUNCNAME, v.VarName) } else { funcdef = msg.deleteBy(wSort, FRUIT, APPLES, APPLE, COLOR, FUNCNAME, v.VarName) } log.Printf("Adding %s\n", funcdef) } // AppendBy() functions. todo: fix these so Append() is for simple things and Insert() is for unique keys var ucount int for _, v := range msg.Vars { if v.IsRepeated { continue } if !v.HasUnique { continue } ucount += 1 var FUNCNAME string if PARENT == VARNAME { // special case because of the enforced .proto format FUNCNAME = "AppendBy" + v.VarName } else { FUNCNAME = "Append" + VARNAME + "By" + v.VarName funcdef := msg.addAppendBy(wSort, PARENT, FUNCNAME, VARNAME, v.VarName, s.VarType) log.Printf("Adding %s\n", funcdef) } } // TODO: do this next // x *Repos) InsertPath( string) *Repo // returns existing record or new record if path != exists if ucount == 1 { for _, v := range msg.Vars { if !v.HasUnique { continue } //funcname = "func (x *" + PARENT + ") Insert(a *" + v.VarType + ") (*" + CHILD + ", isNew bool) // todo" //log.Printf("%20s %s\n", "", funcdef) } } if ucount > 1 { //funcname = "func (x *" + PARENT + ") Insert(a *" + CHILD + ") (*" + CHILD + ", isNew bool) // todo" //log.Printf("%20s %s\n", "", funcdef) } } return nil } func (pf *File) findMsg(s string) *MsgName { if pf.Bases.Name == s { return pf.Bases } if pf.Base.Name == s { return pf.Base } for _, msg := range pf.MsgNames { if msg.Name == s { return msg } } return nil } func (pf *File) allMsg() []*MsgName { var all []*MsgName all = append(all, pf.Bases) all = append(all, pf.Base) for _, msg := range pf.MsgNames { all = append(all, msg) } return all } func (pf *File) specialBases(wFind io.Writer) { var FRUIT string = cases.Title(language.English, cases.NoLower).String(pf.Bases.Name) var APPLES string = cases.Title(language.English, cases.NoLower).String(pf.Bases.Name) var APPLE string = cases.Title(language.English, cases.NoLower).String(pf.Base.Name) pf.Bases.simpleAppend(wFind, FRUIT, APPLES, APPLE) }