@balena/pinejs 15.1.0-build-web-resource-4-c82f246bf4abb69c47ca357f99014eb18f1434a8-1 → 15.1.0-build-upsert-6f9a458cc52c017ddaf48ceb7e4597522cdf4dfc-1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (36) hide show
  1. package/.pinejs-cache.json +1 -1
  2. package/.versionbot/CHANGELOG.yml +12 -11
  3. package/CHANGELOG.md +7 -3
  4. package/README.md +0 -10
  5. package/docker-compose.npm-test.yml +1 -22
  6. package/out/config-loader/config-loader.d.ts +0 -2
  7. package/out/config-loader/config-loader.js +2 -13
  8. package/out/config-loader/config-loader.js.map +1 -1
  9. package/out/migrator/sync.js +24 -1
  10. package/out/migrator/sync.js.map +1 -1
  11. package/out/migrator/utils.d.ts +0 -2
  12. package/out/migrator/utils.js +1 -26
  13. package/out/migrator/utils.js.map +1 -1
  14. package/out/sbvr-api/sbvr-utils.d.ts +4 -7
  15. package/out/sbvr-api/sbvr-utils.js +46 -11
  16. package/out/sbvr-api/sbvr-utils.js.map +1 -1
  17. package/out/server-glue/server.js +2 -0
  18. package/out/server-glue/server.js.map +1 -1
  19. package/package.json +12 -17
  20. package/src/config-loader/config-loader.ts +2 -27
  21. package/src/migrator/sync.ts +26 -1
  22. package/src/migrator/utils.ts +0 -28
  23. package/src/sbvr-api/sbvr-utils.ts +62 -11
  24. package/src/server-glue/server.ts +3 -0
  25. package/out/server-glue/webresource-handler.d.ts +0 -21
  26. package/out/server-glue/webresource-handler.js +0 -219
  27. package/out/server-glue/webresource-handler.js.map +0 -1
  28. package/out/server-glue/webresource-handlers/NoopHandler.d.ts +0 -5
  29. package/out/server-glue/webresource-handlers/NoopHandler.js +0 -17
  30. package/out/server-glue/webresource-handlers/NoopHandler.js.map +0 -1
  31. package/out/server-glue/webresource-handlers/S3Handler.d.ts +0 -12
  32. package/out/server-glue/webresource-handlers/S3Handler.js +0 -71
  33. package/out/server-glue/webresource-handlers/S3Handler.js.map +0 -1
  34. package/src/server-glue/webresource-handler.ts +0 -324
  35. package/src/server-glue/webresource-handlers/NoopHandler.ts +0 -20
  36. package/src/server-glue/webresource-handlers/S3Handler.ts +0 -99
