mapper.go 1.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687
  1. package orm
  2. import (
  3. "fmt"
  4. "reflect"
  5. "strings"
  6. "sync"
  7. )
  8. type tblmap map[string]*tbl
  9. func (t tblmap) String() string {
  10. s := make([]string, 0)
  11. for _, tbl := range t {
  12. s = append(s, tbl.String())
  13. }
  14. return strings.Join(s, "\n")
  15. }
  16. type mapper struct {
  17. lock *sync.Mutex
  18. wg *sync.WaitGroup
  19. tbls tblmap
  20. }
  21. var ctx *mapper = &mapper{
  22. lock: &sync.Mutex{},
  23. wg: &sync.WaitGroup{},
  24. tbls: make(tblmap),
  25. }
  26. func Map(tbl MappableInterface) *mapper {
  27. ctx.wg.Add(1)
  28. go func() {
  29. t := ctx.addTbl(tbl)
  30. if !t.isMapped() {
  31. ctx.mapTbl(t)
  32. }
  33. ctx.wg.Done()
  34. t.Unlock()
  35. }()
  36. return ctx
  37. }
  38. func WaitInit() {
  39. fmt.Println("Waiting in main")
  40. ctx.wg.Wait()
  41. fmt.Println(ctx.tbls)
  42. }
  43. func (m *mapper) addTbl(t MappableInterface) *tbl {
  44. rt := reflect.TypeOf(t).Elem()
  45. m.lock.Lock()
  46. defer m.lock.Unlock()
  47. if t, ok := m.tbls[rt.Name()]; ok {
  48. return t.ReadLock()
  49. }
  50. m.tbls[rt.Name()] = &tbl{
  51. rt: rt,
  52. rv: reflect.ValueOf(t).Elem(),
  53. lock: &sync.RWMutex{},
  54. tFn: t.GetTableMapperFn(),
  55. fFn: t.GetFieldMapperFn(),
  56. mapped: false,
  57. }
  58. return m.tbls[rt.Name()].WriteLock()
  59. }
  60. func (m *mapper) mapTbl(t *tbl) {
  61. for n := 0; n < t.getType().NumField(); n++ {
  62. switch t.getType().Kind() {
  63. case reflect.Ptr:
  64. case reflect.Struct:
  65. if f := t.getType().Field(n); !f.Anonymous {
  66. t.add(t.getType().Field(n))
  67. } else if t.getType().Field(n).Anonymous && t.getValue().Field(n).CanInterface() {
  68. t.add(t.getType().Field(n))
  69. }
  70. default:
  71. }
  72. }
  73. }