2023-07-23 10:08:43 -05:00
|
|
|
import { GraphQLError } from 'graphql'
|
2022-05-12 13:44:21 -05:00
|
|
|
import AWS from 'aws-sdk'
|
|
|
|
import { IMAGE_PIXELS_MAX, UPLOAD_SIZE_MAX, UPLOAD_TYPES_ALLOW } from '../../lib/constants'
|
|
|
|
|
|
|
|
const bucketRegion = 'us-east-1'
|
|
|
|
|
|
|
|
AWS.config.update({
|
|
|
|
region: bucketRegion
|
|
|
|
})
|
|
|
|
|
|
|
|
export default {
|
|
|
|
Mutation: {
|
|
|
|
getSignedPOST: async (parent, { type, size, width, height }, { models, me }) => {
|
|
|
|
if (!me) {
|
2023-07-23 10:08:43 -05:00
|
|
|
throw new GraphQLError('you must be logged in to get a signed url', { extensions: { code: 'FORBIDDEN' } })
|
2022-05-12 13:44:21 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (UPLOAD_TYPES_ALLOW.indexOf(type) === -1) {
|
2023-07-23 10:08:43 -05:00
|
|
|
throw new GraphQLError(`image must be ${UPLOAD_TYPES_ALLOW.map(t => t.replace('image/', '')).join(', ')}`, { extensions: { code: 'BAD_INPUT' } })
|
2022-05-12 13:44:21 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (size > UPLOAD_SIZE_MAX) {
|
2023-07-23 10:08:43 -05:00
|
|
|
throw new GraphQLError(`image must be less than ${UPLOAD_SIZE_MAX} bytes`, { extensions: { code: 'BAD_INPUT' } })
|
2022-05-12 13:44:21 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (width * height > IMAGE_PIXELS_MAX) {
|
2023-07-23 10:08:43 -05:00
|
|
|
throw new GraphQLError(`image must be less than ${IMAGE_PIXELS_MAX} pixels`, { extensions: { code: 'BAD_INPUT' } })
|
2022-05-12 13:44:21 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// create upload record
|
|
|
|
const upload = await models.upload.create({
|
|
|
|
data: {
|
|
|
|
type,
|
|
|
|
size,
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
userId: me.id
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
// get presigned POST ur
|
|
|
|
const s3 = new AWS.S3({ apiVersion: '2006-03-01' })
|
|
|
|
const res = await new Promise((resolve, reject) => {
|
|
|
|
s3.createPresignedPost({
|
2022-05-16 15:51:22 -05:00
|
|
|
Bucket: process.env.NEXT_PUBLIC_AWS_UPLOAD_BUCKET,
|
2022-05-12 13:44:21 -05:00
|
|
|
Fields: {
|
|
|
|
key: String(upload.id)
|
|
|
|
},
|
|
|
|
Expires: 300,
|
|
|
|
Conditions: [
|
|
|
|
{ 'Content-Type': type },
|
2022-05-18 11:59:59 -05:00
|
|
|
{ 'Cache-Control': 'max-age=31536000' },
|
2022-05-12 13:44:21 -05:00
|
|
|
{ acl: 'public-read' },
|
|
|
|
['content-length-range', size, size]
|
|
|
|
]
|
|
|
|
}, (err, preSigned) => { if (err) { reject(err) } else { resolve(preSigned) } })
|
|
|
|
})
|
|
|
|
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|