mirror of
https://github.com/IT4Change/Ocelot-Social.git
synced 2025-12-13 07:45:56 +00:00
Set up backend for DonationsInfo
Co-authored-by: Mike Aono <aonomike@gmail.com>
This commit is contained in:
parent
ffb3ff896e
commit
835a098731
@ -135,6 +135,7 @@ const permissions = shield(
|
|||||||
blockedUsers: isAuthenticated,
|
blockedUsers: isAuthenticated,
|
||||||
notifications: isAuthenticated,
|
notifications: isAuthenticated,
|
||||||
profilePagePosts: or(onlyEnabledContent, isModerator),
|
profilePagePosts: or(onlyEnabledContent, isModerator),
|
||||||
|
Donations: isAuthenticated,
|
||||||
},
|
},
|
||||||
Mutation: {
|
Mutation: {
|
||||||
'*': deny,
|
'*': deny,
|
||||||
@ -177,6 +178,7 @@ const permissions = shield(
|
|||||||
VerifyEmailAddress: isAuthenticated,
|
VerifyEmailAddress: isAuthenticated,
|
||||||
pinPost: isAdmin,
|
pinPost: isAdmin,
|
||||||
unpinPost: isAdmin,
|
unpinPost: isAdmin,
|
||||||
|
UpdateDonations: isAdmin,
|
||||||
},
|
},
|
||||||
User: {
|
User: {
|
||||||
email: or(isMyOwn, isAdmin),
|
email: or(isMyOwn, isAdmin),
|
||||||
|
|||||||
14
backend/src/models/Donations.js
Normal file
14
backend/src/models/Donations.js
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import uuid from 'uuid/v4'
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
id: { type: 'string', primary: true, default: uuid },
|
||||||
|
goal: { type: 'number' },
|
||||||
|
progress: { type: 'number' },
|
||||||
|
createdAt: { type: 'string', isoDate: true, default: () => new Date().toISOString() },
|
||||||
|
updatedAt: {
|
||||||
|
type: 'string',
|
||||||
|
isoDate: true,
|
||||||
|
required: true,
|
||||||
|
default: () => new Date().toISOString(),
|
||||||
|
},
|
||||||
|
}
|
||||||
@ -12,4 +12,5 @@ export default {
|
|||||||
Category: require('./Category.js'),
|
Category: require('./Category.js'),
|
||||||
Tag: require('./Tag.js'),
|
Tag: require('./Tag.js'),
|
||||||
Location: require('./Location.js'),
|
Location: require('./Location.js'),
|
||||||
|
Donations: require('./Donations.js'),
|
||||||
}
|
}
|
||||||
|
|||||||
@ -24,6 +24,7 @@ export default applyScalars(
|
|||||||
'SocialMedia',
|
'SocialMedia',
|
||||||
'NOTIFIED',
|
'NOTIFIED',
|
||||||
'REPORTED',
|
'REPORTED',
|
||||||
|
'Donations',
|
||||||
],
|
],
|
||||||
// add 'User' here as soon as possible
|
// add 'User' here as soon as possible
|
||||||
},
|
},
|
||||||
@ -44,6 +45,7 @@ export default applyScalars(
|
|||||||
'EMOTED',
|
'EMOTED',
|
||||||
'NOTIFIED',
|
'NOTIFIED',
|
||||||
'REPORTED',
|
'REPORTED',
|
||||||
|
'Donations',
|
||||||
],
|
],
|
||||||
// add 'User' here as soon as possible
|
// add 'User' here as soon as possible
|
||||||
},
|
},
|
||||||
|
|||||||
31
backend/src/schema/resolvers/donations.js
Normal file
31
backend/src/schema/resolvers/donations.js
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
export default {
|
||||||
|
Mutation: {
|
||||||
|
UpdateDonations: async (_parent, params, context, _resolveInfo) => {
|
||||||
|
const { driver } = context
|
||||||
|
const session = driver.session()
|
||||||
|
let donations
|
||||||
|
const writeTxResultPromise = session.writeTransaction(async txc => {
|
||||||
|
const updateDonationsTransactionResponse = await txc.run(
|
||||||
|
`
|
||||||
|
MATCH (donations:Donations {id: $params.id})
|
||||||
|
SET donations += $params
|
||||||
|
SET donations.updatedAt = toString(datetime())
|
||||||
|
RETURN donations
|
||||||
|
`,
|
||||||
|
{ params },
|
||||||
|
)
|
||||||
|
return updateDonationsTransactionResponse.records.map(
|
||||||
|
record => record.get('donations').properties,
|
||||||
|
)
|
||||||
|
})
|
||||||
|
try {
|
||||||
|
const txResult = await writeTxResultPromise
|
||||||
|
if (!txResult[0]) return null
|
||||||
|
donations = txResult[0]
|
||||||
|
} finally {
|
||||||
|
session.close()
|
||||||
|
}
|
||||||
|
return donations
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
172
backend/src/schema/resolvers/donations.spec.js
Normal file
172
backend/src/schema/resolvers/donations.spec.js
Normal file
@ -0,0 +1,172 @@
|
|||||||
|
import { createTestClient } from 'apollo-server-testing'
|
||||||
|
import Factory from '../../seed/factories'
|
||||||
|
import { gql } from '../../jest/helpers'
|
||||||
|
import { neode as getNeode, getDriver } from '../../bootstrap/neo4j'
|
||||||
|
import createServer from '../../server'
|
||||||
|
|
||||||
|
let mutate, query, authenticatedUser, variables
|
||||||
|
const factory = Factory()
|
||||||
|
const instance = getNeode()
|
||||||
|
const driver = getDriver()
|
||||||
|
|
||||||
|
const updateDonationsMutation = gql`
|
||||||
|
mutation($id: ID!, $goal: Int, $progress: Int) {
|
||||||
|
UpdateDonations(id: $id, goal: $goal, progress: $progress) {
|
||||||
|
goal
|
||||||
|
progress
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`
|
||||||
|
const donationsQuery = gql`
|
||||||
|
query {
|
||||||
|
Donations {
|
||||||
|
goal
|
||||||
|
progress
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`
|
||||||
|
|
||||||
|
describe('donations', () => {
|
||||||
|
let currentUser, newlyCreatedDonations
|
||||||
|
beforeAll(async () => {
|
||||||
|
await factory.cleanDatabase()
|
||||||
|
authenticatedUser = undefined
|
||||||
|
const { server } = createServer({
|
||||||
|
context: () => {
|
||||||
|
return {
|
||||||
|
driver,
|
||||||
|
neode: instance,
|
||||||
|
user: authenticatedUser,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
mutate = createTestClient(server).mutate
|
||||||
|
query = createTestClient(server).query
|
||||||
|
})
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
variables = {}
|
||||||
|
newlyCreatedDonations = await factory.create('Donations', { id: 'total-donations' })
|
||||||
|
})
|
||||||
|
|
||||||
|
afterEach(async () => {
|
||||||
|
await factory.cleanDatabase()
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('query for donations', () => {
|
||||||
|
describe('unauthenticated', () => {
|
||||||
|
it('throws authorization error', async () => {
|
||||||
|
authenticatedUser = undefined
|
||||||
|
await expect(query({ query: donationsQuery, variables })).resolves.toMatchObject({
|
||||||
|
errors: [{ message: 'Not Authorised!' }],
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('authenticated', () => {
|
||||||
|
beforeEach(async () => {
|
||||||
|
currentUser = await factory.create('User', {
|
||||||
|
id: 'normal-user',
|
||||||
|
role: 'user',
|
||||||
|
})
|
||||||
|
authenticatedUser = await currentUser.toJson()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('returns the current Donations info', async () => {
|
||||||
|
await expect(query({ query: donationsQuery, variables })).resolves.toMatchObject({
|
||||||
|
data: { Donations: [{ goal: 15000, progress: 0 }] },
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('update donations', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
variables = { id: 'total-donations', goal: 20000, progress: 3000 }
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('unauthenticated', () => {
|
||||||
|
it('throws authorization error', async () => {
|
||||||
|
authenticatedUser = undefined
|
||||||
|
await expect(
|
||||||
|
mutate({ mutation: updateDonationsMutation, variables }),
|
||||||
|
).resolves.toMatchObject({
|
||||||
|
errors: [{ message: 'Not Authorised!' }],
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('authenticated', () => {
|
||||||
|
describe('as a normal user', () => {
|
||||||
|
beforeEach(async () => {
|
||||||
|
currentUser = await factory.create('User', {
|
||||||
|
id: 'normal-user',
|
||||||
|
role: 'user',
|
||||||
|
})
|
||||||
|
authenticatedUser = await currentUser.toJson()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('throws authorization error', async () => {
|
||||||
|
await expect(
|
||||||
|
mutate({ mutation: updateDonationsMutation, variables }),
|
||||||
|
).resolves.toMatchObject({
|
||||||
|
data: { UpdateDonations: null },
|
||||||
|
errors: [{ message: 'Not Authorised!' }],
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('as a moderator', () => {
|
||||||
|
beforeEach(async () => {
|
||||||
|
currentUser = await factory.create('User', {
|
||||||
|
id: 'moderator',
|
||||||
|
role: 'moderator',
|
||||||
|
})
|
||||||
|
authenticatedUser = await currentUser.toJson()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('throws authorization error', async () => {
|
||||||
|
await expect(
|
||||||
|
mutate({ mutation: updateDonationsMutation, variables }),
|
||||||
|
).resolves.toMatchObject({
|
||||||
|
data: { UpdateDonations: null },
|
||||||
|
errors: [{ message: 'Not Authorised!' }],
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('as an admin', () => {
|
||||||
|
beforeEach(async () => {
|
||||||
|
currentUser = await factory.create('User', {
|
||||||
|
id: 'admin',
|
||||||
|
role: 'admin',
|
||||||
|
})
|
||||||
|
authenticatedUser = await currentUser.toJson()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('updates Donations info', async () => {
|
||||||
|
await expect(
|
||||||
|
mutate({ mutation: updateDonationsMutation, variables }),
|
||||||
|
).resolves.toMatchObject({
|
||||||
|
data: { UpdateDonations: { goal: 20000, progress: 3000 } },
|
||||||
|
errors: undefined,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('updates the updatedAt attribute', async () => {
|
||||||
|
newlyCreatedDonations = await newlyCreatedDonations.toJson()
|
||||||
|
const {
|
||||||
|
data: { UpdateDonations },
|
||||||
|
} = await mutate({ mutation: updateDonationsMutation, variables })
|
||||||
|
expect(newlyCreatedDonations.updatedAt).toBeTruthy()
|
||||||
|
expect(Date.parse(newlyCreatedDonations.updatedAt)).toEqual(expect.any(Number))
|
||||||
|
expect(UpdateDonations.updatedAt).toBeTruthy()
|
||||||
|
expect(Date.parse(UpdateDonations.updatedAt)).toEqual(expect.any(Number))
|
||||||
|
expect(newlyCreatedDonations.updatedAt).not.toEqual(UpdateDonations.updatedAt)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
14
backend/src/schema/types/type/Donations.gql
Normal file
14
backend/src/schema/types/type/Donations.gql
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
type Donations {
|
||||||
|
goal: Int!
|
||||||
|
progress: Int!
|
||||||
|
createdAt: String
|
||||||
|
updatedAt: String
|
||||||
|
}
|
||||||
|
|
||||||
|
type Query {
|
||||||
|
Donations: [Donations]
|
||||||
|
}
|
||||||
|
|
||||||
|
type Mutation {
|
||||||
|
UpdateDonations(id: ID!, goal: Int, progress: Int): Donations
|
||||||
|
}
|
||||||
18
backend/src/seed/factories/donations.js
Normal file
18
backend/src/seed/factories/donations.js
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
import uuid from 'uuid/v4'
|
||||||
|
|
||||||
|
export default function create() {
|
||||||
|
return {
|
||||||
|
factory: async ({ args, neodeInstance }) => {
|
||||||
|
const defaults = {
|
||||||
|
id: uuid(),
|
||||||
|
goal: 15000,
|
||||||
|
progress: 0,
|
||||||
|
}
|
||||||
|
args = {
|
||||||
|
...defaults,
|
||||||
|
...args,
|
||||||
|
}
|
||||||
|
return neodeInstance.create('Donations', args)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -8,6 +8,7 @@ import createTag from './tags.js'
|
|||||||
import createSocialMedia from './socialMedia.js'
|
import createSocialMedia from './socialMedia.js'
|
||||||
import createLocation from './locations.js'
|
import createLocation from './locations.js'
|
||||||
import createEmailAddress from './emailAddresses.js'
|
import createEmailAddress from './emailAddresses.js'
|
||||||
|
import createDonations from './donations.js'
|
||||||
import createUnverifiedEmailAddresss from './unverifiedEmailAddresses.js'
|
import createUnverifiedEmailAddresss from './unverifiedEmailAddresses.js'
|
||||||
|
|
||||||
const factories = {
|
const factories = {
|
||||||
@ -21,6 +22,7 @@ const factories = {
|
|||||||
Location: createLocation,
|
Location: createLocation,
|
||||||
EmailAddress: createEmailAddress,
|
EmailAddress: createEmailAddress,
|
||||||
UnverifiedEmailAddress: createUnverifiedEmailAddresss,
|
UnverifiedEmailAddress: createUnverifiedEmailAddresss,
|
||||||
|
Donations: createDonations,
|
||||||
}
|
}
|
||||||
|
|
||||||
export const cleanDatabase = async (options = {}) => {
|
export const cleanDatabase = async (options = {}) => {
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user