manifold/functions/src/stripe.ts
James Grugett af6387bbf3
Stripe (#22)
* basic working payments

* Select funds amount and prettier funds button / dialog

* Add funds page and nav menu option

* Format funds amount. Use ghost button for back.

* Add mantic dollars description

* Improve styles of add funds page

* about styling

* change faq => about

* change default font to Courier

* header sign out menu item; remove user  card

* keep logo font

* fix header issue

* stripe webhook: handle repeat events

* Make add funds button a gradient

* add funds referer url

* Fix add funds page after merge

* Slight VisD tweaks

* Update add funds button position. Mantic => Manifold

* Remove Add funds menu option for now.

* Set up product ids and endpoint for stripe prod

* Swap back order in profile menu

Co-authored-by: mantikoros <sgrugett@gmail.com>
Co-authored-by: Austin Chen <akrolsmir@gmail.com>
2022-01-07 16:56:14 -06:00

127 lines
3.3 KiB
TypeScript

import * as functions from 'firebase-functions'
import * as admin from 'firebase-admin'
import Stripe from 'stripe'
import { payUser } from './resolve-market'
const stripe = new Stripe(functions.config().stripe.apikey, {
apiVersion: '2020-08-27',
typescript: true,
})
// manage at https://dashboard.stripe.com/test/products?active=true
const manticDollarStripePrice =
admin.instanceId().app.options.projectId === 'mantic-markets'
? {
500: 'price_1KFQXcGdoFKoCJW770gTNBrm',
1000: 'price_1KFQp1GdoFKoCJW7Iu0dsF65',
2500: 'price_1KFQqNGdoFKoCJW7SDvrSaEB',
10000: 'price_1KFQraGdoFKoCJW77I4XCwM3',
}
: {
500: 'price_1K8W10GdoFKoCJW7KWORLec1',
1000: 'price_1K8bC1GdoFKoCJW76k3g5MJk',
2500: 'price_1K8bDSGdoFKoCJW7avAwpV0e',
10000: 'price_1K8bEiGdoFKoCJW7Us4UkRHE',
}
export const createCheckoutSession = functions
.runWith({ minInstances: 1 })
.https.onRequest(async (req, res) => {
const userId = req.query.userId?.toString()
const manticDollarQuantity = req.query.manticDollarQuantity?.toString()
if (!userId) {
res.status(400).send('Invalid user ID')
return
}
if (
!manticDollarQuantity ||
!Object.keys(manticDollarStripePrice).includes(manticDollarQuantity)
) {
res.status(400).send('Invalid Mantic Dollar quantity')
return
}
const referrer =
req.query.referer || req.headers.referer || 'https://mantic.markets'
const session = await stripe.checkout.sessions.create({
metadata: {
userId,
manticDollarQuantity,
},
line_items: [
{
price:
manticDollarStripePrice[
manticDollarQuantity as unknown as keyof typeof manticDollarStripePrice
],
quantity: 1,
},
],
mode: 'payment',
success_url: `${referrer}?funding-success`,
cancel_url: `${referrer}?funding-failiure`,
})
res.redirect(303, session.url || '')
})
export const stripeWebhook = functions
.runWith({ minInstances: 1 })
.https.onRequest(async (req, res) => {
let event
try {
event = stripe.webhooks.constructEvent(
req.rawBody,
req.headers['stripe-signature'] as string,
functions.config().stripe.webhooksecret
)
} catch (e: any) {
console.log(`Webhook Error: ${e.message}`)
res.status(400).send(`Webhook Error: ${e.message}`)
return
}
if (event.type === 'checkout.session.completed') {
const session = event.data.object as any
await issueMoneys(session)
}
res.status(200).send('success')
})
const issueMoneys = async (session: any) => {
const { id: sessionId } = session
const query = await firestore
.collection('stripe-transactions')
.where('sessionId', '==', sessionId)
.get()
if (!query.empty) {
console.log('session', sessionId, 'already processed')
return
}
const { userId, manticDollarQuantity } = session.metadata
const payout = Number.parseInt(manticDollarQuantity)
await firestore.collection('stripe-transactions').add({
userId,
manticDollarQuantity: payout, // save as number
sessionId,
session,
})
await payUser([userId, payout])
console.log('user', userId, 'paid M$', payout)
}
const firestore = admin.firestore()