mirror of
https://github.com/Ocelot-Social-Community/Ocelot-Social.git
synced 2025-12-13 07:46:06 +00:00
I deprecated the debug flags myself here: https://github.com/neo4j-graphql/neo4j-graphql-js/pull/288 You can now debug the queries run by `neo4j-graphql-js` by starting the backend like this: ```bash DEBUG=neo4j-graphql-js yarn run backend ```
219 lines
7.5 KiB
JavaScript
219 lines
7.5 KiB
JavaScript
import { neo4jgraphql } from 'neo4j-graphql-js'
|
|
import fileUpload from './fileUpload'
|
|
import { neode } from '../../bootstrap/neo4j'
|
|
import { UserInputError, ForbiddenError } from 'apollo-server'
|
|
import Resolver from './helpers/Resolver'
|
|
|
|
const instance = neode()
|
|
|
|
export const getBlockedUsers = async context => {
|
|
const { neode } = context
|
|
const userModel = neode.model('User')
|
|
let blockedUsers = neode
|
|
.query()
|
|
.match('user', userModel)
|
|
.where('user.id', context.user.id)
|
|
.relationship(userModel.relationships().get('blocked'))
|
|
.to('blocked', userModel)
|
|
.return('blocked')
|
|
blockedUsers = await blockedUsers.execute()
|
|
blockedUsers = blockedUsers.records.map(r => r.get('blocked').properties)
|
|
return blockedUsers
|
|
}
|
|
|
|
export const getBlockedByUsers = async context => {
|
|
if (context.user.role === 'moderator' || context.user.role === 'admin') return []
|
|
const { neode } = context
|
|
const userModel = neode.model('User')
|
|
let blockedByUsers = neode
|
|
.query()
|
|
.match('user', userModel)
|
|
.relationship(userModel.relationships().get('blocked'))
|
|
.to('blocked', userModel)
|
|
.where('blocked.id', context.user.id)
|
|
.return('user')
|
|
blockedByUsers = await blockedByUsers.execute()
|
|
blockedByUsers = blockedByUsers.records.map(r => r.get('user').properties)
|
|
return blockedByUsers
|
|
}
|
|
|
|
export default {
|
|
Query: {
|
|
blockedUsers: async (object, args, context, resolveInfo) => {
|
|
try {
|
|
return getBlockedUsers(context)
|
|
} catch (e) {
|
|
throw new UserInputError(e.message)
|
|
}
|
|
},
|
|
User: async (object, args, context, resolveInfo) => {
|
|
const { email } = args
|
|
if (email) {
|
|
const e = await instance.first('EmailAddress', { email })
|
|
let user = e.get('belongsTo')
|
|
user = await user.toJson()
|
|
return [user.node]
|
|
}
|
|
return neo4jgraphql(object, args, context, resolveInfo)
|
|
},
|
|
},
|
|
Mutation: {
|
|
block: async (object, args, context, resolveInfo) => {
|
|
const { user: currentUser } = context
|
|
if (currentUser.id === args.id) return null
|
|
await instance.cypher(
|
|
`
|
|
MATCH(u:User {id: $currentUser.id})-[r:FOLLOWS]->(b:User {id: $args.id})
|
|
DELETE r
|
|
`,
|
|
{ currentUser, args },
|
|
)
|
|
const [user, blockedUser] = await Promise.all([
|
|
instance.find('User', currentUser.id),
|
|
instance.find('User', args.id),
|
|
])
|
|
await user.relateTo(blockedUser, 'blocked')
|
|
return blockedUser.toJson()
|
|
},
|
|
unblock: async (object, args, context, resolveInfo) => {
|
|
const { user: currentUser } = context
|
|
if (currentUser.id === args.id) return null
|
|
await instance.cypher(
|
|
`
|
|
MATCH(u:User {id: $currentUser.id})-[r:BLOCKED]->(b:User {id: $args.id})
|
|
DELETE r
|
|
`,
|
|
{ currentUser, args },
|
|
)
|
|
const blockedUser = await instance.find('User', args.id)
|
|
return blockedUser.toJson()
|
|
},
|
|
UpdateUser: async (object, args, context, resolveInfo) => {
|
|
const { termsAndConditionsAgreedVersion } = args
|
|
if (termsAndConditionsAgreedVersion) {
|
|
const regEx = new RegExp(/^[0-9]+\.[0-9]+\.[0-9]+$/g)
|
|
if (!regEx.test(termsAndConditionsAgreedVersion)) {
|
|
throw new ForbiddenError('Invalid version format!')
|
|
}
|
|
args.termsAndConditionsAgreedAt = new Date().toISOString()
|
|
}
|
|
args = await fileUpload(args, { file: 'avatarUpload', url: 'avatar' })
|
|
try {
|
|
const user = await instance.find('User', args.id)
|
|
if (!user) return null
|
|
await user.update({ ...args, updatedAt: new Date().toISOString() })
|
|
return user.toJson()
|
|
} catch (e) {
|
|
throw new UserInputError(e.message)
|
|
}
|
|
},
|
|
DeleteUser: async (object, params, context, resolveInfo) => {
|
|
const { resource } = params
|
|
const session = context.driver.session()
|
|
|
|
let user
|
|
try {
|
|
if (resource && resource.length) {
|
|
await Promise.all(
|
|
resource.map(async node => {
|
|
await session.run(
|
|
`
|
|
MATCH (resource:${node})<-[:WROTE]-(author:User {id: $userId})
|
|
OPTIONAL MATCH (resource)<-[:COMMENTS]-(comment:Comment)
|
|
SET resource.deleted = true
|
|
SET resource.content = 'UNAVAILABLE'
|
|
SET resource.contentExcerpt = 'UNAVAILABLE'
|
|
SET comment.deleted = true
|
|
RETURN author`,
|
|
{
|
|
userId: context.user.id,
|
|
},
|
|
)
|
|
}),
|
|
)
|
|
}
|
|
|
|
// we cannot set slug to 'UNAVAILABE' because of unique constraints
|
|
const transactionResult = await session.run(
|
|
`
|
|
MATCH (user:User {id: $userId})
|
|
SET user.deleted = true
|
|
SET user.name = 'UNAVAILABLE'
|
|
SET user.about = 'UNAVAILABLE'
|
|
WITH user
|
|
OPTIONAL MATCH (user)<-[:BELONGS_TO]-(email:EmailAddress)
|
|
DETACH DELETE email
|
|
WITH user
|
|
OPTIONAL MATCH (user)<-[:OWNED_BY]-(socialMedia:SocialMedia)
|
|
DETACH DELETE socialMedia
|
|
RETURN user`,
|
|
{ userId: context.user.id },
|
|
)
|
|
user = transactionResult.records.map(r => r.get('user').properties)[0]
|
|
} finally {
|
|
session.close()
|
|
}
|
|
return user
|
|
},
|
|
},
|
|
User: {
|
|
email: async (parent, params, context, resolveInfo) => {
|
|
if (typeof parent.email !== 'undefined') return parent.email
|
|
const { id } = parent
|
|
const statement = `MATCH(u:User {id: {id}})-[:PRIMARY_EMAIL]->(e:EmailAddress) RETURN e`
|
|
const result = await instance.cypher(statement, { id })
|
|
const [{ email }] = result.records.map(r => r.get('e').properties)
|
|
return email
|
|
},
|
|
...Resolver('User', {
|
|
undefinedToNull: [
|
|
'actorId',
|
|
'avatar',
|
|
'coverImg',
|
|
'deleted',
|
|
'disabled',
|
|
'locationName',
|
|
'about',
|
|
'termsAndConditionsAgreedVersion',
|
|
'termsAndConditionsAgreedAt',
|
|
'allowEmbedIframes',
|
|
'locale',
|
|
],
|
|
boolean: {
|
|
followedByCurrentUser:
|
|
'MATCH (this)<-[:FOLLOWS]-(u:User {id: $cypherParams.currentUserId}) RETURN COUNT(u) >= 1',
|
|
isBlocked:
|
|
'MATCH (this)<-[:BLOCKED]-(u:User {id: $cypherParams.currentUserId}) RETURN COUNT(u) >= 1',
|
|
},
|
|
count: {
|
|
contributionsCount:
|
|
'-[:WROTE]->(related:Post) WHERE NOT related.disabled = true AND NOT related.deleted = true',
|
|
friendsCount: '<-[:FRIENDS]->(related:User)',
|
|
followingCount: '-[:FOLLOWS]->(related:User)',
|
|
followedByCount: '<-[:FOLLOWS]-(related:User)',
|
|
commentedCount:
|
|
'-[:WROTE]->(c:Comment)-[:COMMENTS]->(related:Post) WHERE NOT related.disabled = true AND NOT related.deleted = true',
|
|
shoutedCount:
|
|
'-[:SHOUTED]->(related:Post) WHERE NOT related.disabled = true AND NOT related.deleted = true',
|
|
badgesCount: '<-[:REWARDED]-(related:Badge)',
|
|
},
|
|
hasOne: {
|
|
invitedBy: '<-[:INVITED]-(related:User)',
|
|
disabledBy: '<-[:DISABLED]-(related:User)',
|
|
location: '-[:IS_IN]->(related:Location)',
|
|
},
|
|
hasMany: {
|
|
followedBy: '<-[:FOLLOWS]-(related:User)',
|
|
following: '-[:FOLLOWS]->(related:User)',
|
|
friends: '-[:FRIENDS]-(related:User)',
|
|
socialMedia: '<-[:OWNED_BY]-(related:SocialMedia)',
|
|
contributions: '-[:WROTE]->(related:Post)',
|
|
comments: '-[:WROTE]->(related:Comment)',
|
|
shouted: '-[:SHOUTED]->(related:Post)',
|
|
categories: '-[:CATEGORIZED]->(related:Category)',
|
|
badges: '<-[:REWARDED]-(related:Badge)',
|
|
},
|
|
}),
|
|
},
|
|
}
|