middleware_update_relation.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393
  1. package got
  2. import (
  3. "fmt"
  4. "text/template"
  5. api "git.eugeniocarvalho.dev/eugeniucarvalho/apicodegen/api"
  6. . "git.eugeniocarvalho.dev/eugeniucarvalho/apicodegen/common"
  7. // "github.com/davecgh/go-spew/spew"
  8. )
  9. var (
  10. updateRelationStmtsTmpl *template.Template
  11. updateRelationStmtsErr error
  12. )
  13. func init() {
  14. updateRelationStmtsTmpl, updateRelationStmtsErr = ParseTemplate(`
  15. func UpdateRelation{{.entity}}(options *api.Filter) {
  16. var (
  17. filter *api.Filter
  18. err *errs.Error
  19. entity = &models.{{.entity}}Reference{}
  20. wg = sync.WaitGroup{}
  21. )
  22. options.Entity = entity
  23. if _, err = models.Api.FindOne(options); err != nil {
  24. return
  25. }
  26. entity.Id = entity.Id.(primitive.ObjectID).Hex()
  27. {{range .relations}}
  28. wg.Add(1)
  29. go func(){
  30. defer wg.Done()
  31. filter = &api.Filter{
  32. Collection: "{{.Collection}}",
  33. DB: "{{.DB}}",
  34. Patchs: &bson.A{bson.M{"$set": bson.M{"{{.path}}": entity}}},
  35. Query: &bson.M{"{{.attr}}._id": entity.Id},
  36. }
  37. if _, err = models.Api.PatchMany(filter); err != nil {
  38. fmt.Println(err.Error())
  39. }
  40. }()
  41. {{end}}
  42. wg.Wait()
  43. }`)
  44. if updateRelationStmtsErr != nil {
  45. panic(updateRelationStmtsErr)
  46. }
  47. }
  48. func createUpdateRelationMethod(method *Method, context map[string]interface{}) (out string, err error) {
  49. var (
  50. indexMap = map[bool]string{
  51. true: ".$",
  52. false: "",
  53. }
  54. hasUpdateRelation = SR.Has(method.Entity)
  55. relations = []map[string]interface{}{}
  56. updRelation = UPDATE_RELATION + method.Entity
  57. _, updateRelationMethodDefined = relationFunctionDefined[updRelation]
  58. )
  59. context["hasUpdateRelation"] = hasUpdateRelation
  60. context["relations"] = &relations
  61. if hasUpdateRelation {
  62. if updateRelationMethodDefined {
  63. err = fmt.Errorf("function defined")
  64. return
  65. }
  66. relationFunctionDefined[updRelation] = true
  67. fmt.Printf("Generate update callback for entity '%s'\n", method.Entity)
  68. for _, relation := range SR.Get(method.Entity) {
  69. if relation.DB == "" || relation.Collection == "" {
  70. api.LogWarning(0, fmt.Sprintf("DB and Collection are required! %s -> %s", relation.Target, relation.Source))
  71. continue
  72. }
  73. relations = append(relations, map[string]interface{}{
  74. "DB": relation.DB,
  75. "Collection": relation.Collection,
  76. "attr": relation.Attr,
  77. "path": fmt.Sprintf("%s%s", relation.Attr, indexMap[relation.IsArray]),
  78. })
  79. }
  80. out, err = TemplateToString(updateRelationStmtsTmpl, context)
  81. }
  82. return
  83. }
  84. // var (
  85. // GenUpdateStmts = &Middleware{
  86. // Id: "implement",
  87. // Type: "method",
  88. // Fn: func(ctx *MiddlewareContext) error {
  89. // var (
  90. // project = ctx.Project
  91. // method = ctx.Method
  92. // dbName = project.GetEntityDB(method.Entity)
  93. // collectionName = project.GetCollection(method.Entity)
  94. // dependenceMethod = BASE_HAS_DEPENDE + method.Entity
  95. // updRelation = UPDATE_RELATION + method.Entity
  96. // indexMap = map[bool]string{
  97. // true: ".$",
  98. // false: "",
  99. // }
  100. // beforeSend = method.Hook("beforeSend")
  101. // beforePersist = method.Hook("beforePersist")
  102. // hasUpdateRelation = SR.Has(method.Entity)
  103. // _, updateRelationMethodDefined = relationFunctionDefined[updRelation]
  104. // relations = []map[string]interface{}{}
  105. // context = map[string]interface{}{
  106. // "dbName": dbName,
  107. // "collectionName": collectionName,
  108. // "entity": method.Entity,
  109. // "dependenceMethod": dependenceMethod,
  110. // "beforeSend": beforeSend,
  111. // "beforePersist": beforePersist,
  112. // "hasUpdateRelation": hasUpdateRelation,
  113. // "relations": &relations,
  114. // }
  115. // )
  116. // if beforePersist {
  117. // generateHookCall(project, method, "beforePersist")
  118. // }
  119. // if beforeSend {
  120. // generateHookCall(project, method, "beforeSend")
  121. // }
  122. // // Nome do metodo que verifica se a entidade tem dependencias
  123. // out, _ := TemplateToString(updateStmtsTmpl, context)
  124. // afterMethod := ctx.Statement.Block(G.Id(out)).Line()
  125. // return nil
  126. // },
  127. // }
  128. // )
  129. // j = G.Id("filter").Op("=").Op("&").Qual(API_URL, "Filter").Values(G.Dict{
  130. // // Especifica a collection
  131. // G.Id("DB"): G.Lit(relation.DB),
  132. // G.Id("Collection"): G.Lit(relation.Collection),
  133. // // Especifica a query
  134. // G.Id("Query"): G.Op("&").Qual(BSON, "M").Values(G.Dict{
  135. // G.Lit(attrId + "._id"): G.Id("f").Dot("Id").Dot("Hex").Call(),
  136. // }),
  137. // // G.Id("Entity"): G.Qual(BSON, "M").Values(G.Dict{
  138. // // G.Lit("$set"): G.Qual(BSON, "M").Values(G.Dict{
  139. // // G.Lit(attrId + index): G.Id("f").Dot("Entity"),
  140. // // }),
  141. // // }),
  142. // G.Id("Patchs"): G.Op("&").Qual(BSON, "A").Values(
  143. // G.Qual(BSON, "M").Values(G.Dict{
  144. // G.Lit("$set"): G.Qual(BSON, "M").Values(G.Dict{
  145. // G.Lit(attrId + index): G.Id("f").Dot("Entity"),
  146. // }),
  147. // }),
  148. // ),
  149. // }).Line().If(
  150. // G.List(G.Id("_"), G.Id("err")).Op("=").Id("Models").Dot("PatchMany").Call(
  151. // G.Id("filter"),
  152. // ),
  153. // G.Id("err").Op("!=").Id("nil"),
  154. // ).Block(
  155. // G.Qual("fmt", "Println").Call(G.Id("err").Dot("Error").Call()),
  156. // ).Line().Line()
  157. // k.Add(j)
  158. // .Do(func(s *G.Statement) {
  159. // _, found := relationFunctionDefined[updRelation]
  160. // if !createUpdateRelation || found {
  161. // return
  162. // }
  163. // s.Comment(updRelation + " atualiza as referências da entidade " + method.Entity).Line()
  164. // s.Func().Id(updRelation).Params(
  165. // G.Id("f").Op("*").Qual(API_URL, "Filter"),
  166. // ).Block(
  167. // G.Var().Defs(
  168. // G.Id("filter").Op("*").Qual(API_URL, "Filter"),
  169. // G.Id("err").Op("*").Qual(API_ERROR, "Error"),
  170. // ),
  171. // // atualiza o filtro para a entidade de referencia
  172. // G.Id("f").Dot("Entity").Op("=").Op("&").Id(method.Entity+"Reference").Values(),
  173. // // Carrega os dados do banco para atualizar nas entidades relacionadas
  174. // G.If(
  175. // G.List(G.Id("_"), G.Id("err")).Op("=").Id("Models").Dot("FindOne").Call(
  176. // G.Id("f"),
  177. // ),
  178. // G.Id("err").Op("!=").Id("nil"),
  179. // ).Block(
  180. // G.Return(),
  181. // ),
  182. // G.Do(func(k *G.Statement) { // Adiciona as regras de atualizacao
  183. // var (
  184. // j *G.Statement
  185. // index string
  186. // )
  187. // for _, relation := range SR.Get(method.Entity) {
  188. // if relation.DB == "" || relation.Collection == "" {
  189. // api.LogWarning(0, fmt.Sprintf("Relation DB or Collection can't be empty! %s -> %s", relation.Target, relation.Source))
  190. // continue
  191. // }
  192. // // spew.Dump(relation)
  193. // // panic("sss")
  194. // // e := project.GetSchema(relation)
  195. // attrId := relation.Attr
  196. // if relation.IsArray {
  197. // index = ".$"
  198. // } else {
  199. // index = ""
  200. // }
  201. // j = G.Id("filter").Op("=").Op("&").Qual(API_URL, "Filter").Values(G.Dict{
  202. // // Especifica a collection
  203. // G.Id("DB"): G.Lit(relation.DB),
  204. // G.Id("Collection"): G.Lit(relation.Collection),
  205. // // Especifica a query
  206. // G.Id("Query"): G.Op("&").Qual(BSON, "M").Values(G.Dict{
  207. // G.Lit(attrId + "._id"): G.Id("f").Dot("Id").Dot("Hex").Call(),
  208. // }),
  209. // // G.Id("Entity"): G.Qual(BSON, "M").Values(G.Dict{
  210. // // G.Lit("$set"): G.Qual(BSON, "M").Values(G.Dict{
  211. // // G.Lit(attrId + index): G.Id("f").Dot("Entity"),
  212. // // }),
  213. // // }),
  214. // G.Id("Patchs"): G.Op("&").Qual(BSON, "A").Values(
  215. // G.Qual(BSON, "M").Values(G.Dict{
  216. // G.Lit("$set"): G.Qual(BSON, "M").Values(G.Dict{
  217. // G.Lit(attrId + index): G.Id("f").Dot("Entity"),
  218. // }),
  219. // }),
  220. // ),
  221. // }).Line().If(
  222. // G.List(G.Id("_"), G.Id("err")).Op("=").Id("Models").Dot("PatchMany").Call(
  223. // G.Id("filter"),
  224. // ),
  225. // G.Id("err").Op("!=").Id("nil"),
  226. // ).Block(
  227. // G.Qual("fmt", "Println").Call(G.Id("err").Dot("Error").Call()),
  228. // ).Line().Line()
  229. // k.Add(j)
  230. // }
  231. // }),
  232. // )
  233. // })
  234. // var (
  235. // // UpdateRelationRotine *G.Statement
  236. // method = ctx.Method
  237. // project = ctx.Project
  238. // updRelation = UPDATE_RELATION + method.Entity
  239. // createUpdateRelation = false
  240. // entity = method.Request
  241. // responseEntity = GenericPart.ReplaceAllString(method.Response, "")
  242. // )
  243. // if entity == "" {
  244. // panic("Modelo de request não definido em UpdateStmt!")
  245. // }
  246. // if responseEntity == "" {
  247. // panic("Modelo de response não definido!")
  248. // }
  249. // ctx.Statement.Block(
  250. // // Declaracao das variaveis
  251. // G.Var().Defs(
  252. // // G.Id("err").Op("*").Qual(API_ERROR, "Error"),
  253. // G.Id("entity").Op("=").Op("&").Id(entity).Values(),
  254. // ).Line(),
  255. // // Fazendo o parse do body
  256. // G.If(
  257. // G.Id("err").Op("=").Qual(API_URL, "ReadJson").Call(G.Id("ctx"), G.Id("entity")),
  258. // G.Id("err").Op("!=").Id("nil"),
  259. // ).Block(
  260. // // G.Id("api").Dot("Falha").Call(
  261. // // G.Id("ctx"),
  262. // // G.Id("api").Dot("ErrGeneral"),
  263. // // G.Id("err").Dot("Error").Call(),
  264. // // G.Nil(),
  265. // // ),
  266. // // G.Return(G.Id("err")),
  267. // G.Return(),
  268. // ),
  269. // G.Do(func(s *G.Statement) {
  270. // entity := ctx.Project.EntityDesc(method.Entity)
  271. // if entity != nil && entity.HasMode {
  272. // s.Add(G.Id(`entity.SetMode("create")`))
  273. // }
  274. // }),
  275. // G.If(
  276. // G.Id("err").Op("=").Qual(API_URL, "Validate").Call(G.Id("entity")),
  277. // G.Id("err").Op("!=").Id("nil"),
  278. // ).Block(
  279. // // G.Return(G.Id("err")),
  280. // G.Return(),
  281. // ).Line(),
  282. // // Captura a base de values
  283. // G.Id("values").Op(":=").Id("ctx").Dot("Values").Call(),
  284. // // Gera as atribuicoes de variaveis que seram atualizadas (usuario e ultimo update quando existir)
  285. // G.Do(func(part *G.Statement) {
  286. // entity := ctx.Project.GetSchema(method.Entity)
  287. // user := false
  288. // for _, prop := range entity.Properties {
  289. // if def, ok := prop.Autogenerate["update"]; ok {
  290. // switch def.Type {
  291. // case "user":
  292. // if !user {
  293. // part.Add(G.Id(`user := values.Get("$user.ref").(*UserReference)`).Line())
  294. // user = true
  295. // }
  296. // part.Add(G.Id("entity").Dot(strings.Title(prop.ID)).Op("=").Id("user").Line())
  297. // case "now":
  298. // part.Add(G.Id("entity").Dot(strings.Title(prop.ID)).Op("=").Qual("time", "Now").Call().Id(".Unix()").Line())
  299. // }
  300. // }
  301. // }
  302. // }).Line(),
  303. // // Carrega o filtro do contexto para adicionar a entidade
  304. // G.Id("filter").Op(":=").Id("values").Dot("Get").Call(
  305. // G.Lit("$filter"),
  306. // ).Assert(G.Op("*").Qual(API_URL, "Filter")).Line(),
  307. // // Adiciona a entidade
  308. // G.Id("filter").Dot("DB").Op("=").Lit(ctx.Project.GetEntityDB(method.Entity)),
  309. // G.Id("filter").Dot("Collection").Op("=").Lit(ctx.Project.GetCollection(method.Entity)),
  310. // G.Id("filter").Dot("Entity").Op("=").Id("entity").Line(),
  311. // generateHookCall(project, method, "beforePersist"),
  312. // // Inseri a entidade na coleção e verifica se a operação ocorreu com exito
  313. // G.If(
  314. // G.List(G.Id("_"), G.Id("err")).Op("=").Id("Models").Dot("UpdateOne").Call(
  315. // // G.Lit(ctx.Project.GetCollection(method.Entity)),
  316. // G.Id("filter"),
  317. // ),
  318. // G.Id("err").Op("!=").Id("nil"),
  319. // ).Block(
  320. // // G.Id("api").Dot("Falha").Call(
  321. // // G.Id("ctx"),
  322. // // G.Id("api").Dot("ErrGeneral"),
  323. // // G.Id("err").Dot("Error").Call(),
  324. // // G.Nil(),
  325. // // ),
  326. // // bson.IsObjectIdHex(m.Id.Hex())
  327. // // G.Return(G.Id("err")),
  328. // G.Return(),
  329. // ),
  330. // // Cria a rotina de atualização de relacionamentos
  331. // G.Do(func(s *G.Statement) {
  332. // if SR.Has(method.Entity) {
  333. // createUpdateRelation = true
  334. // relationFunctionDefined[updRelation] = true
  335. // s.Comment("Cria uma thread que executa a atualizaca das referências.").Line().Go().Func().Params().Block(
  336. // G.Id(updRelation).Call(G.Id("filter")),
  337. // ).Call()
  338. // }
  339. // }),
  340. // // Envia a resposta pro usuario em caso de sucesso
  341. // // G.Line().Id("resp").Op("=").Id(responseEntity).Values(G.Dict{
  342. // // G.Id("Entity"): G.Id("entity"),
  343. // // }),
  344. // G.Line().Id("resp").Op("=").Id("entity"),
  345. // generateHookCall(project, method, "beforeSend"),
  346. // // G.Return(G.Nil()),
  347. // G.Return(),
  348. // ).Line()