models.go 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925
  1. package common
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "fmt"
  6. "regexp"
  7. "strconv"
  8. "strings"
  9. "text/template"
  10. "git.eugeniocarvalho.dev/eugeniucarvalho/apicodegen/api"
  11. "git.eugeniocarvalho.dev/eugeniucarvalho/apicodegen/api/errs"
  12. ts "git.eugeniocarvalho.dev/eugeniucarvalho/gg/generators/typescript"
  13. "github.com/kataras/iris/v12/context"
  14. "go.mongodb.org/mongo-driver/bson/primitive"
  15. )
  16. const (
  17. BSON = "go.mongodb.org/mongo-driver/bson"
  18. BSONX = "go.mongodb.org/mongo-driver/x/bsonx"
  19. MONGO = "go.mongodb.org/mongo-driver/mongo"
  20. BSON_PRIMITIVE = "go.mongodb.org/mongo-driver/bson/primitive"
  21. IRIS_CTX = "github.com/kataras/iris/v12/context"
  22. IRIS = "github.com/kataras/iris/v12"
  23. UPDATE_RELATION = "UpdateRelation"
  24. BASE_HAS_DEPENDE = "HasDep"
  25. )
  26. var (
  27. API_URL = "git.eugeniocarvalho.dev/eugeniucarvalho/apicodegen/api"
  28. API_ERROR = "git.eugeniocarvalho.dev/eugeniucarvalho/apicodegen/api/errs"
  29. CODE_GEN_V2_COMMON = "git.eugeniocarvalho.dev/eugeniucarvalho/apicodegen/common"
  30. CODE_GEN_V2_AUTHORIZATION = "git.eugeniocarvalho.dev/eugeniucarvalho/apicodegen/authorization"
  31. // Variavel de controle de acesso aos models da API.
  32. Models = &api.Mongo{}
  33. camelToUnderRegex = regexp.MustCompile(`([^[:lower:]])`)
  34. //Generic e
  35. Generic = regexp.MustCompile("(?P<type>[\\w-_]+)<(?P<dtype>[\\w\\*]+)>")
  36. //GenericPart e
  37. GenericPart = regexp.MustCompile("<(?P<id>[\\w\\*]+)>")
  38. //ImportMap e
  39. importMap = map[string]string{
  40. "bson": BSON,
  41. "primitive": BSON_PRIMITIVE,
  42. }
  43. SR = SchemasRelations{
  44. R: map[string][]*Relation{},
  45. }
  46. )
  47. type BuildOptions struct {
  48. Mode string
  49. IgnoreBuildSteps string
  50. IgnoreBuildStepsValues map[int]bool
  51. }
  52. func (b *BuildOptions) IgnoreStep(step int) bool {
  53. _, ok := b.IgnoreBuildStepsValues[step]
  54. return ok
  55. }
  56. func (b *BuildOptions) Parse() error {
  57. var (
  58. value int
  59. err error
  60. )
  61. if b.IgnoreBuildStepsValues == nil {
  62. b.IgnoreBuildStepsValues = map[int]bool{}
  63. }
  64. for _, v := range strings.Split(b.IgnoreBuildSteps, ",") {
  65. if value, err = strconv.Atoi(v); err != nil {
  66. return err
  67. }
  68. b.IgnoreBuildStepsValues[value] = true
  69. }
  70. return nil
  71. }
  72. func ImportMap(base string) string {
  73. if v, ok := importMap[base]; ok {
  74. return v
  75. }
  76. panic(fmt.Sprintf("Import %s não definido", base))
  77. }
  78. type Project struct {
  79. OutPath string `json:"outPath"`
  80. Package string `json:"package"`
  81. Kind string `json:"kind"`
  82. Etag string `json:"etag"`
  83. Version string `json:"version"`
  84. BuildVersion string `json:"buildVersion"`
  85. ID string `json:"id"`
  86. Name string `json:"name"`
  87. DataBaseSufix string `json:"dataBaseSufix"`
  88. Mode string `json:"mode"`
  89. Revision string `json:"revision"`
  90. Title string `json:"title"`
  91. Description string `json:"description"`
  92. OwnerDomain string `json:"ownerDomain"`
  93. OwnerName string `json:"ownerName"`
  94. DocumentationLink string `json:"documentationLink"`
  95. Protocol string `json:"protocol"`
  96. BaseURL string `json:"baseUrl"`
  97. BasePath string `json:"basePath"`
  98. Middlewares []string `json:"middlewares"`
  99. ServicePath string `json:"servicePath"`
  100. GitRepository string `json:"git.repository"`
  101. Environment Environment `json:"environment"`
  102. Variables map[string]interface{} `json:"variables"`
  103. Resource *Resource `json:"-"`
  104. Schemas []*Entity `json:"schemas"`
  105. SchemasRef map[string]*Entity `json:"-"`
  106. Resources []*Resource `json:"resources"`
  107. Auth Auth `json:"auth"`
  108. TypeScriptSource *ts.File `json:"-"`
  109. Icons map[string]string `json:"icons"`
  110. ReplaceWhenEmpty map[string]bool `json:"ReplaceWhenEmpty"`
  111. OmitEmpty map[string]bool `json:"omitempty"`
  112. Clients []*Client `json:"clients,omitempty"`
  113. Translators map[string]TranslationFn `json:"-"`
  114. FormatMap map[string]string `json:"-"`
  115. Queries *QueryDef `json:"queries"`
  116. ACL *ACL `json:"acl"`
  117. Custom map[string]interface{} `json:"custom"`
  118. }
  119. type ACL struct {
  120. Roles []*Role `json:"roles"`
  121. Permissions []*Permission `json:"permissions"`
  122. }
  123. type QueryDef struct {
  124. Blacklistwords map[string][]string `json:"blacklistwords"`
  125. Queries map[string]string `json:"queries"`
  126. Common map[string]string `json:"common"`
  127. }
  128. type Role struct {
  129. Title string `json:"title"`
  130. Description string `json:"description"`
  131. ID string `json:"id"`
  132. AllowRemove bool `json:"allowRemove,omitempty"`
  133. Permissions []string `json:"permissions"`
  134. }
  135. type Permission struct {
  136. Title string `json:"title"`
  137. Description string `json:"description"`
  138. ID string `json:"id"`
  139. }
  140. type Client struct {
  141. Id string `json:"id,omitempty"`
  142. OutputDir string `json:"outputDir,omitempty"`
  143. }
  144. type Auth struct {
  145. AuthCookieDomain string `json:"authCookieDomain"`
  146. AuthTokenID string `json:"authTokenId"`
  147. Oauth2 Oauth2 `json:"oauth2"`
  148. }
  149. type Oauth2 struct {
  150. URI string `json:"uri"`
  151. Client Oauth2Client `json:"client"`
  152. Scopes []Scope `json:"scopes"`
  153. }
  154. type Oauth2Client struct {
  155. RedirectURI string `json:"redirect_uri"`
  156. ClientID string `json:"client_id"`
  157. ClientSecret string `json:"client_secret"`
  158. Scope []string `json:"scope"`
  159. }
  160. type Scope struct {
  161. ID string `json:"id"`
  162. PromptToUser []string `json:"promptToUser"`
  163. Description string `json:"description"`
  164. }
  165. type EnvironmentVariable struct {
  166. ID string `json:"id"`
  167. CamelID string `json:"-"`
  168. Default string `json:"default"`
  169. Required bool `json:"required,omitempty"`
  170. Description string `json:"description"`
  171. }
  172. type Environment map[string]*EnvironmentVariable
  173. type Entity struct {
  174. HasMode bool `json:"hasMode"`
  175. ID string `json:"id"`
  176. Type string `json:"type"`
  177. Description string `json:"description"`
  178. Collection string `json:"collection"`
  179. DB string `json:"db"`
  180. Extends []string `json:"extends"`
  181. Properties []*Propertie `json:"properties"`
  182. Representations map[string][]string `json:"representations"`
  183. Custom map[string]interface{} `json:"custom"`
  184. }
  185. type Resource struct {
  186. ID string `json:"id"`
  187. Description string `json:"description"`
  188. Entity string `json:"entity"`
  189. Formats []*Value `json:"formats"`
  190. Methods []*Method `json:"methods"`
  191. CommonParams map[string]*Parameter `json:"commonParams"`
  192. Custom map[string]interface{} `json:"custom"`
  193. }
  194. type Method struct {
  195. ID string `json:"id"`
  196. Entity string `json:"entity"`
  197. Type string `json:"type"` // Assume valores {one, list, implement}
  198. Path string `json:"path"`
  199. Template string `json:"template"`
  200. BeforePersistAction bool `json:"beforePersistAction"`
  201. HttpMethod string `json:"httpMethod"`
  202. Description string `json:"description"`
  203. Response string `json:"response"`
  204. Request string `json:"request"`
  205. Scopes []string `json:"scopes"`
  206. Middlewares []string `json:"middlewares"`
  207. Postresponse []string `json:"postresponse"`
  208. ParameterOrder []string `json:"parameterOrder"`
  209. ParametersString []string `json:"parameters"`
  210. Resource *Resource `json:"-"`
  211. Hooks map[string]bool `json:"hooks"`
  212. Parameters map[string]*Parameter `json:"parametersmap"`
  213. Preconditions []Action `json:"preconditions"`
  214. BeforeResponse []Action `json:"beforeResponse"`
  215. Custom map[string]interface{} `json:"custom"`
  216. // Parameters map[string]*Parameter `json:"parameters"`
  217. }
  218. type Action struct {
  219. ID string `json:"id"`
  220. Context map[string]interface{} `json:"context"`
  221. }
  222. type Parameter struct {
  223. ID string `json:"id"`
  224. Type string `json:"type"`
  225. Required bool `json:"required"`
  226. Description string `json:"description"`
  227. Default string `json:"default"`
  228. Location string `json:"location"`
  229. ConvertTo string `json:"convertTo"`
  230. Custom map[string]interface{} `json:"custom"`
  231. // Validation *ValidationRule `json:"validation"`
  232. Validation map[string]interface{} `json:"validation"`
  233. }
  234. // type ValidationRule struct {
  235. // Accept []*Value `json:"-"`
  236. // AcceptRef []string `json:"accept"`
  237. // Reject []*Value `json:"reject"`
  238. // RejectRef []string `json:"-"`
  239. // In []string `json:"in"`
  240. // Contains string `json:"contains"`
  241. // Regex string `json:"regex"`
  242. // Min string `json:"min"`
  243. // Max string `json:"max"`
  244. // Type string `json:"type"`
  245. // }
  246. type Value struct {
  247. Id string `json:"id"`
  248. Value string `json:"value"`
  249. Default bool `json:"default"`
  250. Fields string `json:"fields"`
  251. Description string `json:"description"`
  252. }
  253. type Propertie struct {
  254. ID string `json:"id"`
  255. Name string `json:"name"`
  256. Type string `json:"type"`
  257. Description string `json:"description"`
  258. AutogenerateInput string `json:"autogenerate"`
  259. Autogenerate map[string]AutoGenDef `json:"-"`
  260. Targets string `json:"targets"`
  261. Array bool `json:"array"`
  262. Relation bool `json:"relation"`
  263. TagVisited bool `json:"-"`
  264. Reference bool `json:"reference"`
  265. Readonly bool `json:"readonly"`
  266. Unique bool `json:"uniq"`
  267. Default interface{} `json:"default"`
  268. Enum []string `json:"enum"`
  269. Values []interface{} `json:"values"`
  270. EnumDescriptions []string `json:"enumDescriptions"`
  271. Tags map[string]string `json:"tags"`
  272. Filter []*Filter `json:"filter"`
  273. Custom map[string]interface{} `json:"custom"`
  274. }
  275. type AutoGenDef struct {
  276. Type string
  277. Args []string
  278. }
  279. type Filter struct {
  280. Path string `json:"path"`
  281. Type string `json:"type"`
  282. Label string `json:"label"`
  283. UserEnumAsOptions bool `json:"userEnumAsOptions"`
  284. Multiples bool `json:"multiples"`
  285. Options []FilterOption `json:"options,omitempty"`
  286. }
  287. type FilterOption struct {
  288. Value interface{} `json:"value"`
  289. Label string `json:"label"`
  290. }
  291. type ApiFilter struct {
  292. Id string `json:"id"`
  293. Date int64 `json:"date"`
  294. Fields []*Filter `json:"fields"`
  295. }
  296. func NewApiFilter(id string) *ApiFilter {
  297. return &ApiFilter{
  298. Id: id,
  299. Fields: []*Filter{},
  300. }
  301. }
  302. func RequestParams(args string, params map[string]*Parameter) func(ctx context.Context) (resp interface{}, err *errs.Error) {
  303. argsList := strings.Split(args, ",")
  304. return func(ctx context.Context) (resp interface{}, err *errs.Error) {
  305. var (
  306. values = ctx.Values()
  307. id string
  308. value interface{}
  309. sourceValue interface{}
  310. param *Parameter
  311. paramsMap = map[string]interface{}{}
  312. )
  313. values.Set("$params", paramsMap)
  314. for _, arg := range argsList {
  315. param = params[arg]
  316. switch param.Location {
  317. case "query":
  318. id = "query." + arg
  319. value = api.Q(ctx, arg, param.Default)
  320. case "path":
  321. id = "path." + arg
  322. value = api.P(ctx, arg, param.Default)
  323. }
  324. sourceValue = value
  325. emptyValue := (value == "" || value == nil)
  326. if param.Required && emptyValue {
  327. invalidArgument := errs.InvalidArgument()
  328. invalidArgument.Message = fmt.Sprintf(
  329. "ParamRequired: param '%s' in '%s'",
  330. param.ID,
  331. param.Location,
  332. )
  333. return nil, invalidArgument
  334. }
  335. if !emptyValue && param.ConvertTo != "" {
  336. if value, err = convertValueByType(param.ConvertTo, value); err != nil {
  337. invalidArgument := errs.InvalidArgument()
  338. invalidArgument.Message = fmt.Sprintf(
  339. "ParamTypeConversionError: param '%s' in '%s' with value '%v'. Waiting a %s ",
  340. param.ID,
  341. param.Location,
  342. value,
  343. param.ConvertTo,
  344. )
  345. return nil, invalidArgument
  346. }
  347. }
  348. if param.Validation != nil {
  349. for validator, args := range param.Validation {
  350. if fn, found := validationParamFunctions[validator]; found {
  351. ctx.Application().Logger().Info(fmt.Sprintf("validadete[%s][%s][%v]", validator, args, value))
  352. if err = fn(param, value, args); err != nil {
  353. return nil, err
  354. }
  355. }
  356. }
  357. }
  358. values.Set(fmt.Sprintf("%s_conv", id), value)
  359. values.Set(id, sourceValue)
  360. paramsMap[fmt.Sprintf("%s_conv", arg)] = value
  361. paramsMap[arg] = sourceValue
  362. }
  363. ctx.Next()
  364. return
  365. }
  366. }
  367. var (
  368. convertionTypeFunctions = map[string]func(interface{}) (interface{}, *errs.Error){
  369. "ObjectID": stringToObjectId,
  370. "bool": stringToBool,
  371. "int": stringToInt,
  372. "number": stringToFloat,
  373. }
  374. validationParamFunctions = map[string]func(*Parameter,interface{}, interface{}) *errs.Error{
  375. "min": func(param *Parameter, value interface{}, minString interface{}) *errs.Error {
  376. var input float64
  377. if v, ok := value.(int64); ok {
  378. input = float64(v)
  379. } else if v, ok := value.(float64); ok {
  380. input = v
  381. } else if v, ok := value.(string); ok {
  382. input = float64(len(v))
  383. } else {
  384. invalidArgument := errs.InvalidArgument()
  385. invalidArgument.Message = fmt.Sprintf(
  386. "[%s] ValueRestriction: mim validation requires (int,float,string)",
  387. param.ID,
  388. )
  389. return invalidArgument
  390. }
  391. if min, convert := minString.(float64); !convert || input < min {
  392. invalidArgument := errs.InvalidArgument()
  393. invalidArgument.Message = fmt.Sprintf(
  394. "[%s] ValueRestriction: value > %v. Received (%v)",
  395. param.ID,
  396. minString,
  397. value,
  398. )
  399. return invalidArgument
  400. }
  401. return nil
  402. },
  403. "max": func(param *Parameter, value interface{}, maxString interface{}) *errs.Error {
  404. var input float64
  405. if v, ok := value.(int64); ok {
  406. input = float64(v)
  407. } else if v, ok := value.(float64); ok {
  408. input = v
  409. } else if v, ok := value.(string); ok {
  410. input = float64(len(v))
  411. } else {
  412. invalidArgument := errs.InvalidArgument()
  413. invalidArgument.Message = fmt.Sprintf(
  414. "[%s] ValueRestriction: mim validation requires (int,float,string)",
  415. param.ID,
  416. )
  417. return invalidArgument
  418. }
  419. if max, convert := maxString.(float64); !convert || input > max {
  420. invalidArgument := errs.InvalidArgument()
  421. invalidArgument.Message = fmt.Sprintf(
  422. "[%s] ValueRestriction: value < %v. Received (%v)",
  423. param.ID,
  424. maxString,
  425. value,
  426. )
  427. return invalidArgument
  428. }
  429. return nil
  430. },
  431. "accept": func(param *Parameter, input interface{}, accept interface{}) *errs.Error {
  432. var (
  433. acceptValues = accept.([]interface{})
  434. acceptValuesString = []string{}
  435. value = fmt.Sprintf("%v", input)
  436. )
  437. for _, acceptValue := range acceptValues {
  438. if value == acceptValue.(string) {
  439. return nil
  440. }
  441. acceptValuesString = append(acceptValuesString, acceptValue.(string))
  442. }
  443. invalidArgument := errs.InvalidArgument()
  444. invalidArgument.Message = fmt.Sprintf(
  445. "[%s] ValueRestriction: '%s' isn't accept. Accept [%s]",
  446. param.ID,
  447. value,
  448. strings.Join(acceptValuesString, ","),
  449. )
  450. return invalidArgument
  451. },
  452. "reject": func(param *Parameter, input interface{}, reject interface{}) *errs.Error {
  453. var (
  454. rejectValues = reject.([]interface{})
  455. value = fmt.Sprintf("%v", input)
  456. )
  457. for _, rejectValue := range rejectValues {
  458. if value == rejectValue.(string) {
  459. invalidArgument := errs.InvalidArgument()
  460. invalidArgument.Message = fmt.Sprintf(
  461. "[%s] ValueRestriction: '%s' isn't accept",
  462. param.ID,
  463. value,
  464. )
  465. return invalidArgument
  466. }
  467. }
  468. return nil
  469. },
  470. "regex": func(param *Parameter, input interface{}, regex interface{}) *errs.Error {
  471. var (
  472. regexString = regex.(string)
  473. value = input.(string)
  474. )
  475. regexInstance := regexp.MustCompile(regexString)
  476. if !regexInstance.Match([]byte(value)) {
  477. invalidArgument := errs.InvalidArgument()
  478. invalidArgument.Message = fmt.Sprintf(
  479. "[%s] ValueRestriction: '%s' isn't accept",
  480. param.ID,
  481. value,
  482. )
  483. return invalidArgument
  484. }
  485. return nil
  486. },
  487. }
  488. )
  489. func stringToObjectId(value interface{}) (interface{}, *errs.Error) {
  490. var (
  491. valueString = value.(string)
  492. valueObjectID primitive.ObjectID
  493. err error
  494. )
  495. if valueObjectID, err = primitive.ObjectIDFromHex(valueString); err != nil {
  496. invalidArgument := errs.InvalidArgument()
  497. invalidArgument.Message = fmt.Sprintf("The value '%s' is'nt a valid ObjectId", valueString)
  498. return nil, invalidArgument
  499. }
  500. return valueObjectID, nil
  501. }
  502. func stringToBool(value interface{}) (interface{}, *errs.Error) {
  503. var (
  504. valueBool bool
  505. err error
  506. )
  507. if valueBool, err = strconv.ParseBool(value.(string)); err != nil {
  508. invalidArgument := errs.InvalidArgument()
  509. invalidArgument.Message = fmt.Sprintf("The value '%s' is'nt a valid boolean. Accept [true,1,T,false,0,F]", valueBool)
  510. return nil, invalidArgument
  511. }
  512. return valueBool, nil
  513. }
  514. func stringToInt(value interface{}) (interface{}, *errs.Error) {
  515. var (
  516. valueInt int64
  517. err error
  518. )
  519. if valueInt, err = strconv.ParseInt(value.(string), 10, 64); err != nil {
  520. invalidArgument := errs.InvalidArgument()
  521. invalidArgument.Message = fmt.Sprintf("The value '%s' is'nt a valid int", valueInt)
  522. return nil, invalidArgument
  523. }
  524. return valueInt, nil
  525. }
  526. func stringToFloat(value interface{}) (interface{}, *errs.Error) {
  527. var (
  528. valueFloat float64
  529. err error
  530. )
  531. if valueFloat, err = strconv.ParseFloat(value.(string), 64); err != nil {
  532. invalidArgument := errs.InvalidArgument()
  533. invalidArgument.Message = fmt.Sprintf("The value '%s' is'nt a valid number", valueFloat)
  534. return nil, invalidArgument
  535. }
  536. return valueFloat, nil
  537. }
  538. func convertValueByType(typ string, value interface{}) (interface{}, *errs.Error) {
  539. var err *errs.Error
  540. if fn, found := convertionTypeFunctions[typ]; found {
  541. if value, err = fn(value); err != nil {
  542. return nil, err
  543. }
  544. }
  545. return value, nil
  546. }
  547. // func validateParam(param *Parameter, value interface{}) (interface{}, *errs.Error) {
  548. // var err *errs.Error
  549. // return value, nil
  550. // }
  551. func (t *Method) Hook(id string) bool {
  552. // active := t.Hooks[id]
  553. // return active
  554. return t.Hooks[id]
  555. }
  556. func (t *Propertie) ParseAutogenerate() error {
  557. if t.AutogenerateInput != "" {
  558. parts := strings.Split(t.AutogenerateInput, ":")
  559. if len(parts) < 2 {
  560. return fmt.Errorf("Invalid autogenerate input '%s' in attribute '%s'.", t.AutogenerateInput, t.ID)
  561. }
  562. if t.Autogenerate == nil {
  563. t.Autogenerate = map[string]AutoGenDef{}
  564. }
  565. args := strings.Split(parts[1], "#")
  566. for _, k := range strings.Split(parts[0], ",") {
  567. t.Autogenerate[k] = AutoGenDef{
  568. Type: args[0],
  569. Args: args[1:],
  570. }
  571. }
  572. }
  573. return nil
  574. }
  575. type SchemasRelations struct {
  576. R map[string][]*Relation
  577. }
  578. type Relation struct {
  579. Source string
  580. Target string
  581. Attr string
  582. Collection string
  583. DB string
  584. IsArray bool
  585. }
  586. type EntityInfo struct {
  587. Name string
  588. Origin string
  589. NewName string
  590. DynamicType string
  591. DynamicTypeId string
  592. IsGeneric bool
  593. }
  594. type TranslationFn func(p *Project) error
  595. func (p *Project) Build(b *BuildOptions) error {
  596. var err error
  597. for _, c := range p.Clients {
  598. if fn, found := p.Translators[c.Id]; found {
  599. if err = fn(p); err != nil {
  600. fmt.Println("error on ", c.Id)
  601. return err
  602. }
  603. } else {
  604. return fmt.Errorf("Middleware '%s' not defined!", c.Id)
  605. }
  606. }
  607. // fmt.Println("--- RunBuildCommads")
  608. return RunBuildCommads(p, b)
  609. }
  610. func (p *Project) OutDirectory(path string) {
  611. p.OutPath = path
  612. }
  613. func (p *Project) Client(id string) *Client {
  614. for _, c := range p.Clients {
  615. if c.Id == id {
  616. return c
  617. }
  618. }
  619. return nil
  620. }
  621. func (p *Project) Save(path string) error {
  622. data, err := json.MarshalIndent(p, "", " ")
  623. if err == nil {
  624. err = FilePutContentsBytes(path, data, 0777)
  625. }
  626. return err
  627. }
  628. func (p *Project) GetCollection(entity string) string {
  629. for _, e := range p.Schemas {
  630. if e.ID == entity {
  631. return e.Collection
  632. }
  633. }
  634. return "undefined"
  635. }
  636. func getCustom(options map[string]interface{}, path string) (resp interface{}) {
  637. if options != nil {
  638. resp = options[path]
  639. }
  640. return
  641. }
  642. func (p *Project) GetEntityDB(entity string) string {
  643. if en, found := p.SchemasRef[entity]; found {
  644. return en.DB + p.DataBaseSufix
  645. }
  646. panic(fmt.Sprintf("DB attribute is empty in entity '%s'", entity))
  647. }
  648. func (p *Project) EntityDesc(ID string) *Entity {
  649. if _, y := p.SchemasRef[ID]; !y {
  650. fmt.Println("EntityDesc(ID)", ID)
  651. return nil
  652. }
  653. return p.SchemasRef[ID]
  654. }
  655. func (m *Method) HasPathParams() bool {
  656. return len(m.ParameterOrder) > 0
  657. }
  658. func (m *Method) HasFormatParam() (bool, *Parameter) {
  659. for id, param := range m.Parameters {
  660. // param = m.Parameters[id]
  661. // fmt.Println("param:", param.ID)
  662. if id == "format" {
  663. return true, param
  664. }
  665. }
  666. return false, nil
  667. }
  668. func (p *Project) GetUrlFromMethod(method *Method) string {
  669. return p.BaseURL + method.Path
  670. }
  671. func (p *Project) ResponseEntity(property string) *EntityInfo {
  672. var (
  673. pi = &EntityInfo{
  674. Origin: property,
  675. }
  676. )
  677. match := Generic.FindStringSubmatch(property)
  678. if len(match) == 0 {
  679. return pi
  680. }
  681. for i, name := range Generic.SubexpNames() {
  682. switch name {
  683. case "type":
  684. pi.Name = match[i]
  685. case "dtype":
  686. pi.DynamicType = match[i]
  687. pi.IsGeneric = true
  688. }
  689. }
  690. if pi.IsGeneric {
  691. entity := p.GetSchema(pi.Name)
  692. match = GenericPart.FindStringSubmatch(entity.ID)
  693. for i, name := range GenericPart.SubexpNames() {
  694. switch name {
  695. case "id":
  696. pi.DynamicTypeId = match[i]
  697. }
  698. }
  699. }
  700. pi.NewName = pi.Name + UpFirst(strings.Replace(pi.DynamicType, "*", "", -1))
  701. return pi
  702. }
  703. func (p *Project) GetPath(m *Method) string {
  704. path := []byte(p.BasePath + m.Path)
  705. for attr, param := range m.Parameters {
  706. path = regexp.MustCompile("{"+attr+"}").ReplaceAll(path, []byte("{"+attr+":"+param.Type+"}"))
  707. }
  708. return string(path)
  709. }
  710. func (p *Project) GetSchema(id string) *Entity {
  711. id = strings.Replace(id, "*", "", -1)
  712. if model, ok := p.SchemasRef[id]; ok {
  713. return model
  714. }
  715. panic(fmt.Sprintf("Entity '%s' not defined!", id))
  716. }
  717. // Metodos das propriedades
  718. func (p *Propertie) FillTags(project *Project, propName string) {
  719. if p.TagVisited {
  720. return
  721. }
  722. if propName == "Id" {
  723. }
  724. if p.Tags != nil {
  725. for k, v := range p.Tags {
  726. if _, found := project.ReplaceWhenEmpty[k]; found && v == "" {
  727. p.Tags[k] = LcFirst(p.ID)
  728. }
  729. if _, found := project.OmitEmpty[k]; found {
  730. if p.Tags[k] != "-" {
  731. p.Tags[k] += ",omitempty"
  732. }
  733. }
  734. }
  735. }
  736. p.TagVisited = true
  737. }
  738. func (p *Propertie) GetType() string {
  739. return strings.Replace(p.Type, "*", "", 1)
  740. }
  741. // Metodos das informacoes da entidade
  742. func (p *EntityInfo) TranslateType(typ string) string {
  743. if typ == p.DynamicTypeId {
  744. return p.DynamicType
  745. }
  746. return typ
  747. }
  748. // Metodos do esquema de relacoes
  749. // Add adiciona uma relação ao esquema
  750. func (s *SchemasRelations) Has(entity string) bool {
  751. // spew.Dump(s)
  752. _, found := s.R[entity]
  753. return found
  754. }
  755. // Add adiciona uma relação ao esquema
  756. func (s *SchemasRelations) Get(entity string) []*Relation {
  757. if e, found := s.R[entity]; found {
  758. return e
  759. }
  760. return []*Relation{}
  761. }
  762. // Add adiciona uma relação ao esquema
  763. func (s *SchemasRelations) Add(r *Relation) {
  764. if _, found := s.R[r.Source]; !found {
  765. s.R[r.Source] = []*Relation{}
  766. }
  767. s.R[r.Source] = append(s.R[r.Source], r)
  768. }
  769. func ParseTemplate(input string, name ...string) (*template.Template, error) {
  770. var tmpl, err = template.New(strings.Join(name, "")).Parse(input)
  771. return tmpl, err
  772. }
  773. func TemplateToString(template *template.Template, data interface{}) (string, error) {
  774. var result bytes.Buffer
  775. if err := template.Execute(&result, data); err != nil {
  776. return "", err
  777. }
  778. return result.String(), nil
  779. }
  780. func NewProject() *Project {
  781. return &Project{
  782. Mode: "",
  783. SchemasRef: map[string]*Entity{},
  784. Icons: map[string]string{},
  785. ReplaceWhenEmpty: map[string]bool{},
  786. OmitEmpty: map[string]bool{},
  787. FormatMap: map[string]string{},
  788. Queries: &QueryDef{},
  789. Schemas: []*Entity{},
  790. Resources: []*Resource{},
  791. Translators: map[string]TranslationFn{},
  792. }
  793. }