166 lines
6.0 KiB
JavaScript
166 lines
6.0 KiB
JavaScript
'use strict'
|
|
const { withFilter } = require( 'apollo-server-express' )
|
|
const appRoot = require( 'app-root-path' )
|
|
|
|
const {
|
|
createStream,
|
|
getStream,
|
|
updateStream,
|
|
deleteStream,
|
|
getUserStreams,
|
|
getUserStreamsCount,
|
|
getStreamUsers,
|
|
grantPermissionsStream,
|
|
revokePermissionsStream
|
|
} = require( '../../services/streams' )
|
|
|
|
const { validateServerRole, validateScopes, authorizeResolver, pubsub } = require( `${appRoot}/modules/shared` )
|
|
|
|
// subscription events
|
|
const USER_STREAM_CREATED = 'USER_STREAM_CREATED'
|
|
const USER_STREAM_DELETED = 'USER_STREAM_DELETED'
|
|
const STREAM_UPDATED = 'STREAM_UPDATED'
|
|
const STREAM_DELETED = 'STREAM_DELETED'
|
|
|
|
module.exports = {
|
|
Query: {
|
|
async stream( parent, args, context, info ) {
|
|
// await validateScopes( context.scopes, 'streams:read' )
|
|
// await authorizeResolver( context.userId, args.id, 'stream:reviewer' )
|
|
|
|
let stream = await getStream( { streamId: args.id } )
|
|
return stream
|
|
},
|
|
async streams( parent, args, context, info ) {
|
|
// await validateScopes( context.scopes, 'streams:read' )
|
|
|
|
if ( args.limit && args.limit > 100 )
|
|
throw new UserInputError( 'Cannot return more than 100 items, please use pagination.' )
|
|
|
|
let totalCount = await getUserStreamsCount( { userId: context.userId, publicOnly: false, searchQuery: args.query } )
|
|
|
|
let { cursor, streams } = await getUserStreams( { userId: context.userId, limit: args.limit, cursor: args.cursor, publicOnly: false, searchQuery: args.query } )
|
|
return { totalCount, cursor: cursor, items: streams }
|
|
}
|
|
},
|
|
Stream: {
|
|
|
|
async collaborators( parent, args, context, info ) {
|
|
let users = await getStreamUsers( { streamId: parent.id } )
|
|
return users
|
|
}
|
|
|
|
},
|
|
User: {
|
|
|
|
async streams( parent, args, context, info ) {
|
|
if ( args.limit && args.limit > 100 )
|
|
throw new UserInputError( 'Cannot return more than 100 items, please use pagination.' )
|
|
// Return only the user's public streams if parent.id !== context.userId
|
|
let publicOnly = parent.id !== context.userId
|
|
let totalCount = await getUserStreamsCount( { userId: parent.id, publicOnly } )
|
|
|
|
let { cursor, streams } = await getUserStreams( { userId: parent.id, limit: args.limit, cursor: args.cursor, publicOnly: publicOnly } )
|
|
return { totalCount, cursor: cursor, items: streams }
|
|
}
|
|
|
|
},
|
|
Mutation: {
|
|
async streamCreate( parent, args, context, info ) {
|
|
// await validateServerRole( context, 'server:user' )
|
|
// await validateScopes( context.scopes, 'streams:write' )
|
|
|
|
let id = await createStream( { ...args.stream, ownerId: context.userId } )
|
|
await pubsub.publish( USER_STREAM_CREATED, { userStreamCreated: { id: id, ...args.stream }, ownerId: context.userId } )
|
|
return id
|
|
},
|
|
|
|
async streamUpdate( parent, args, context, info ) {
|
|
// await validateServerRole( context, 'server:user' )
|
|
// await validateScopes( context.scopes, 'streams:write' )
|
|
await authorizeResolver( context.userId, args.stream.id, 'stream:owner' )
|
|
|
|
let update = { streamId: args.stream.id, name: args.stream.name, description: args.stream.description }
|
|
|
|
await updateStream( update )
|
|
|
|
await pubsub.publish( STREAM_UPDATED, { streamUpdated: update, streamId: args.stream.id } )
|
|
|
|
return true
|
|
},
|
|
|
|
async streamDelete( parent, args, context, info ) {
|
|
// await validateServerRole( context, 'server:user' )
|
|
// await validateScopes( context.scopes, 'streams:write' )
|
|
await authorizeResolver( context.userId, args.id, 'stream:owner' )
|
|
|
|
// TODO: Notify all stream userss
|
|
let users = await getStreamUsers( { streamId: args.id } )
|
|
|
|
for ( let user of users ) {
|
|
await pubsub.publish( USER_STREAM_DELETED, { userStreamDeleted: { streamId: args.id }, ownerId: user.id } )
|
|
}
|
|
|
|
// TODO: Notify any listeners on the streamId
|
|
await pubsub.publish( STREAM_DELETED, { streamDeleted: args.id, streamId: args.id } )
|
|
|
|
// Delete after event so we can do authz
|
|
await deleteStream( { streamId: args.id } )
|
|
return true
|
|
},
|
|
|
|
async streamGrantPermission( parent, args, context, info ) {
|
|
// await validateServerRole( context, 'server:user' )
|
|
// await validateScopes( context.scopes, 'streams:write' )
|
|
await authorizeResolver( context.userId, args.streamId, 'stream:owner' )
|
|
|
|
if ( context.userId === args.userId ) throw new Error( 'You cannot set roles for yourself.' )
|
|
|
|
let permissionParams = { streamId: args.streamId, userId: args.userId, role: args.role.toLowerCase( ) || 'read' }
|
|
let granted = await grantPermissionsStream( permissionParams )
|
|
|
|
if ( granted ) {
|
|
let stream = await getStream( { streamId: args.streamId } )
|
|
await pubsub.publish( USER_STREAM_CREATED, { userStreamCreated: { id: stream.id, name: stream.name, description: stream.description }, userId: args.userId } )
|
|
}
|
|
|
|
return granted
|
|
},
|
|
|
|
async streamRevokePermission( parent, args, context, info ) {
|
|
// await validateServerRole( context, 'server:user' )
|
|
// await validateScopes( context.scopes, 'streams:write' )
|
|
await authorizeResolver( context.userId, args.streamId, 'stream:owner' )
|
|
let revoked = await revokePermissionsStream( { ...args } )
|
|
|
|
if ( revoked ) {
|
|
await pubsub.publish( USER_STREAM_DELETED, { userStreamDeleted: args.streamId, userId: args.userId } )
|
|
}
|
|
|
|
return revoked
|
|
}
|
|
},
|
|
Subscription: {
|
|
userStreamCreated: {
|
|
subscribe: withFilter( () => pubsub.asyncIterator( [ USER_STREAM_CREATED ] ),
|
|
( payload, variables ) => {
|
|
return payload.ownerId === variables.ownerId
|
|
} )
|
|
},
|
|
streamUpdated: {
|
|
subscribe: withFilter(
|
|
( ) => pubsub.asyncIterator( [ STREAM_UPDATED ] ),
|
|
( payload, variables ) => {
|
|
return payload.streamId === variables.streamId
|
|
} )
|
|
},
|
|
userStreamDeleted: {
|
|
subscribe: withFilter( () => pubsub.asyncIterator( [ USER_STREAM_DELETED ] ),
|
|
( payload, variables ) => {
|
|
return payload.ownerId === variables.ownerId
|
|
} )
|
|
}
|
|
}
|
|
|
|
}
|