8063652dbf
* feat(server): WIP automations api * feat(server): Automations Backend api WIP take 2 * feat(automations): add validation to automation run schema with zod * fix(server): add zod to package.json * fix(server): automations pr cleanup
174 lines
4.9 KiB
JavaScript
174 lines
4.9 KiB
JavaScript
'use strict'
|
|
const fs = require('fs')
|
|
const path = require('path')
|
|
const { appRoot, packageRoot } = require('@/bootstrap')
|
|
const { values, merge, camelCase } = require('lodash')
|
|
const baseTypeDefs = require('@/modules/core/graph/schema/baseTypeDefs')
|
|
const { scalarResolvers } = require('./core/graph/scalars')
|
|
const { makeExecutableSchema } = require('@graphql-tools/schema')
|
|
const { moduleLogger } = require('@/logging/logging')
|
|
|
|
/**
|
|
* Cached speckle module requires
|
|
* @type {import('@/modules/shared/helpers/typeHelper').SpeckleModule[]}
|
|
* */
|
|
const loadedModules = []
|
|
|
|
/**
|
|
* Module init will be ran multiple times in tests, so it's useful for modules to know
|
|
* when an initialization is a repeat one, so as to not introduce unnecessary resources/listeners
|
|
*/
|
|
let hasInitializationOccurred = false
|
|
|
|
function autoloadFromDirectory(dirPath) {
|
|
if (!fs.existsSync(dirPath)) return
|
|
|
|
const results = {}
|
|
fs.readdirSync(dirPath).forEach((file) => {
|
|
const pathToFile = path.join(dirPath, file)
|
|
const stat = fs.statSync(pathToFile)
|
|
if (stat.isFile()) {
|
|
const ext = path.extname(file)
|
|
if (['.js', '.ts'].includes(ext)) {
|
|
const name = camelCase(path.basename(file, ext))
|
|
results[name] = require(pathToFile)
|
|
}
|
|
}
|
|
})
|
|
|
|
return results
|
|
}
|
|
|
|
async function getSpeckleModules() {
|
|
if (loadedModules.length) return loadedModules
|
|
|
|
const moduleDirs = [
|
|
'./core',
|
|
'./auth',
|
|
'./apiexplorer',
|
|
'./emails',
|
|
'./pwdreset',
|
|
'./serverinvites',
|
|
'./previews',
|
|
'./fileuploads',
|
|
'./comments',
|
|
'./blobstorage',
|
|
'./notifications',
|
|
'./activitystream',
|
|
'./accessrequests',
|
|
'./webhooks',
|
|
'./cross-server-sync',
|
|
'./automations'
|
|
]
|
|
|
|
for (const dir of moduleDirs) {
|
|
loadedModules.push(require(dir))
|
|
}
|
|
|
|
return loadedModules
|
|
}
|
|
|
|
exports.init = async (app) => {
|
|
const modules = await getSpeckleModules()
|
|
const isInitial = !hasInitializationOccurred
|
|
|
|
// Stage 1: initialise all modules
|
|
for (const module of modules) {
|
|
await module.init?.(app, isInitial)
|
|
}
|
|
|
|
// Stage 2: finalize init all modules
|
|
for (const module of modules) {
|
|
await module.finalize?.(app, isInitial)
|
|
}
|
|
|
|
hasInitializationOccurred = true
|
|
}
|
|
|
|
exports.shutdown = async () => {
|
|
moduleLogger.info('Triggering module shutdown...')
|
|
const modules = await getSpeckleModules()
|
|
|
|
for (const module of modules) {
|
|
await module.shutdown?.()
|
|
}
|
|
moduleLogger.info('...module shutdown finished')
|
|
}
|
|
|
|
/**
|
|
* @returns {Pick<import('apollo-server-express').Config, 'resolvers' | 'typeDefs'> & { directiveBuilders: Record<string, import('@/modules/core/graph/helpers/directiveHelper').GraphqlDirectiveBuilder>}}
|
|
*/
|
|
const graphComponents = () => {
|
|
// Base query and mutation to allow for type extension by modules.
|
|
const typeDefs = [baseTypeDefs]
|
|
|
|
let resolverObjs = []
|
|
let directiveBuilders = {}
|
|
|
|
// load typedefs from /assets
|
|
const assetModuleDirs = fs.readdirSync(`${packageRoot}/assets`)
|
|
assetModuleDirs.forEach((dir) => {
|
|
const typeDefDirPath = path.join(`${packageRoot}/assets`, dir, 'typedefs')
|
|
if (fs.existsSync(typeDefDirPath)) {
|
|
const moduleSchemas = fs.readdirSync(typeDefDirPath)
|
|
moduleSchemas.forEach((schema) => {
|
|
typeDefs.push(fs.readFileSync(path.join(typeDefDirPath, schema), 'utf8'))
|
|
})
|
|
}
|
|
})
|
|
|
|
// load code modules from /modules
|
|
const codeModuleDirs = fs.readdirSync(`${appRoot}/modules`)
|
|
codeModuleDirs.forEach((file) => {
|
|
const fullPath = path.join(`${appRoot}/modules`, file)
|
|
|
|
// first pass load of resolvers
|
|
const resolversPath = path.join(fullPath, 'graph', 'resolvers')
|
|
if (fs.existsSync(resolversPath)) {
|
|
resolverObjs = [...resolverObjs, ...values(autoloadFromDirectory(resolversPath))]
|
|
}
|
|
|
|
// load directives
|
|
const directivesPath = path.join(fullPath, 'graph', 'directives')
|
|
if (fs.existsSync(directivesPath)) {
|
|
directiveBuilders = Object.assign(
|
|
...values(autoloadFromDirectory(directivesPath))
|
|
)
|
|
}
|
|
})
|
|
|
|
const resolvers = { ...scalarResolvers }
|
|
resolverObjs.forEach((o) => {
|
|
merge(resolvers, o)
|
|
})
|
|
|
|
return { resolvers, typeDefs, directiveBuilders }
|
|
}
|
|
|
|
exports.graphSchema = () => {
|
|
const { resolvers, typeDefs, directiveBuilders } = graphComponents()
|
|
|
|
/** @type {string[]} */
|
|
const directiveTypedefs = []
|
|
/** @type {import('@/modules/core/graph/helpers/directiveHelper').SchemaTransformer[]} */
|
|
const directiveSchemaTransformers = []
|
|
for (const directiveBuilder of Object.values(directiveBuilders)) {
|
|
const { typeDefs, schemaTransformer } = directiveBuilder()
|
|
directiveTypedefs.push(typeDefs)
|
|
directiveSchemaTransformers.push(schemaTransformer)
|
|
}
|
|
|
|
// Init schema w/ base resolvers & typedefs
|
|
let schema = makeExecutableSchema({
|
|
resolvers,
|
|
typeDefs: [...directiveTypedefs, ...typeDefs]
|
|
})
|
|
|
|
// Apply directives
|
|
for (const schemaTransformer of directiveSchemaTransformers) {
|
|
schema = schemaTransformer(schema)
|
|
}
|
|
|
|
return schema
|
|
}
|