123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122 |
- local path = (...):gsub('%.[^%.]+$', '')
- local types = require(path .. '.types')
- local introspection = require(path .. '.introspection')
- local schema = {}
- schema.__index = schema
- schema.__emptyList = {}
- schema.__emptyObject = {}
- function schema.create(config)
- assert(type(config.query) == 'table', 'must provide query object')
- assert(not config.mutation or type(config.mutation) == 'table', 'mutation must be a table if provided')
- local self = setmetatable({}, schema)
- for k, v in pairs(config) do
- self[k] = v
- end
- self.directives = self.directives or {
- types.include,
- types.skip
- }
- self.typeMap = {}
- self.interfaceMap = {}
- self.directiveMap = {}
- self:generateTypeMap(self.query)
- self:generateTypeMap(self.mutation)
- self:generateTypeMap(self.subscription)
- self:generateTypeMap(introspection.__Schema)
- self:generateDirectiveMap()
- return self
- end
- function schema:generateTypeMap(node)
- if not node or (self.typeMap[node.name] and self.typeMap[node.name] == node) then return end
- if node.__type == 'NonNull' or node.__type == 'List' then
- return self:generateTypeMap(node.ofType)
- end
- if self.typeMap[node.name] and self.typeMap[node.name] ~= node then
- error('Encountered multiple types named "' .. node.name .. '"')
- end
- node.fields = type(node.fields) == 'function' and node.fields() or node.fields
- self.typeMap[node.name] = node
- if node.__type == 'Object' and node.interfaces then
- for _, interface in ipairs(node.interfaces) do
- self:generateTypeMap(interface)
- self.interfaceMap[interface.name] = self.interfaceMap[interface.name] or {}
- self.interfaceMap[interface.name][node] = node
- end
- end
- if node.__type == 'Object' or node.__type == 'Interface' or node.__type == 'InputObject' then
- for fieldName, field in pairs(node.fields) do
- if field.arguments then
- for name, argument in pairs(field.arguments) do
- local argumentType = argument.__type and argument or argument.kind
- assert(argumentType, 'Must supply type for argument "' .. name .. '" on "' .. fieldName .. '"')
- self:generateTypeMap(argumentType)
- end
- end
- self:generateTypeMap(field.kind)
- end
- end
- end
- function schema:generateDirectiveMap()
- for _, directive in ipairs(self.directives) do
- self.directiveMap[directive.name] = directive
- end
- end
- function schema:getType(name)
- if not name then return end
- return self.typeMap[name]
- end
- function schema:getImplementors(interface)
- local kind = self:getType(interface)
- local isInterface = kind and kind.__type == 'Interface'
- return self.interfaceMap[interface] or (isInterface and {} or nil)
- end
- function schema:getDirective(name)
- if not name then return false end
- return self.directiveMap[name]
- end
- function schema:getQueryType()
- return self.query
- end
- function schema:getMutationType()
- return self.mutation
- end
- function schema:getSubscriptionType()
- return self.subscription
- end
- function schema:getTypeMap()
- return self.typeMap
- end
- function schema:getPossibleTypes(abstractType)
- if abstractType.__type == 'Union' then
- return abstractType.types
- end
- return self:getImplementors(abstractType)
- end
- return schema
|