front-models.js 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  1. export default async function createModels2(gql, config={create: 'Upsert', update: 'Upsert', delete: "Delete", findOne: "FindOne", find: 'Find', count: 'Count'}){
  2. const universeQuery = `query universe{
  3. __schema{
  4. types {
  5. name,
  6. kind,
  7. inputFields{
  8. name,
  9. type {
  10. kind,
  11. ofType{
  12. name,
  13. fields{
  14. name,
  15. type{
  16. name,
  17. kind
  18. }
  19. }
  20. }
  21. name,
  22. fields{
  23. name,
  24. type{
  25. name,
  26. kind
  27. }
  28. }
  29. }
  30. }
  31. fields {
  32. name,
  33. type {
  34. kind,
  35. ofType{
  36. name,
  37. fields{
  38. name,
  39. type{
  40. name,
  41. kind
  42. }
  43. }
  44. }
  45. name,
  46. fields{
  47. name,
  48. type{
  49. name,
  50. kind
  51. }
  52. }
  53. }
  54. args{
  55. name,
  56. type{
  57. name,
  58. inputFields{
  59. name,
  60. type{
  61. name,
  62. kind,
  63. ofType{
  64. name
  65. }
  66. }
  67. }
  68. }
  69. }
  70. }
  71. }
  72. }}
  73. `
  74. const universe = await gql.request(universeQuery)
  75. console.log(universe)
  76. const types = []
  77. const inputs = []
  78. for (let type of universe.__schema.types){
  79. if (!["Query", "Mutation"].includes(type.name) && type.kind === "OBJECT" && !type.name.startsWith('__')) types.push(type)
  80. if (!["Query", "Mutation"].includes(type.name) && type.kind === "INPUT_OBJECT" && !type.name.startsWith('__')) inputs.push(type)
  81. }
  82. console.log(types, inputs)
  83. let classes = {}
  84. const inTypes = name => types.find(type => type.name === name)
  85. const inInputs = name => inputs.find(type => type.name === name)
  86. const projectionBuilder = (type, allFields=true) => {
  87. if (!allFields && type.fields[0].name !== '_id') allFields = true
  88. if (allFields)
  89. return '{' +
  90. type.fields.map(field => {
  91. return field.name + ((field.type.kind === 'OBJECT' && (inTypes(field.type.name)) && projectionBuilder(inTypes(field.type.name), false)) ||
  92. (field.type.kind === 'LIST' && (inTypes(field.type.ofType.name)) && projectionBuilder(inTypes(field.type.ofType.name), false)) || '')
  93. })
  94. .join(',')
  95. + '}'
  96. else return `{_id}`
  97. }
  98. const identityMap = {}
  99. const createClass = (name, type, input) => {
  100. if (!(name in classes)) {
  101. classes[name] = class {
  102. constructor(data={}, empty = false){
  103. if (data._id && data._id in identityMap)
  104. return identityMap[data._id]
  105. this.populate(data)
  106. this.empty = empty
  107. if (this._id) identityMap[this._id] = this
  108. }
  109. populate(data){
  110. type.fields.forEach(({name, type:{ ofType, kind, name:otherName}}) => {
  111. ({SCALAR(){
  112. if (data && typeof data === 'object' && name in data) this[name] = data[name]
  113. },
  114. LIST(){
  115. const otherType = inTypes(ofType.name)
  116. if (otherType && data[name])
  117. this[name] = data[name].map(otherEntity => new classes[otherType.name](otherEntity, otherEntity._id && Object.keys(otherEntity).length === 1))
  118. else if (data && typeof data === 'object' && name in data) this[name] = data[name]
  119. },
  120. OBJECT(){
  121. const otherType = inTypes(otherName)
  122. if (otherType && data[name])
  123. this[name] = new classes[otherType.name](data[name], data[name]._id && Object.keys(data[name]).length === 1)
  124. else if (data && typeof data === 'object' && name in data) this[name] = data[name]
  125. }})[kind].call(this)
  126. })
  127. }
  128. get empty(){
  129. return this.then
  130. }
  131. set empty(value){
  132. if (value){
  133. this.then = async (onFullfilled, onRejected) => {
  134. const gqlQuery = `
  135. query ${name}FindOne($query: String){
  136. ${name}FindOne(query: $query)
  137. ${projectionBuilder(type)}
  138. }
  139. `
  140. const data = await gql.request(gqlQuery, {query: JSON.stringify([{_id: this._id}])})
  141. this.populate(data[name + 'FindOne'])
  142. this.empty = false
  143. const thenResult = onFullfilled(this)
  144. if (thenResult && typeof thenResult === 'object' && typeof thenResult.then === 'function'){
  145. return await thenResult
  146. }
  147. return thenResult
  148. }
  149. }
  150. else delete this.then
  151. }
  152. static get type(){
  153. return type
  154. }
  155. static get input(){
  156. return input
  157. }
  158. static get fields(){
  159. return this.type.fields
  160. }
  161. async save(){
  162. if (this.empty) throw new ReferenceError('Cannot save empty object')
  163. const data = {}
  164. input.inputFields.forEach(({name, type:{ ofType, kind, name:otherName}}) => {
  165. ({SCALAR(){
  166. if (this[name]) data[name] = this[name]
  167. },
  168. LIST(){
  169. const otherType = inInputs(ofType.name)
  170. if (otherType && this[name] && this[name] instanceof Array)
  171. data[name] = this[name].map(otherEntity => (otherEntity._id ? {_id: otherEntity._id} : otherEntity))
  172. },
  173. INPUT_OBJECT(){
  174. const otherType = inInputs(otherName)
  175. if (otherType && this[name] && typeof this[name] === 'object')
  176. data[name] = (this[name]._id ? {_id: this[name]._id} : this[name])
  177. }})[kind].call(this)
  178. })
  179. const gqlQuery = `
  180. mutation ${name}Upsert($data: ${input.name}){
  181. ${name}Upsert(${name.toLowerCase()}: $data)
  182. ${projectionBuilder(type)}
  183. }
  184. `
  185. let result = await gql.request(gqlQuery, {data})
  186. this.populate(result)
  187. }
  188. static async find(query={}, cursorCalls={}){
  189. const gqlQuery = `
  190. query ${name}Find($query: String){
  191. ${name}Find(query: $query)
  192. ${projectionBuilder(type)}
  193. }
  194. `
  195. let result = await gql.request(gqlQuery, {query: JSON.stringify([query, cursorCalls])})
  196. return result[name + 'Find'].map(entity => new classes[name](entity))
  197. }
  198. static async findOne(query){
  199. const gqlQuery = `
  200. query ${name}FindOne($query: String){
  201. ${name}FindOne(query: $query)
  202. ${projectionBuilder(type)}
  203. }
  204. `
  205. let result = await gql.request(gqlQuery, {query: JSON.stringify([query])})
  206. return result[name + 'FindOne']
  207. }
  208. static async count(query){
  209. const gqlQuery = `
  210. query ${name}Count($query: String){
  211. ${name}Count(query: $query)
  212. }
  213. `
  214. let result = await gql.request(gqlQuery, {query: JSON.stringify([query])})
  215. return result[name + 'Count']
  216. }
  217. }
  218. Object.defineProperty(classes[name], 'name', {value: name})
  219. }
  220. return classes[name]
  221. }
  222. types.forEach((type) => createClass(type.name, type, inputs.find(input => input.name === `${type.name}Input`)))
  223. return classes;
  224. }