@@ -1,219 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.setupUploadHooks = exports.getDefaultHandler = exports.getUploaderMiddlware = void 0;
4
- const busboy = require("busboy");
5
- const is = require("type-is");
6
- const uriParser = require("../sbvr-api/uri-parser");
7
- const sbvr_utils_1 = require("../sbvr-api/sbvr-utils");
8
- const permissions_1 = require("../sbvr-api/permissions");
9
- const sbvrUtils = require("../sbvr-api/sbvr-utils");
10
- const S3Handler_1 = require("./webresource-handlers/S3Handler");
11
- const NoopHandler_1 = require("./webresource-handlers/NoopHandler");
12
- const odata_to_abstract_sql_1 = require("@balena/odata-to-abstract-sql");
13
- const module_1 = require("./module");
14
- const ifFileInValidPath = async (fieldname, req) => {
15
- if (req.method !== 'POST' && req.method !== 'PATCH') {
16
- return false;
17
- }
18
- const apiRoot = (0, sbvr_utils_1.getApiRoot)(req);
19
- if (apiRoot == null) {
20
- return false;
21
- }
22
- const model = (0, sbvr_utils_1.getModel)(apiRoot);
23
- const { resourceName } = await uriParser.parseOData({
24
- url: req.url,
25
- method: req.method,
26
- });
27
- const permission = req.method === 'POST' ? 'create' : 'update';
28
- const vocab = model.versions[model.versions.length - 1];
29
- const hasPermissions = await (0, permissions_1.checkPermissions)(req, permission, resourceName, vocab);
30
- if (!hasPermissions) {
31
- return false;
32
- }
33
- const fields = model.abstractSql.tables[resourceName].fields;
34
- const dbFieldName = (0, odata_to_abstract_sql_1.odataNameToSqlName)(fieldname);
35
- for (const field of fields) {
36
- if (field.fieldName === dbFieldName && field.dataType === 'WebResource') {
37
- return true;
38
- }
39
- }
40
- return false;
41
- };
42
- const getUploaderMiddlware = (handler) => {
43
- return async (req, res, next) => {
44
- if (!is(req, ['multipart'])) {
45
- return next();
46
- }
47
- const filesUploaded = [];
48
- const completeUploads = [];
49
- const bb = busboy({ headers: req.headers });
50
- let isAborting = false;
51
- const done = () => {
52
- req.unpipe(bb);
53
- req.on('readable', req.read.bind(req));
54
- bb.removeAllListeners();
55
- };
56
- const clearFiles = () => {
57
- isAborting = true;
58
- const deletions = filesUploaded.map((file) => handler.removeFile(file));
59
- return Promise.all(deletions).catch((err) => console.error('Error deleting file', err));
60
- };
61
- bb.on('file', async (fieldname, filestream, info) => {
62
- if (!isAborting && (await ifFileInValidPath(fieldname, req))) {
63
- const file = {
64
- originalname: info.filename,
65
- encoding: info.encoding,
66
- mimetype: info.mimeType,
67
- stream: filestream,
68
- fieldname,
69
- };
70
- const promise = handler.handleFile(file).then((result) => {
71
- req.body[fieldname] = {
72
- filename: info.filename,
73
- contentType: info.mimeType,
74
- contentDisposition: undefined,
75
- size: result.size,
76
- href: result.filename,
77
- };
78
- filesUploaded.push(result.filename);
79
- });
80
- completeUploads.push(promise);
81
- }
82
- else {
83
- filestream.resume();
84
- }
85
- });
86
- bb.on('field', (name, val, _info) => {
87
- req.body[name] = val;
88
- });
89
- bb.on('finish', async () => {
90
- try {
91
- await Promise.all(completeUploads);
92
- done();
93
- next();
94
- }
95
- catch (err) {
96
- await clearFiles();
97
- if (err?.message.includes('File size exceeded')) {
98
- return sbvrUtils.handleHttpErrors(req, res, new module_1.errors.BadRequestError('File size exceeded'));
99
- }
100
- console.error('Error uploading file', err);
101
- next(err);
102
- }
103
- });
104
- bb.on('error', async (err) => {
105
- await clearFiles();
106
- done();
107
- next(err);
108
- });
109
- req.pipe(bb);
110
- };
111
- };
112
- exports.getUploaderMiddlware = getUploaderMiddlware;
113
- const getWebResourceFields = (request) => {
114
- const fields = request.abstractSqlModel?.tables[request.resourceName]?.modifyFields ??
115
- request.abstractSqlModel?.tables[request.resourceName]?.fields;
116
- if (fields == null) {
117
- return [];
118
- }
119
- return fields
120
- .filter((f) => f.dataType === 'WebResource')
121
- .map((f) => (0, odata_to_abstract_sql_1.sqlNameToODataName)(f.fieldName));
122
- };
123
- const getModifiedFields = (request) => {
124
- return Object.entries(request.values)
125
- .filter(([_key, value]) => value !== undefined)
126
- .map(([key, _value]) => key);
127
- };
128
- const deleteFiles = async (keysToDelete, webResourceHandler) => {
129
- const promises = keysToDelete.map((r) => webResourceHandler.removeFile(r));
130
- await Promise.all(promises);
131
- };
132
- const getCreateWebResourceHook = (webResourceHandler) => {
133
- return {
134
- 'POSTRUN-ERROR': async ({ tx, request }) => {
135
- tx?.on('rollback', async () => {
136
- const fields = getWebResourceFields(request);
137
- if (fields.length === 0) {
138
- return;
139
- }
140
- const keysToDelete = fields
141
- .filter((f) => isDefined(request.values[f]))
142
- .map((f) => request.values[f].href);
143
- await deleteFiles(keysToDelete, webResourceHandler);
144
- });
145
- },
146
- };
147
- };
148
- const isDefined = (x) => x != null;
149
- const getWebResourcesHrefs = (webResources) => {
150
- const hrefs = Object.values(webResources ?? {})
151
- .filter(isDefined)
152
- .map((resourceKey) => resourceKey.href);
153
- return hrefs;
154
- };
155
- const getRemoveWebResourceHook = (webResourceHandler) => {
156
- return {
157
- PRERUN: async (args) => {
158
- const { api, request } = args;
159
- let fields = getWebResourceFields(request);
160
- if (request.method === 'PATCH') {
161
- if (request.odataQuery?.key == null) {
162
- throw new module_1.errors.BadRequestError('WebResources can only be updated when providing a resource key.');
163
- }
164
- const allFields = getModifiedFields(request);
165
- fields = fields.filter((f) => allFields.includes(f));
166
- }
167
- if (fields.length === 0) {
168
- return;
169
- }
170
- const [id, rest] = await sbvrUtils.getAffectedIds(args);
171
- if (id == null) {
172
- return;
173
- }
174
- if (rest != null) {
175
- throw new module_1.errors.BadRequestError('Resources containing webresources can only be updated/deleted one at a time.');
176
- }
177
- const webResources = (await api.get({
178
- resource: request.resourceName,
179
- passthrough: {
180
- tx: args.tx,
181
- req: module_1.permissions.root,
182
- },
183
- id,
184
- options: {
185
- $select: fields,
186
- },
187
- }));
188
- request.custom.$pineWebResourcesToDelete =
189
- getWebResourcesHrefs(webResources);
190
- },
191
- POSTRUN: ({ tx, request }) => {
192
- tx.on('end', () => {
193
- const keysToDelete = request.custom.$pineWebResourcesToDelete || [];
194
- deleteFiles(keysToDelete, webResourceHandler).catch((err) => {
195
- console.error(`Failed to delete pending files`, err);
196
- });
197
- });
198
- },
199
- };
200
- };
201
- const getDefaultHandler = () => {
202
- let handler;
203
- try {
204
- handler = new S3Handler_1.S3Handler();
205
- }
206
- catch (e) {
207
- console.warn(`Failed to initialize S3 handler, using noop ${e}`);
208
- handler = new NoopHandler_1.NoopHandler();
209
- }
210
- return handler;
211
- };
212
- exports.getDefaultHandler = getDefaultHandler;
213
- const setupUploadHooks = (handler, apiRoot, resourceName) => {
214
- sbvrUtils.addPureHook('DELETE', apiRoot, resourceName, getRemoveWebResourceHook(handler));
215
- sbvrUtils.addPureHook('PATCH', apiRoot, resourceName, getRemoveWebResourceHook(handler));
216
- sbvrUtils.addPureHook('POST', apiRoot, resourceName, getCreateWebResourceHook(handler));
217
- };
218
- exports.setupUploadHooks = setupUploadHooks;
219
- //# sourceMappingURL=webresource-handler.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"webresource-handler.js","sourceRoot":"","sources":["../../src/server-glue/webresource-handler.ts"],"names":[],"mappings":"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"}
@@ -1,5 +0,0 @@
1
- import { IncomingFile, UploadResponse, WebResourceHandler } from '../webresource-handler';
2
- export declare class NoopHandler implements WebResourceHandler {
3
- handleFile(resource: IncomingFile): Promise<UploadResponse>;
4
- removeFile(_fileReference: string): Promise<void>;
5
- }
@@ -1,17 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.NoopHandler = void 0;
4
- class NoopHandler {
5
- async handleFile(resource) {
6
- resource.stream.resume();
7
- return {
8
- filename: 'noop',
9
- size: 0,
10
- };
11
- }
12
- async removeFile(_fileReference) {
13
- return;
14
- }
15
- }
16
- exports.NoopHandler = NoopHandler;
17
- //# sourceMappingURL=NoopHandler.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"NoopHandler.js","sourceRoot":"","sources":["../../../src/server-glue/webresource-handlers/NoopHandler.ts"],"names":[],"mappings":";;;AAMA,MAAa,WAAW;IAChB,KAAK,CAAC,UAAU,CAAC,QAAsB;QAE7C,QAAQ,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;QACzB,OAAO;YACN,QAAQ,EAAE,MAAM;YAChB,IAAI,EAAE,CAAC;SACP,CAAC;IACH,CAAC;IAEM,KAAK,CAAC,UAAU,CAAC,cAAsB;QAC7C,OAAO;IACR,CAAC;CACD;AAbD,kCAaC"}
@@ -1,12 +0,0 @@
1
- import { IncomingFile, UploadResponse, WebResourceHandler } from '../webresource-handler';
2
- export declare class S3Handler implements WebResourceHandler {
3
- private readonly config;
4
- private readonly bucket;
5
- private readonly endpoint;
6
- private readonly maxFileSize;
7
- private client;
8
- constructor();
9
- handleFile(resource: IncomingFile): Promise<UploadResponse>;
10
- removeFile(fileReference: string): Promise<void>;
11
- private getS3URL;
12
- }
@@ -1,71 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.S3Handler = void 0;
4
- const env_parsing_1 = require("@balena/env-parsing");
5
- const client_s3_1 = require("@aws-sdk/client-s3");
6
- const lib_storage_1 = require("@aws-sdk/lib-storage");
7
- const crypto_1 = require("crypto");
8
- class S3Handler {
9
- constructor() {
10
- this.maxFileSize = (0, env_parsing_1.intVar)('PINEJS_WEBRESOURCE_MAXFILESIZE', 52428800);
11
- this.endpoint = (0, env_parsing_1.requiredVar)('S3_ENDPOINT');
12
- this.config = {
13
- region: (0, env_parsing_1.optionalVar)('S3_REGION', 'us-east-1'),
14
- credentials: {
15
- accessKeyId: (0, env_parsing_1.requiredVar)('S3_ACCESS_KEY'),
16
- secretAccessKey: (0, env_parsing_1.requiredVar)('S3_SECRET_KEY'),
17
- },
18
- endpoint: this.endpoint,
19
- forcePathStyle: true,
20
- };
21
- this.bucket = (0, env_parsing_1.optionalVar)('S3_STORAGE_ADAPTER_BUCKET', 'balena-pine-web-resources');
22
- this.client = new client_s3_1.S3Client(this.config);
23
- }
24
- async handleFile(resource) {
25
- let size = 0;
26
- const key = `${resource.fieldname}_${(0, crypto_1.randomUUID)()}_${resource.originalname}`;
27
- const params = {
28
- Bucket: this.bucket,
29
- ACL: 'public-read',
30
- StorageClass: 'STANDARD',
31
- Key: key,
32
- Body: resource.stream,
33
- ContentType: resource.mimetype,
34
- };
35
- const upload = new lib_storage_1.Upload({ client: this.client, params });
36
- upload.on('httpUploadProgress', async (ev) => {
37
- size = ev.total ? ev.total : ev.loaded;
38
- if (size > this.maxFileSize) {
39
- await upload.abort();
40
- }
41
- });
42
- try {
43
- await upload.done();
44
- }
45
- catch (err) {
46
- resource.stream.resume();
47
- if (size > this.maxFileSize) {
48
- throw new Error('File size exceeded');
49
- }
50
- throw err;
51
- }
52
- const filename = this.getS3URL(key);
53
- return { size, filename };
54
- }
55
- async removeFile(fileReference) {
56
- const fileReferences = fileReference.split('/');
57
- const fileKey = fileReferences[fileReferences.length - 1];
58
- const command = new client_s3_1.DeleteObjectCommand({
59
- Bucket: this.bucket,
60
- Key: fileKey,
61
- });
62
- await this.client.send(command);
63
- }
64
- getS3URL(key) {
65
- return this.endpoint.includes(this.bucket)
66
- ? `${this.endpoint}/${key}`
67
- : `${this.endpoint}/${this.bucket}/${key}`;
68
- }
69
- }
70
- exports.S3Handler = S3Handler;
71
- //# sourceMappingURL=S3Handler.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"S3Handler.js","sourceRoot":"","sources":["../../../src/server-glue/webresource-handlers/S3Handler.ts"],"names":[],"mappings":"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"}
@@ -1,324 +0,0 @@
1
- import type * as Express from 'express';
2
- import * as busboy from 'busboy';
3
- import * as is from 'type-is';
4
- import * as stream from 'stream';
5
- import * as uriParser from '../sbvr-api/uri-parser';
6
- import { getApiRoot, getModel } from '../sbvr-api/sbvr-utils';
7
- import { checkPermissions } from '../sbvr-api/permissions';
8
- import * as sbvrUtils from '../sbvr-api/sbvr-utils';
9
- import { S3Handler } from './webresource-handlers/S3Handler';
10
- import { NoopHandler } from './webresource-handlers/NoopHandler';
11
- import {
12
- odataNameToSqlName,
13
- sqlNameToODataName,
14
- } from '@balena/odata-to-abstract-sql';
15
- import { errors, permissions } from './module';
16
-
17
- export interface IncomingFile {
18
- fieldname: string;
19
- originalname: string;
20
- encoding: string;
21
- mimetype: string;
22
- stream: stream.Readable;
23
- }
24
-
25
- export interface UploadResponse {
26
- size: number;
27
- filename: string;
28
- }
29
-
30
- export interface WebResourceHandler {
31
- handleFile: (resource: IncomingFile) => Promise<UploadResponse>;
32
- removeFile: (fileReference: string) => Promise<void>;
33
- }
34
-
35
- type WebResourcesDbResponse = {
36
- [fieldname: string]: { href: string } | undefined | null;
37
- };
38
-
39
- const ifFileInValidPath = async (
40
- fieldname: string,
41
- req: Express.Request,
42
- ): Promise<boolean> => {
43
- if (req.method !== 'POST' && req.method !== 'PATCH') {
44
- return false;
45
- }
46
-
47
- const apiRoot = getApiRoot(req);
48
- if (apiRoot == null) {
49
- return false;
50
- }
51
- const model = getModel(apiRoot);
52
- const { resourceName } = await uriParser.parseOData({
53
- url: req.url,
54
- method: req.method,
55
- });
56
-
57
- const permission = req.method === 'POST' ? 'create' : 'update';
58
- const vocab = model.versions[model.versions.length - 1];
59
- const hasPermissions = await checkPermissions(
60
- req,
61
- permission,
62
- resourceName,
63
- vocab,
64
- );
65
-
66
- if (!hasPermissions) {
67
- return false;
68
- }
69
-
70
- // TODO: This could be cached
71
- const fields = model.abstractSql.tables[resourceName].fields;
72
- const dbFieldName = odataNameToSqlName(fieldname);
73
- for (const field of fields) {
74
- if (field.fieldName === dbFieldName && field.dataType === 'WebResource') {
75
- return true;
76
- }
77
- }
78
-
79
- // TODO: We could do a pre-check if there is a SBVR rule specifying file max size
80
- // This would avoid needing the roundtrip to DB and uploading a file just to remove it
81
-
82
- return false;
83
- };
84
-
85
- export const getUploaderMiddlware = (
86
- handler: WebResourceHandler,
87
- ): Express.RequestHandler => {
88
- return async (req, res, next) => {
89
- if (!is(req, ['multipart'])) {
90
- return next();
91
- }
92
- const filesUploaded: string[] = [];
93
- const completeUploads: Array<Promise<void>> = [];
94
-
95
- const bb = busboy({ headers: req.headers });
96
- let isAborting = false;
97
-
98
- const done = () => {
99
- req.unpipe(bb);
100
- req.on('readable', req.read.bind(req));
101
- bb.removeAllListeners();
102
- };
103
-
104
- const clearFiles = () => {
105
- isAborting = true;
106
- const deletions = filesUploaded.map((file) => handler.removeFile(file));
107
- // Best effort: We try to remove all uploaded files, but if this fails, there is not much to do
108
- return Promise.all(deletions).catch((err) =>
109
- console.error('Error deleting file', err),
110
- );
111
- };
112
-
113
- bb.on('file', async (fieldname, filestream, info) => {
114
- if (!isAborting && (await ifFileInValidPath(fieldname, req))) {
115
- const file: IncomingFile = {
116
- originalname: info.filename,
117
- encoding: info.encoding,
118
- mimetype: info.mimeType,
119
- stream: filestream,
120
- fieldname,
121
- };
122
- const promise = handler.handleFile(file).then((result) => {
123
- req.body[fieldname] = {
124
- filename: info.filename,
125
- contentType: info.mimeType,
126
- contentDisposition: undefined,
127
- size: result.size,
128
- href: result.filename,
129
- };
130
- filesUploaded.push(result.filename);
131
- });
132
- completeUploads.push(promise);
133
- } else {
134
- filestream.resume();
135
- }
136
- });
137
-
138
- bb.on('field', (name, val, _info) => {
139
- req.body[name] = val;
140
- });
141
-
142
- bb.on('finish', async () => {
143
- try {
144
- await Promise.all(completeUploads);
145
- done();
146
- next();
147
- } catch (err: any) {
148
- await clearFiles();
149
-
150
- if (err?.message.includes('File size exceeded')) {
151
- return sbvrUtils.handleHttpErrors(
152
- req,
153
- res,
154
- new errors.BadRequestError('File size exceeded'),
155
- );
156
- }
157
-
158
- console.error('Error uploading file', err);
159
- next(err);
160
- }
161
- });
162
-
163
- bb.on('error', async (err) => {
164
- await clearFiles();
165
- done();
166
- next(err);
167
- });
168
- req.pipe(bb);
169
- };
170
- };
171
-
172
- // TODO: this can be cached
173
- const getWebResourceFields = (request: uriParser.ODataRequest): string[] => {
174
- const fields =
175
- request.abstractSqlModel?.tables[request.resourceName]?.modifyFields ??
176
- request.abstractSqlModel?.tables[request.resourceName]?.fields;
177
- if (fields == null) {
178
- return [];
179
- }
180
-
181
- return fields
182
- .filter((f) => f.dataType === 'WebResource')
183
- .map((f) => sqlNameToODataName(f.fieldName));
184
- };
185
-
186
- const getModifiedFields = (request: uriParser.ODataRequest): string[] => {
187
- return Object.entries(request.values)
188
- .filter(([_key, value]) => value !== undefined)
189
- .map(([key, _value]) => key);
190
- };
191
-
192
- const deleteFiles = async (
193
- keysToDelete: string[],
194
- webResourceHandler: WebResourceHandler,
195
- ) => {
196
- const promises = keysToDelete.map((r) => webResourceHandler.removeFile(r));
197
- await Promise.all(promises);
198
- };
199
-
200
- const getCreateWebResourceHook = (webResourceHandler: WebResourceHandler) => {
201
- return {
202
- 'POSTRUN-ERROR': async ({ tx, request }) => {
203
- tx?.on('rollback', async () => {
204
- const fields = getWebResourceFields(request);
205
-
206
- if (fields.length === 0) {
207
- return;
208
- }
209
-
210
- const keysToDelete: string[] = fields
211
- .filter((f) => isDefined(request.values[f]))
212
- .map((f) => request.values[f].href);
213
- await deleteFiles(keysToDelete, webResourceHandler);
214
- });
215
- },
216
- } as sbvrUtils.Hooks;
217
- };
218
-
219
- const isDefined = <T>(x: T | undefined | null): x is T => x != null;
220
-
221
- const getWebResourcesHrefs = (webResources?: WebResourcesDbResponse | null) => {
222
- const hrefs = Object.values(webResources ?? {})
223
- .filter(isDefined)
224
- .map((resourceKey) => resourceKey.href);
225
- return hrefs;
226
- };
227
-
228
- const getRemoveWebResourceHook = (webResourceHandler: WebResourceHandler) => {
229
- return {
230
- PRERUN: async (args) => {
231
- const { api, request } = args;
232
- let fields = getWebResourceFields(request);
233
-
234
- if (request.method === 'PATCH') {
235
- if (request.odataQuery?.key == null) {
236
- throw new errors.BadRequestError(
237
- 'WebResources can only be updated when providing a resource key.',
238
- );
239
- }
240
- const allFields = getModifiedFields(request);
241
- fields = fields.filter((f) => allFields.includes(f));
242
- }
243
-
244
- if (fields.length === 0) {
245
- return;
246
- }
247
-
248
- const [id, rest] = await sbvrUtils.getAffectedIds(args);
249
- if (id == null) {
250
- return;
251
- }
252
-
253
- if (rest != null) {
254
- throw new errors.BadRequestError(
255
- 'Resources containing webresources can only be updated/deleted one at a time.',
256
- );
257
- }
258
-
259
- const webResources = (await api.get({
260
- resource: request.resourceName,
261
- passthrough: {
262
- tx: args.tx,
263
- req: permissions.root,
264
- },
265
- id,
266
- options: {
267
- $select: fields,
268
- },
269
- })) as WebResourcesDbResponse | undefined | null;
270
-
271
- request.custom.$pineWebResourcesToDelete =
272
- getWebResourcesHrefs(webResources);
273
- },
274
- POSTRUN: ({ tx, request }) => {
275
- tx.on('end', () => {
276
- const keysToDelete: string[] =
277
- request.custom.$pineWebResourcesToDelete || [];
278
- // on purpose does not await for this promise to resolve
279
- deleteFiles(keysToDelete, webResourceHandler).catch((err) => {
280
- console.error(`Failed to delete pending files`, err);
281
- });
282
- });
283
- },
284
- } as sbvrUtils.Hooks;
285
- };
286
-
287
- export const getDefaultHandler = (): WebResourceHandler => {
288
- let handler: WebResourceHandler;
289
- try {
290
- handler = new S3Handler();
291
- } catch (e) {
292
- console.warn(`Failed to initialize S3 handler, using noop ${e}`);
293
- handler = new NoopHandler();
294
- }
295
- return handler;
296
- };
297
-
298
- export const setupUploadHooks = (
299
- handler: WebResourceHandler,
300
- apiRoot: string,
301
- resourceName: string,
302
- ) => {
303
- sbvrUtils.addPureHook(
304
- 'DELETE',
305
- apiRoot,
306
- resourceName,
307
- getRemoveWebResourceHook(handler),
308
- );
309
-
310
- sbvrUtils.addPureHook(
311
- 'PATCH',
312
- apiRoot,
313
- resourceName,
314
- // PATCH also needs to remove the old resource in case a webresource was modified
315
- getRemoveWebResourceHook(handler),
316
- );
317
-
318
- sbvrUtils.addPureHook(
319
- 'POST',
320
- apiRoot,
321
- resourceName,
322
- getCreateWebResourceHook(handler),
323
- );
324
- };