2022-04-29 23:35:56 +00:00
|
|
|
import * as functions from 'firebase-functions'
|
|
|
|
import * as admin from 'firebase-admin'
|
|
|
|
|
2022-05-15 17:39:42 +00:00
|
|
|
import { User } from '../../common/user'
|
|
|
|
import { Txn } from '../../common/txn'
|
|
|
|
import { removeUndefinedProps } from '../../common/util/object'
|
2022-04-29 23:35:56 +00:00
|
|
|
|
|
|
|
export const transact = functions
|
|
|
|
.runWith({ minInstances: 1 })
|
|
|
|
.https.onCall(async (data: Omit<Txn, 'id' | 'createdTime'>, context) => {
|
|
|
|
const userId = context?.auth?.uid
|
|
|
|
if (!userId) return { status: 'error', message: 'Not authorized' }
|
|
|
|
|
2022-06-18 03:28:16 +00:00
|
|
|
const {
|
|
|
|
amount,
|
|
|
|
fromType,
|
|
|
|
fromId,
|
|
|
|
toId,
|
|
|
|
toType,
|
|
|
|
category,
|
|
|
|
token,
|
|
|
|
data: innerData,
|
|
|
|
description,
|
|
|
|
} = data
|
2022-04-29 23:35:56 +00:00
|
|
|
|
|
|
|
if (fromType !== 'USER')
|
|
|
|
return {
|
|
|
|
status: 'error',
|
|
|
|
message: "From type is only implemented for type 'user'.",
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fromId !== userId)
|
|
|
|
return {
|
|
|
|
status: 'error',
|
|
|
|
message: 'Must be authenticated with userId equal to specified fromId.',
|
|
|
|
}
|
|
|
|
|
2022-06-18 03:28:16 +00:00
|
|
|
if (isNaN(amount) || !isFinite(amount))
|
2022-04-29 23:35:56 +00:00
|
|
|
return { status: 'error', message: 'Invalid amount' }
|
|
|
|
|
|
|
|
// Run as transaction to prevent race conditions.
|
|
|
|
return await firestore.runTransaction(async (transaction) => {
|
|
|
|
const fromDoc = firestore.doc(`users/${userId}`)
|
|
|
|
const fromSnap = await transaction.get(fromDoc)
|
|
|
|
if (!fromSnap.exists) {
|
|
|
|
return { status: 'error', message: 'User not found' }
|
|
|
|
}
|
|
|
|
const fromUser = fromSnap.data() as User
|
|
|
|
|
2022-06-21 16:52:02 +00:00
|
|
|
if (amount > 0 && fromUser.balance < amount) {
|
2022-04-29 23:35:56 +00:00
|
|
|
return {
|
|
|
|
status: 'error',
|
|
|
|
message: `Insufficient balance: ${fromUser.username} needed ${amount} but only had ${fromUser.balance} `,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (toType === 'USER') {
|
|
|
|
const toDoc = firestore.doc(`users/${toId}`)
|
|
|
|
const toSnap = await transaction.get(toDoc)
|
|
|
|
if (!toSnap.exists) {
|
|
|
|
return { status: 'error', message: 'User not found' }
|
|
|
|
}
|
|
|
|
const toUser = toSnap.data() as User
|
2022-06-21 16:52:02 +00:00
|
|
|
if (amount < 0 && toUser.balance < -amount) {
|
|
|
|
return {
|
|
|
|
status: 'error',
|
|
|
|
message: `Insufficient balance: ${
|
|
|
|
toUser.username
|
|
|
|
} needed ${-amount} but only had ${toUser.balance} `,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-30 12:52:24 +00:00
|
|
|
transaction.update(toDoc, {
|
|
|
|
balance: toUser.balance + amount,
|
|
|
|
totalDeposits: toUser.totalDeposits + amount,
|
|
|
|
})
|
2022-04-29 23:35:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const newTxnDoc = firestore.collection(`txns/`).doc()
|
|
|
|
|
2022-06-22 08:13:25 +00:00
|
|
|
const txn = removeUndefinedProps({
|
2022-04-29 23:35:56 +00:00
|
|
|
id: newTxnDoc.id,
|
|
|
|
createdTime: Date.now(),
|
|
|
|
|
|
|
|
fromId,
|
|
|
|
fromType,
|
|
|
|
toId,
|
|
|
|
toType,
|
|
|
|
|
|
|
|
amount,
|
2022-06-18 03:28:16 +00:00
|
|
|
category,
|
|
|
|
data: innerData,
|
|
|
|
token,
|
|
|
|
|
2022-04-29 23:35:56 +00:00
|
|
|
description,
|
|
|
|
})
|
|
|
|
|
|
|
|
transaction.create(newTxnDoc, txn)
|
2022-04-30 12:52:24 +00:00
|
|
|
transaction.update(fromDoc, {
|
|
|
|
balance: fromUser.balance - amount,
|
|
|
|
totalDeposits: fromUser.totalDeposits - amount,
|
|
|
|
})
|
2022-04-29 23:35:56 +00:00
|
|
|
|
|
|
|
return { status: 'success', txn }
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
const firestore = admin.firestore()
|