mirror of
https://github.com/IT4Change/Ocelot-Social.git
synced 2025-12-13 07:45:56 +00:00
124 lines
3.8 KiB
JavaScript
124 lines
3.8 KiB
JavaScript
import log from './databaseLogger'
|
|
|
|
export const undefinedToNullResolver = list => {
|
|
const resolvers = {}
|
|
list.forEach(key => {
|
|
resolvers[key] = async parent => {
|
|
return typeof parent[key] === 'undefined' ? null : parent[key]
|
|
}
|
|
})
|
|
return resolvers
|
|
}
|
|
|
|
export default function Resolver(type, options = {}) {
|
|
const {
|
|
idAttribute = 'id',
|
|
undefinedToNull = [],
|
|
boolean = {},
|
|
count = {},
|
|
hasOne = {},
|
|
hasMany = {},
|
|
} = options
|
|
|
|
const _hasResolver = (resolvers, { key, connection }, { returnType }) => {
|
|
return async (parent, params, { driver, cypherParams }, resolveInfo) => {
|
|
if (typeof parent[key] !== 'undefined') return parent[key]
|
|
const id = parent[idAttribute]
|
|
const session = driver.session()
|
|
const readTxResultPromise = session.readTransaction(async txc => {
|
|
const cypher = `
|
|
MATCH(:${type} {${idAttribute}: $id})${connection}
|
|
RETURN related {.*} as related
|
|
`
|
|
const result = await txc.run(cypher, { id, cypherParams })
|
|
log(result)
|
|
return result.records.map(r => r.get('related'))
|
|
})
|
|
try {
|
|
let response = await readTxResultPromise
|
|
if (returnType === 'object') response = response[0] || null
|
|
return response
|
|
} finally {
|
|
session.close()
|
|
}
|
|
}
|
|
}
|
|
|
|
const booleanResolver = obj => {
|
|
const resolvers = {}
|
|
for (const [key, condition] of Object.entries(obj)) {
|
|
resolvers[key] = async (parent, params, { cypherParams, driver }, resolveInfo) => {
|
|
if (typeof parent[key] !== 'undefined') return parent[key]
|
|
const id = parent[idAttribute]
|
|
const session = driver.session()
|
|
const readTxResultPromise = session.readTransaction(async txc => {
|
|
const nodeCondition = condition.replace('this', 'this {id: $id}')
|
|
const cypher = `${nodeCondition} as ${key}`
|
|
const result = await txc.run(cypher, { id, cypherParams })
|
|
log(result)
|
|
const [response] = result.records.map(r => r.get(key))
|
|
return response
|
|
})
|
|
try {
|
|
return await readTxResultPromise
|
|
} finally {
|
|
session.close()
|
|
}
|
|
}
|
|
}
|
|
return resolvers
|
|
}
|
|
|
|
const countResolver = obj => {
|
|
const resolvers = {}
|
|
for (const [key, connection] of Object.entries(obj)) {
|
|
resolvers[key] = async (parent, params, { driver, cypherParams }, resolveInfo) => {
|
|
if (typeof parent[key] !== 'undefined') return parent[key]
|
|
const session = driver.session()
|
|
const readTxResultPromise = session.readTransaction(async txc => {
|
|
const id = parent[idAttribute]
|
|
const cypher = `
|
|
MATCH(u:${type} {${idAttribute}: $id})${connection}
|
|
RETURN COUNT(DISTINCT(related)) as count
|
|
`
|
|
const result = await txc.run(cypher, { id, cypherParams })
|
|
log(result)
|
|
const [response] = result.records.map(r => r.get('count').toNumber())
|
|
return response
|
|
})
|
|
try {
|
|
return await readTxResultPromise
|
|
} finally {
|
|
session.close()
|
|
}
|
|
}
|
|
}
|
|
return resolvers
|
|
}
|
|
|
|
const hasManyResolver = obj => {
|
|
const resolvers = {}
|
|
for (const [key, connection] of Object.entries(obj)) {
|
|
resolvers[key] = _hasResolver(resolvers, { key, connection }, { returnType: 'iterable' })
|
|
}
|
|
return resolvers
|
|
}
|
|
|
|
const hasOneResolver = obj => {
|
|
const resolvers = {}
|
|
for (const [key, connection] of Object.entries(obj)) {
|
|
resolvers[key] = _hasResolver(resolvers, { key, connection }, { returnType: 'object' })
|
|
}
|
|
return resolvers
|
|
}
|
|
|
|
const result = {
|
|
...undefinedToNullResolver(undefinedToNull),
|
|
...booleanResolver(boolean),
|
|
...countResolver(count),
|
|
...hasOneResolver(hasOne),
|
|
...hasManyResolver(hasMany),
|
|
}
|
|
return result
|
|
}
|