schema.lua 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. local path = (...):gsub('%.[^%.]+$', '')
  2. local types = require(path .. '.types')
  3. local introspection = require(path .. '.introspection')
  4. local schema = {}
  5. schema.__index = schema
  6. function schema.create(config)
  7. assert(type(config.query) == 'table', 'must provide query object')
  8. assert(not config.mutation or type(config.mutation) == 'table', 'mutation must be a table if provided')
  9. local self = setmetatable({}, schema)
  10. for k, v in pairs(config) do
  11. self[k] = v
  12. end
  13. self.directives = self.directives or {
  14. types.include,
  15. types.skip
  16. }
  17. self.typeMap = {}
  18. self.interfaceMap = {}
  19. self.directiveMap = {}
  20. self:generateTypeMap(self.query)
  21. self:generateTypeMap(self.mutation)
  22. self:generateTypeMap(introspection.__Schema)
  23. self:generateDirectiveMap()
  24. return self
  25. end
  26. function schema:generateTypeMap(node)
  27. if not node or (self.typeMap[node.name] and self.typeMap[node.name] == node) then return end
  28. if node.__type == 'NonNull' or node.__type == 'List' then
  29. return self:generateTypeMap(node.ofType)
  30. end
  31. if self.typeMap[node.name] and self.typeMap[node.name] ~= node then
  32. error('Encountered multiple types named "' .. node.name .. '"')
  33. end
  34. node.fields = type(node.fields) == 'function' and node.fields() or node.fields
  35. self.typeMap[node.name] = node
  36. if node.__type == 'Object' and node.interfaces then
  37. for _, interface in ipairs(node.interfaces) do
  38. self:generateTypeMap(interface)
  39. self.interfaceMap[interface.name] = self.interfaceMap[interface.name] or {}
  40. self.interfaceMap[interface.name][node] = node
  41. end
  42. end
  43. if node.__type == 'Object' or node.__type == 'Interface' or node.__type == 'InputObject' then
  44. for fieldName, field in pairs(node.fields) do
  45. if field.arguments then
  46. for name, argument in pairs(field.arguments) do
  47. local argumentType = argument.__type and argument or argument.kind
  48. assert(argumentType, 'Must supply type for argument "' .. name .. '" on "' .. fieldName .. '"')
  49. self:generateTypeMap(argumentType)
  50. end
  51. end
  52. self:generateTypeMap(field.kind)
  53. end
  54. end
  55. end
  56. function schema:generateDirectiveMap()
  57. for _, directive in ipairs(self.directives) do
  58. self.directiveMap[directive.name] = directive
  59. end
  60. end
  61. function schema:getType(name)
  62. if not name then return end
  63. return self.typeMap[name]
  64. end
  65. function schema:getImplementors(interface)
  66. local kind = self:getType(interface)
  67. local isInterface = kind and kind.__type == 'Interface'
  68. return self.interfaceMap[interface] or (isInterface and {} or nil)
  69. end
  70. function schema:getDirective(name)
  71. if not name then return false end
  72. return self.directiveMap[name]
  73. end
  74. function schema:getQueryType()
  75. return self.query
  76. end
  77. function schema:getMutationType()
  78. return self.mutation
  79. end
  80. function schema:getTypeMap()
  81. return self.typeMap
  82. end
  83. function schema:getPossibleTypes(abstractType)
  84. if abstractType.__type == 'Union' then
  85. return abstractType.types
  86. end
  87. return self:getImplementors(abstractType)
  88. end
  89. return schema