-
Notifications
You must be signed in to change notification settings - Fork 22
/
Copy pathfileUpload.ts
89 lines (66 loc) · 2.03 KB
/
fileUpload.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
import { GraphQLScalarType } from "./deps.ts";
export const GraphQLUpload = new GraphQLScalarType({
name: "Upload",
description: "The `Upload` scalar type represents a file upload.",
parseValue: (value: any) => value,
parseLiteral() {
throw new Error("`Upload` scalar literal unsupported.");
},
serialize() {
throw new Error("`Upload` scalar serialization unsupported.");
},
});
// const isMultipart = (contentType: string) => {};
const setPath = (object: any, path: string[], value: any) => {
console.log("setPath", object, path, value);
if (!object) return;
const [key, ...rest] = path;
if (path.length === 1) {
object[key] = value;
} else if (path.length > 1) {
if (!object[key]) {
object[key] = {};
}
setPath(object[key], rest, value);
}
};
export const fileUploadMiddleware = async (ctx: any, next: any) => {
const formData = await parseFormData(ctx.request);
const operations = getField(formData, 'operations');
if(!operations) {
throw new Error('operations not provided');
}
const map = getField(formData, 'map');
if(!map) {
throw new Error('map not provided');
}
// Populate operations from other fields
for(let [key, value] of formData) {
if(key !== 'operations' && key !== 'map') {
const paths = map[key] as string[];
if(paths) {
paths.forEach(path => setPath(operations, path.split('.'), value))
}
}
}
ctx.params.operations = operations;
await next();
};
function getField(form: FormData, name: any) {
const value = form.get(name) as string;
if(value) {
return JSON.parse(value);
}
}
async function parseFormData(request: Request) {
const contentType = request.headers.get('content-type');
if(!contentType || !contentType.startsWith('multipart/form-data')) {
throw new Error('Invalid content-type');
}
const formData = new FormData();
// Native FormData handles parsing for us
formData.append('operations', '{}');
formData.append('map', '{}');
await request.formData();
return formData;
}