fauxqs 2.0.0 → 2.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +4 -2
- package/dist/persistence.js +130 -130
- package/dist/s3/actions/postObject.d.ts +21 -0
- package/dist/s3/actions/postObject.d.ts.map +1 -0
- package/dist/s3/actions/postObject.js +166 -0
- package/dist/s3/actions/postObject.js.map +1 -0
- package/dist/s3/s3Router.d.ts.map +1 -1
- package/dist/s3/s3Router.js +13 -2
- package/dist/s3/s3Router.js.map +1 -1
- package/package.json +2 -1
package/README.md
CHANGED
|
@@ -629,7 +629,7 @@ The spy tracks events across all three services using a discriminated union on `
|
|
|
629
629
|
- **`published`** — message was published to a topic (before fan-out to SQS subscriptions)
|
|
630
630
|
|
|
631
631
|
**S3 events** (`service: 's3'`):
|
|
632
|
-
- **`uploaded`** — object was put (PutObject or CompleteMultipartUpload)
|
|
632
|
+
- **`uploaded`** — object was put (PutObject, PostObject, or CompleteMultipartUpload)
|
|
633
633
|
- **`downloaded`** — object was retrieved (GetObject)
|
|
634
634
|
- **`deleted`** — object was deleted (DeleteObject, only when key existed)
|
|
635
635
|
- **`copied`** — object was copied (CopyObject; also emits `uploaded` for the destination)
|
|
@@ -995,6 +995,7 @@ Platform application, SMS, and phone number actions are not supported.
|
|
|
995
995
|
| ListObjects | Yes |
|
|
996
996
|
| ListObjectsV2 | Yes |
|
|
997
997
|
| PutObject | Yes |
|
|
998
|
+
| PostObject | Yes |
|
|
998
999
|
| GetObject | Yes |
|
|
999
1000
|
| HeadObject | Yes |
|
|
1000
1001
|
| DeleteObject | Yes |
|
|
@@ -1058,10 +1059,11 @@ Returns a mock identity with account `000000000000` and ARN `arn:aws:iam::000000
|
|
|
1058
1059
|
## S3 Features
|
|
1059
1060
|
|
|
1060
1061
|
- **Bucket management** — CreateBucket (idempotent), DeleteBucket (rejects non-empty), HeadBucket, ListBuckets, ListObjects (V1 and V2)
|
|
1061
|
-
- **Object operations** — PutObject, GetObject, DeleteObject, HeadObject, CopyObject with ETag, Content-Type, and Last-Modified headers
|
|
1062
|
+
- **Object operations** — PutObject, PostObject (presigned POST form uploads), GetObject, DeleteObject, HeadObject, CopyObject with ETag, Content-Type, and Last-Modified headers
|
|
1062
1063
|
- **Multipart uploads** — CreateMultipartUpload, UploadPart, UploadPartCopy, CompleteMultipartUpload, AbortMultipartUpload with correct multipart ETag calculation (`MD5-of-part-digests-partCount`), metadata preservation, and part overwrite support
|
|
1063
1064
|
- **ListObjects V2** — prefix filtering, delimiter-based virtual directories, MaxKeys, continuation tokens, StartAfter
|
|
1064
1065
|
- **CopyObject** — same-bucket and cross-bucket copy via `x-amz-copy-source` header, with metadata preservation
|
|
1066
|
+
- **PostObject** — presigned POST form uploads (`multipart/form-data`). Supports `key` field with `${filename}` substitution, `Content-Type`, `success_action_status` (200/201/204), `success_action_redirect`, and user metadata via `x-amz-meta-*` fields. Policy signature validation is skipped (mock server).
|
|
1065
1067
|
- **User metadata** — `x-amz-meta-*` headers are stored and returned on GetObject and HeadObject
|
|
1066
1068
|
- **Bulk delete** — DeleteObjects for batch key deletion with proper XML entity handling
|
|
1067
1069
|
- **Keys with slashes** — full support for slash-delimited keys (e.g., `path/to/file.txt`)
|
package/dist/persistence.js
CHANGED
|
@@ -1,103 +1,103 @@
|
|
|
1
1
|
import { mkdirSync } from "node:fs";
|
|
2
2
|
import { join } from "node:path";
|
|
3
3
|
import { DatabaseSync } from "node:sqlite";
|
|
4
|
-
const SCHEMA = `
|
|
5
|
-
CREATE TABLE IF NOT EXISTS sqs_queues (
|
|
6
|
-
name TEXT PRIMARY KEY,
|
|
7
|
-
url TEXT NOT NULL,
|
|
8
|
-
arn TEXT NOT NULL,
|
|
9
|
-
attributes TEXT NOT NULL,
|
|
10
|
-
tags TEXT NOT NULL,
|
|
11
|
-
created_timestamp INTEGER NOT NULL,
|
|
12
|
-
last_modified_timestamp INTEGER NOT NULL,
|
|
13
|
-
sequence_counter INTEGER NOT NULL DEFAULT 0
|
|
14
|
-
);
|
|
15
|
-
|
|
16
|
-
CREATE TABLE IF NOT EXISTS sqs_messages (
|
|
17
|
-
message_id TEXT PRIMARY KEY,
|
|
18
|
-
queue_name TEXT NOT NULL REFERENCES sqs_queues(name) ON DELETE CASCADE,
|
|
19
|
-
body TEXT NOT NULL,
|
|
20
|
-
md5_of_body TEXT NOT NULL,
|
|
21
|
-
message_attributes TEXT NOT NULL,
|
|
22
|
-
md5_of_message_attributes TEXT NOT NULL,
|
|
23
|
-
sent_timestamp INTEGER NOT NULL,
|
|
24
|
-
approximate_receive_count INTEGER NOT NULL DEFAULT 0,
|
|
25
|
-
approximate_first_receive_timestamp INTEGER,
|
|
26
|
-
delay_until INTEGER,
|
|
27
|
-
message_group_id TEXT,
|
|
28
|
-
message_deduplication_id TEXT,
|
|
29
|
-
sequence_number TEXT,
|
|
30
|
-
receipt_handle TEXT,
|
|
31
|
-
visibility_deadline INTEGER
|
|
32
|
-
);
|
|
33
|
-
|
|
34
|
-
CREATE TABLE IF NOT EXISTS sns_topics (
|
|
35
|
-
arn TEXT PRIMARY KEY,
|
|
36
|
-
name TEXT NOT NULL,
|
|
37
|
-
attributes TEXT NOT NULL,
|
|
38
|
-
tags TEXT NOT NULL,
|
|
39
|
-
subscription_arns TEXT NOT NULL
|
|
40
|
-
);
|
|
41
|
-
|
|
42
|
-
CREATE TABLE IF NOT EXISTS sns_subscriptions (
|
|
43
|
-
arn TEXT PRIMARY KEY,
|
|
44
|
-
topic_arn TEXT NOT NULL,
|
|
45
|
-
protocol TEXT NOT NULL,
|
|
46
|
-
endpoint TEXT NOT NULL,
|
|
47
|
-
confirmed INTEGER NOT NULL DEFAULT 0,
|
|
48
|
-
attributes TEXT NOT NULL
|
|
49
|
-
);
|
|
50
|
-
|
|
51
|
-
CREATE TABLE IF NOT EXISTS s3_buckets (
|
|
52
|
-
name TEXT PRIMARY KEY,
|
|
53
|
-
creation_date TEXT NOT NULL,
|
|
54
|
-
type TEXT NOT NULL DEFAULT 'general-purpose'
|
|
55
|
-
);
|
|
56
|
-
|
|
57
|
-
CREATE TABLE IF NOT EXISTS s3_objects (
|
|
58
|
-
bucket TEXT NOT NULL,
|
|
59
|
-
key TEXT NOT NULL,
|
|
60
|
-
body BLOB NOT NULL,
|
|
61
|
-
content_type TEXT NOT NULL,
|
|
62
|
-
content_length INTEGER NOT NULL,
|
|
63
|
-
etag TEXT NOT NULL,
|
|
64
|
-
last_modified TEXT NOT NULL,
|
|
65
|
-
metadata TEXT NOT NULL,
|
|
66
|
-
content_language TEXT,
|
|
67
|
-
content_disposition TEXT,
|
|
68
|
-
cache_control TEXT,
|
|
69
|
-
content_encoding TEXT,
|
|
70
|
-
parts TEXT,
|
|
71
|
-
checksum_algorithm TEXT,
|
|
72
|
-
checksum_value TEXT,
|
|
73
|
-
checksum_type TEXT,
|
|
74
|
-
part_checksums TEXT,
|
|
75
|
-
PRIMARY KEY (bucket, key)
|
|
76
|
-
);
|
|
77
|
-
|
|
78
|
-
CREATE TABLE IF NOT EXISTS s3_multipart_uploads (
|
|
79
|
-
upload_id TEXT PRIMARY KEY,
|
|
80
|
-
bucket TEXT NOT NULL,
|
|
81
|
-
key TEXT NOT NULL,
|
|
82
|
-
content_type TEXT NOT NULL,
|
|
83
|
-
metadata TEXT NOT NULL,
|
|
84
|
-
initiated TEXT NOT NULL,
|
|
85
|
-
content_language TEXT,
|
|
86
|
-
content_disposition TEXT,
|
|
87
|
-
cache_control TEXT,
|
|
88
|
-
content_encoding TEXT,
|
|
89
|
-
checksum_algorithm TEXT
|
|
90
|
-
);
|
|
91
|
-
|
|
92
|
-
CREATE TABLE IF NOT EXISTS s3_multipart_parts (
|
|
93
|
-
upload_id TEXT NOT NULL REFERENCES s3_multipart_uploads(upload_id) ON DELETE CASCADE,
|
|
94
|
-
part_number INTEGER NOT NULL,
|
|
95
|
-
body BLOB NOT NULL,
|
|
96
|
-
etag TEXT NOT NULL,
|
|
97
|
-
last_modified TEXT NOT NULL,
|
|
98
|
-
checksum_value TEXT,
|
|
99
|
-
PRIMARY KEY (upload_id, part_number)
|
|
100
|
-
);
|
|
4
|
+
const SCHEMA = `
|
|
5
|
+
CREATE TABLE IF NOT EXISTS sqs_queues (
|
|
6
|
+
name TEXT PRIMARY KEY,
|
|
7
|
+
url TEXT NOT NULL,
|
|
8
|
+
arn TEXT NOT NULL,
|
|
9
|
+
attributes TEXT NOT NULL,
|
|
10
|
+
tags TEXT NOT NULL,
|
|
11
|
+
created_timestamp INTEGER NOT NULL,
|
|
12
|
+
last_modified_timestamp INTEGER NOT NULL,
|
|
13
|
+
sequence_counter INTEGER NOT NULL DEFAULT 0
|
|
14
|
+
);
|
|
15
|
+
|
|
16
|
+
CREATE TABLE IF NOT EXISTS sqs_messages (
|
|
17
|
+
message_id TEXT PRIMARY KEY,
|
|
18
|
+
queue_name TEXT NOT NULL REFERENCES sqs_queues(name) ON DELETE CASCADE,
|
|
19
|
+
body TEXT NOT NULL,
|
|
20
|
+
md5_of_body TEXT NOT NULL,
|
|
21
|
+
message_attributes TEXT NOT NULL,
|
|
22
|
+
md5_of_message_attributes TEXT NOT NULL,
|
|
23
|
+
sent_timestamp INTEGER NOT NULL,
|
|
24
|
+
approximate_receive_count INTEGER NOT NULL DEFAULT 0,
|
|
25
|
+
approximate_first_receive_timestamp INTEGER,
|
|
26
|
+
delay_until INTEGER,
|
|
27
|
+
message_group_id TEXT,
|
|
28
|
+
message_deduplication_id TEXT,
|
|
29
|
+
sequence_number TEXT,
|
|
30
|
+
receipt_handle TEXT,
|
|
31
|
+
visibility_deadline INTEGER
|
|
32
|
+
);
|
|
33
|
+
|
|
34
|
+
CREATE TABLE IF NOT EXISTS sns_topics (
|
|
35
|
+
arn TEXT PRIMARY KEY,
|
|
36
|
+
name TEXT NOT NULL,
|
|
37
|
+
attributes TEXT NOT NULL,
|
|
38
|
+
tags TEXT NOT NULL,
|
|
39
|
+
subscription_arns TEXT NOT NULL
|
|
40
|
+
);
|
|
41
|
+
|
|
42
|
+
CREATE TABLE IF NOT EXISTS sns_subscriptions (
|
|
43
|
+
arn TEXT PRIMARY KEY,
|
|
44
|
+
topic_arn TEXT NOT NULL,
|
|
45
|
+
protocol TEXT NOT NULL,
|
|
46
|
+
endpoint TEXT NOT NULL,
|
|
47
|
+
confirmed INTEGER NOT NULL DEFAULT 0,
|
|
48
|
+
attributes TEXT NOT NULL
|
|
49
|
+
);
|
|
50
|
+
|
|
51
|
+
CREATE TABLE IF NOT EXISTS s3_buckets (
|
|
52
|
+
name TEXT PRIMARY KEY,
|
|
53
|
+
creation_date TEXT NOT NULL,
|
|
54
|
+
type TEXT NOT NULL DEFAULT 'general-purpose'
|
|
55
|
+
);
|
|
56
|
+
|
|
57
|
+
CREATE TABLE IF NOT EXISTS s3_objects (
|
|
58
|
+
bucket TEXT NOT NULL,
|
|
59
|
+
key TEXT NOT NULL,
|
|
60
|
+
body BLOB NOT NULL,
|
|
61
|
+
content_type TEXT NOT NULL,
|
|
62
|
+
content_length INTEGER NOT NULL,
|
|
63
|
+
etag TEXT NOT NULL,
|
|
64
|
+
last_modified TEXT NOT NULL,
|
|
65
|
+
metadata TEXT NOT NULL,
|
|
66
|
+
content_language TEXT,
|
|
67
|
+
content_disposition TEXT,
|
|
68
|
+
cache_control TEXT,
|
|
69
|
+
content_encoding TEXT,
|
|
70
|
+
parts TEXT,
|
|
71
|
+
checksum_algorithm TEXT,
|
|
72
|
+
checksum_value TEXT,
|
|
73
|
+
checksum_type TEXT,
|
|
74
|
+
part_checksums TEXT,
|
|
75
|
+
PRIMARY KEY (bucket, key)
|
|
76
|
+
);
|
|
77
|
+
|
|
78
|
+
CREATE TABLE IF NOT EXISTS s3_multipart_uploads (
|
|
79
|
+
upload_id TEXT PRIMARY KEY,
|
|
80
|
+
bucket TEXT NOT NULL,
|
|
81
|
+
key TEXT NOT NULL,
|
|
82
|
+
content_type TEXT NOT NULL,
|
|
83
|
+
metadata TEXT NOT NULL,
|
|
84
|
+
initiated TEXT NOT NULL,
|
|
85
|
+
content_language TEXT,
|
|
86
|
+
content_disposition TEXT,
|
|
87
|
+
cache_control TEXT,
|
|
88
|
+
content_encoding TEXT,
|
|
89
|
+
checksum_algorithm TEXT
|
|
90
|
+
);
|
|
91
|
+
|
|
92
|
+
CREATE TABLE IF NOT EXISTS s3_multipart_parts (
|
|
93
|
+
upload_id TEXT NOT NULL REFERENCES s3_multipart_uploads(upload_id) ON DELETE CASCADE,
|
|
94
|
+
part_number INTEGER NOT NULL,
|
|
95
|
+
body BLOB NOT NULL,
|
|
96
|
+
etag TEXT NOT NULL,
|
|
97
|
+
last_modified TEXT NOT NULL,
|
|
98
|
+
checksum_value TEXT,
|
|
99
|
+
PRIMARY KEY (upload_id, part_number)
|
|
100
|
+
);
|
|
101
101
|
`;
|
|
102
102
|
export class PersistenceManager {
|
|
103
103
|
db;
|
|
@@ -112,41 +112,41 @@ export class PersistenceManager {
|
|
|
112
112
|
}
|
|
113
113
|
prepareStatements() {
|
|
114
114
|
return {
|
|
115
|
-
insertQueue: this.db.prepare(`
|
|
116
|
-
INSERT OR REPLACE INTO sqs_queues (name, url, arn, attributes, tags, created_timestamp, last_modified_timestamp, sequence_counter)
|
|
117
|
-
VALUES (?, ?, ?, ?, ?, ?, ?, ?)
|
|
115
|
+
insertQueue: this.db.prepare(`
|
|
116
|
+
INSERT OR REPLACE INTO sqs_queues (name, url, arn, attributes, tags, created_timestamp, last_modified_timestamp, sequence_counter)
|
|
117
|
+
VALUES (?, ?, ?, ?, ?, ?, ?, ?)
|
|
118
118
|
`),
|
|
119
119
|
deleteQueue: this.db.prepare("DELETE FROM sqs_queues WHERE name = ?"),
|
|
120
120
|
updateQueueAttributes: this.db.prepare("UPDATE sqs_queues SET attributes = ?, last_modified_timestamp = ? WHERE name = ?"),
|
|
121
121
|
updateQueueSequenceCounter: this.db.prepare("UPDATE sqs_queues SET sequence_counter = ? WHERE name = ?"),
|
|
122
122
|
loadQueues: this.db.prepare("SELECT * FROM sqs_queues"),
|
|
123
|
-
insertMessage: this.db.prepare(`
|
|
124
|
-
INSERT OR REPLACE INTO sqs_messages (
|
|
125
|
-
message_id, queue_name, body, md5_of_body, message_attributes, md5_of_message_attributes,
|
|
126
|
-
sent_timestamp, approximate_receive_count, approximate_first_receive_timestamp,
|
|
127
|
-
delay_until, message_group_id, message_deduplication_id, sequence_number,
|
|
128
|
-
receipt_handle, visibility_deadline
|
|
129
|
-
) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
123
|
+
insertMessage: this.db.prepare(`
|
|
124
|
+
INSERT OR REPLACE INTO sqs_messages (
|
|
125
|
+
message_id, queue_name, body, md5_of_body, message_attributes, md5_of_message_attributes,
|
|
126
|
+
sent_timestamp, approximate_receive_count, approximate_first_receive_timestamp,
|
|
127
|
+
delay_until, message_group_id, message_deduplication_id, sequence_number,
|
|
128
|
+
receipt_handle, visibility_deadline
|
|
129
|
+
) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
130
130
|
`),
|
|
131
131
|
deleteMessage: this.db.prepare("DELETE FROM sqs_messages WHERE message_id = ?"),
|
|
132
|
-
updateMessageInflight: this.db.prepare(`
|
|
133
|
-
UPDATE sqs_messages SET receipt_handle = ?, visibility_deadline = ?, approximate_receive_count = ?, approximate_first_receive_timestamp = ?
|
|
134
|
-
WHERE message_id = ?
|
|
132
|
+
updateMessageInflight: this.db.prepare(`
|
|
133
|
+
UPDATE sqs_messages SET receipt_handle = ?, visibility_deadline = ?, approximate_receive_count = ?, approximate_first_receive_timestamp = ?
|
|
134
|
+
WHERE message_id = ?
|
|
135
135
|
`),
|
|
136
136
|
deleteQueueMessages: this.db.prepare("DELETE FROM sqs_messages WHERE queue_name = ?"),
|
|
137
137
|
loadMessages: this.db.prepare("SELECT * FROM sqs_messages WHERE queue_name = ?"),
|
|
138
138
|
deleteAllMessages: this.db.prepare("DELETE FROM sqs_messages"),
|
|
139
|
-
insertTopic: this.db.prepare(`
|
|
140
|
-
INSERT OR REPLACE INTO sns_topics (arn, name, attributes, tags, subscription_arns)
|
|
141
|
-
VALUES (?, ?, ?, ?, ?)
|
|
139
|
+
insertTopic: this.db.prepare(`
|
|
140
|
+
INSERT OR REPLACE INTO sns_topics (arn, name, attributes, tags, subscription_arns)
|
|
141
|
+
VALUES (?, ?, ?, ?, ?)
|
|
142
142
|
`),
|
|
143
143
|
deleteTopic: this.db.prepare("DELETE FROM sns_topics WHERE arn = ?"),
|
|
144
144
|
updateTopicAttributes: this.db.prepare("UPDATE sns_topics SET attributes = ? WHERE arn = ?"),
|
|
145
145
|
updateTopicSubscriptionArns: this.db.prepare("UPDATE sns_topics SET subscription_arns = ? WHERE arn = ?"),
|
|
146
146
|
loadTopics: this.db.prepare("SELECT * FROM sns_topics"),
|
|
147
|
-
insertSubscription: this.db.prepare(`
|
|
148
|
-
INSERT OR REPLACE INTO sns_subscriptions (arn, topic_arn, protocol, endpoint, confirmed, attributes)
|
|
149
|
-
VALUES (?, ?, ?, ?, ?, ?)
|
|
147
|
+
insertSubscription: this.db.prepare(`
|
|
148
|
+
INSERT OR REPLACE INTO sns_subscriptions (arn, topic_arn, protocol, endpoint, confirmed, attributes)
|
|
149
|
+
VALUES (?, ?, ?, ?, ?, ?)
|
|
150
150
|
`),
|
|
151
151
|
deleteSubscription: this.db.prepare("DELETE FROM sns_subscriptions WHERE arn = ?"),
|
|
152
152
|
updateSubscriptionAttributes: this.db.prepare("UPDATE sns_subscriptions SET attributes = ? WHERE arn = ?"),
|
|
@@ -154,28 +154,28 @@ export class PersistenceManager {
|
|
|
154
154
|
insertBucket: this.db.prepare("INSERT OR REPLACE INTO s3_buckets (name, creation_date, type) VALUES (?, ?, ?)"),
|
|
155
155
|
deleteBucket: this.db.prepare("DELETE FROM s3_buckets WHERE name = ?"),
|
|
156
156
|
loadBuckets: this.db.prepare("SELECT * FROM s3_buckets"),
|
|
157
|
-
upsertObject: this.db.prepare(`
|
|
158
|
-
INSERT OR REPLACE INTO s3_objects (
|
|
159
|
-
bucket, key, body, content_type, content_length, etag, last_modified, metadata,
|
|
160
|
-
content_language, content_disposition, cache_control, content_encoding,
|
|
161
|
-
parts, checksum_algorithm, checksum_value, checksum_type, part_checksums
|
|
162
|
-
) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
157
|
+
upsertObject: this.db.prepare(`
|
|
158
|
+
INSERT OR REPLACE INTO s3_objects (
|
|
159
|
+
bucket, key, body, content_type, content_length, etag, last_modified, metadata,
|
|
160
|
+
content_language, content_disposition, cache_control, content_encoding,
|
|
161
|
+
parts, checksum_algorithm, checksum_value, checksum_type, part_checksums
|
|
162
|
+
) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
163
163
|
`),
|
|
164
164
|
deleteObject: this.db.prepare("DELETE FROM s3_objects WHERE bucket = ? AND key = ?"),
|
|
165
165
|
deleteObjectsByBucket: this.db.prepare("DELETE FROM s3_objects WHERE bucket = ?"),
|
|
166
166
|
deleteAllObjects: this.db.prepare("DELETE FROM s3_objects"),
|
|
167
167
|
loadObjects: this.db.prepare("SELECT * FROM s3_objects"),
|
|
168
|
-
insertMultipartUpload: this.db.prepare(`
|
|
169
|
-
INSERT OR REPLACE INTO s3_multipart_uploads (
|
|
170
|
-
upload_id, bucket, key, content_type, metadata, initiated,
|
|
171
|
-
content_language, content_disposition, cache_control, content_encoding, checksum_algorithm
|
|
172
|
-
) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
168
|
+
insertMultipartUpload: this.db.prepare(`
|
|
169
|
+
INSERT OR REPLACE INTO s3_multipart_uploads (
|
|
170
|
+
upload_id, bucket, key, content_type, metadata, initiated,
|
|
171
|
+
content_language, content_disposition, cache_control, content_encoding, checksum_algorithm
|
|
172
|
+
) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
173
173
|
`),
|
|
174
174
|
deleteMultipartUpload: this.db.prepare("DELETE FROM s3_multipart_uploads WHERE upload_id = ?"),
|
|
175
175
|
loadMultipartUploads: this.db.prepare("SELECT * FROM s3_multipart_uploads"),
|
|
176
|
-
upsertMultipartPart: this.db.prepare(`
|
|
177
|
-
INSERT OR REPLACE INTO s3_multipart_parts (upload_id, part_number, body, etag, last_modified, checksum_value)
|
|
178
|
-
VALUES (?, ?, ?, ?, ?, ?)
|
|
176
|
+
upsertMultipartPart: this.db.prepare(`
|
|
177
|
+
INSERT OR REPLACE INTO s3_multipart_parts (upload_id, part_number, body, etag, last_modified, checksum_value)
|
|
178
|
+
VALUES (?, ?, ?, ?, ?, ?)
|
|
179
179
|
`),
|
|
180
180
|
deleteMultipartParts: this.db.prepare("DELETE FROM s3_multipart_parts WHERE upload_id = ?"),
|
|
181
181
|
loadMultipartParts: this.db.prepare("SELECT * FROM s3_multipart_parts WHERE upload_id = ?"),
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import type { FastifyRequest, FastifyReply } from "fastify";
|
|
2
|
+
import type { S3Store } from "../s3Store.ts";
|
|
3
|
+
/**
|
|
4
|
+
* Handles S3 POST Object (presigned POST form data uploads).
|
|
5
|
+
* See: https://docs.aws.amazon.com/AmazonS3/latest/API/RESTObjectPOST.html
|
|
6
|
+
*
|
|
7
|
+
* The client sends multipart/form-data with policy fields and a file.
|
|
8
|
+
* The "file" field MUST be the last field; any fields after it are ignored.
|
|
9
|
+
* The object key is specified in the "key" form field, not in the URL.
|
|
10
|
+
*/
|
|
11
|
+
export declare function postObject(request: FastifyRequest<{
|
|
12
|
+
Params: {
|
|
13
|
+
bucket: string;
|
|
14
|
+
};
|
|
15
|
+
}>, reply: FastifyReply, store: S3Store): void;
|
|
16
|
+
/**
|
|
17
|
+
* Checks if a POST /:bucket request is a PostObject (multipart/form-data)
|
|
18
|
+
* vs DeleteObjects (XML body). This is used by the router to dispatch correctly.
|
|
19
|
+
*/
|
|
20
|
+
export declare function isPostObjectRequest(contentType: string | undefined): boolean;
|
|
21
|
+
//# sourceMappingURL=postObject.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"postObject.d.ts","sourceRoot":"","sources":["../../../src/s3/actions/postObject.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AAE5D,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,eAAe,CAAC;AAE7C;;;;;;;GAOG;AACH,wBAAgB,UAAU,CACxB,OAAO,EAAE,cAAc,CAAC;IAAE,MAAM,EAAE;QAAE,MAAM,EAAE,MAAM,CAAA;KAAE,CAAA;CAAE,CAAC,EACvD,KAAK,EAAE,YAAY,EACnB,KAAK,EAAE,OAAO,GACb,IAAI,CAqGN;AAED;;;GAGG;AACH,wBAAgB,mBAAmB,CAAC,WAAW,EAAE,MAAM,GAAG,SAAS,GAAG,OAAO,CAE5E"}
|
|
@@ -0,0 +1,166 @@
|
|
|
1
|
+
import { S3Error } from "../../common/errors.js";
|
|
2
|
+
/**
|
|
3
|
+
* Handles S3 POST Object (presigned POST form data uploads).
|
|
4
|
+
* See: https://docs.aws.amazon.com/AmazonS3/latest/API/RESTObjectPOST.html
|
|
5
|
+
*
|
|
6
|
+
* The client sends multipart/form-data with policy fields and a file.
|
|
7
|
+
* The "file" field MUST be the last field; any fields after it are ignored.
|
|
8
|
+
* The object key is specified in the "key" form field, not in the URL.
|
|
9
|
+
*/
|
|
10
|
+
export function postObject(request, reply, store) {
|
|
11
|
+
const bucket = request.params.bucket;
|
|
12
|
+
const contentType = request.headers["content-type"] ?? "";
|
|
13
|
+
const boundaryMatch = /boundary=([^\s;]+)/i.exec(contentType);
|
|
14
|
+
if (!boundaryMatch) {
|
|
15
|
+
throw new S3Error("MalformedPOSTRequest", "The body of your POST request is not well-formed multipart/form-data.", 400);
|
|
16
|
+
}
|
|
17
|
+
const boundary = boundaryMatch[1];
|
|
18
|
+
const body = Buffer.isBuffer(request.body) ? request.body : Buffer.from(request.body);
|
|
19
|
+
const { fields, file, filename } = parseMultipart(body, boundary);
|
|
20
|
+
// "key" is a required field
|
|
21
|
+
let key = fields["key"];
|
|
22
|
+
if (!key) {
|
|
23
|
+
throw new S3Error("InvalidArgument", "Bucket POST must contain a field named 'key'.", 400);
|
|
24
|
+
}
|
|
25
|
+
// ${filename} substitution in the key field
|
|
26
|
+
if (key.includes("${filename}")) {
|
|
27
|
+
const resolvedFilename = filename ?? "";
|
|
28
|
+
key = key.replaceAll("${filename}", resolvedFilename);
|
|
29
|
+
}
|
|
30
|
+
// Content-Type for the stored object comes from the form field, default binary/octet-stream
|
|
31
|
+
const objectContentType = fields["Content-Type"] ?? "binary/octet-stream";
|
|
32
|
+
const fileBody = file ?? Buffer.alloc(0);
|
|
33
|
+
// Extract user metadata from x-amz-meta-* form fields
|
|
34
|
+
const metadata = {};
|
|
35
|
+
for (const [fieldName, fieldValue] of Object.entries(fields)) {
|
|
36
|
+
if (fieldName.toLowerCase().startsWith("x-amz-meta-")) {
|
|
37
|
+
const metaKey = fieldName.toLowerCase().slice("x-amz-meta-".length);
|
|
38
|
+
metadata[metaKey] = fieldValue;
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
// System metadata from form fields
|
|
42
|
+
const systemMetadata = {};
|
|
43
|
+
if (fields["Content-Disposition"])
|
|
44
|
+
systemMetadata.contentDisposition = fields["Content-Disposition"];
|
|
45
|
+
if (fields["Cache-Control"])
|
|
46
|
+
systemMetadata.cacheControl = fields["Cache-Control"];
|
|
47
|
+
if (fields["Content-Encoding"])
|
|
48
|
+
systemMetadata.contentEncoding = fields["Content-Encoding"];
|
|
49
|
+
const obj = store.putObject(bucket, key, fileBody, objectContentType, metadata, systemMetadata);
|
|
50
|
+
if (store.spy) {
|
|
51
|
+
store.spy.addMessage({
|
|
52
|
+
service: "s3",
|
|
53
|
+
bucket,
|
|
54
|
+
key,
|
|
55
|
+
status: "uploaded",
|
|
56
|
+
timestamp: Date.now(),
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
// success_action_redirect takes precedence over success_action_status
|
|
60
|
+
const redirect = fields["success_action_redirect"] ?? fields["redirect"];
|
|
61
|
+
if (redirect) {
|
|
62
|
+
try {
|
|
63
|
+
const redirectUrl = new URL(redirect);
|
|
64
|
+
redirectUrl.searchParams.set("bucket", bucket);
|
|
65
|
+
redirectUrl.searchParams.set("key", key);
|
|
66
|
+
redirectUrl.searchParams.set("etag", obj.etag);
|
|
67
|
+
reply.header("location", redirectUrl.toString());
|
|
68
|
+
reply.status(303).send();
|
|
69
|
+
return;
|
|
70
|
+
}
|
|
71
|
+
catch {
|
|
72
|
+
// Invalid redirect URL — fall through to success_action_status
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
// success_action_status: 200, 201, or 204 (default). Invalid values fall back to 204.
|
|
76
|
+
const rawStatus = Number.parseInt(fields["success_action_status"] ?? "204", 10);
|
|
77
|
+
const status = [200, 201, 204].includes(rawStatus) ? rawStatus : 204;
|
|
78
|
+
// ETag header on all success responses
|
|
79
|
+
reply.header("etag", obj.etag);
|
|
80
|
+
if (status === 201) {
|
|
81
|
+
const xml = [
|
|
82
|
+
`<?xml version="1.0" encoding="UTF-8"?>`,
|
|
83
|
+
`<PostResponse>`,
|
|
84
|
+
` <Bucket>${bucket}</Bucket>`,
|
|
85
|
+
` <Key>${key}</Key>`,
|
|
86
|
+
` <ETag>${obj.etag}</ETag>`,
|
|
87
|
+
`</PostResponse>`,
|
|
88
|
+
].join("\n");
|
|
89
|
+
reply.header("content-type", "application/xml");
|
|
90
|
+
reply.status(201).send(xml);
|
|
91
|
+
}
|
|
92
|
+
else {
|
|
93
|
+
reply.status(status).send();
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* Checks if a POST /:bucket request is a PostObject (multipart/form-data)
|
|
98
|
+
* vs DeleteObjects (XML body). This is used by the router to dispatch correctly.
|
|
99
|
+
*/
|
|
100
|
+
export function isPostObjectRequest(contentType) {
|
|
101
|
+
return !!contentType && contentType.toLowerCase().includes("multipart/form-data");
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Parse multipart/form-data from raw buffer.
|
|
105
|
+
*
|
|
106
|
+
* Per the S3 spec, the "file" field must be the last field.
|
|
107
|
+
* Any fields appearing after "file" are ignored.
|
|
108
|
+
*/
|
|
109
|
+
function parseMultipart(body, boundary) {
|
|
110
|
+
const fields = {};
|
|
111
|
+
let file;
|
|
112
|
+
let filename;
|
|
113
|
+
const delimiter = Buffer.from(`--${boundary}`);
|
|
114
|
+
const crlfCrlf = Buffer.from("\r\n\r\n");
|
|
115
|
+
const crlf = Buffer.from("\r\n");
|
|
116
|
+
// Find all boundary positions
|
|
117
|
+
const positions = [];
|
|
118
|
+
let searchFrom = 0;
|
|
119
|
+
while (true) {
|
|
120
|
+
const idx = body.indexOf(delimiter, searchFrom);
|
|
121
|
+
if (idx === -1)
|
|
122
|
+
break;
|
|
123
|
+
positions.push(idx);
|
|
124
|
+
searchFrom = idx + delimiter.length;
|
|
125
|
+
}
|
|
126
|
+
for (let i = 0; i < positions.length - 1; i++) {
|
|
127
|
+
const partStart = positions[i] + delimiter.length;
|
|
128
|
+
const partEnd = positions[i + 1];
|
|
129
|
+
// Skip the \r\n after the boundary delimiter
|
|
130
|
+
const contentStart = body.indexOf(crlf, partStart);
|
|
131
|
+
if (contentStart === -1 || contentStart >= partEnd)
|
|
132
|
+
continue;
|
|
133
|
+
const headersStart = contentStart + crlf.length;
|
|
134
|
+
// Find \r\n\r\n that separates headers from body
|
|
135
|
+
const headerEnd = body.indexOf(crlfCrlf, headersStart);
|
|
136
|
+
if (headerEnd === -1 || headerEnd >= partEnd)
|
|
137
|
+
continue;
|
|
138
|
+
const headerSection = body.subarray(headersStart, headerEnd).toString("utf-8");
|
|
139
|
+
const valueStart = headerEnd + crlfCrlf.length;
|
|
140
|
+
// Value ends at \r\n before the next boundary
|
|
141
|
+
let valueEnd = partEnd;
|
|
142
|
+
if (body[partEnd - 2] === 0x0d && body[partEnd - 1] === 0x0a) {
|
|
143
|
+
valueEnd = partEnd - 2; // strip trailing \r\n before boundary
|
|
144
|
+
}
|
|
145
|
+
// Parse Content-Disposition
|
|
146
|
+
const dispositionMatch = /Content-Disposition:\s*form-data;\s*name="([^"]+)"(?:;\s*filename="([^"]*)")?/i.exec(headerSection);
|
|
147
|
+
if (!dispositionMatch)
|
|
148
|
+
continue;
|
|
149
|
+
const fieldName = dispositionMatch[1];
|
|
150
|
+
const filenamePart = dispositionMatch[2];
|
|
151
|
+
if (filenamePart !== undefined || fieldName === "file") {
|
|
152
|
+
// File field — extract binary data directly from buffer
|
|
153
|
+
file = body.subarray(valueStart, valueEnd);
|
|
154
|
+
// Extract just the filename (strip path components per AWS spec)
|
|
155
|
+
if (filenamePart !== undefined) {
|
|
156
|
+
const lastSlash = Math.max(filenamePart.lastIndexOf("/"), filenamePart.lastIndexOf("\\"));
|
|
157
|
+
filename = lastSlash >= 0 ? filenamePart.substring(lastSlash + 1) : filenamePart;
|
|
158
|
+
}
|
|
159
|
+
// Per AWS spec, file must be the last field — stop processing
|
|
160
|
+
break;
|
|
161
|
+
}
|
|
162
|
+
fields[fieldName] = body.subarray(valueStart, valueEnd).toString("utf-8");
|
|
163
|
+
}
|
|
164
|
+
return { fields, file, filename };
|
|
165
|
+
}
|
|
166
|
+
//# sourceMappingURL=postObject.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"postObject.js","sourceRoot":"","sources":["../../../src/s3/actions/postObject.ts"],"names":[],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"s3Router.d.ts","sourceRoot":"","sources":["../../src/s3/s3Router.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAE/C,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;
|
|
1
|
+
{"version":3,"file":"s3Router.d.ts","sourceRoot":"","sources":["../../src/s3/s3Router.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAE/C,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAmB5C,wBAAgB,gBAAgB,CAAC,GAAG,EAAE,eAAe,EAAE,KAAK,EAAE,OAAO,GAAG,IAAI,CAoL3E"}
|
package/dist/s3/s3Router.js
CHANGED
|
@@ -15,6 +15,7 @@ import { completeMultipartUpload } from "./actions/completeMultipartUpload.js";
|
|
|
15
15
|
import { abortMultipartUpload } from "./actions/abortMultipartUpload.js";
|
|
16
16
|
import { getObjectAttributes } from "./actions/getObjectAttributes.js";
|
|
17
17
|
import { renameObject } from "./actions/renameObject.js";
|
|
18
|
+
import { postObject, isPostObjectRequest } from "./actions/postObject.js";
|
|
18
19
|
export function registerS3Routes(app, store) {
|
|
19
20
|
const handleError = (err, reply, isHead = false) => {
|
|
20
21
|
if (err instanceof S3Error) {
|
|
@@ -88,7 +89,12 @@ export function registerS3Routes(app, store) {
|
|
|
88
89
|
});
|
|
89
90
|
app.post("/:bucket", async (request, reply) => {
|
|
90
91
|
try {
|
|
91
|
-
|
|
92
|
+
if (isPostObjectRequest(request.headers["content-type"])) {
|
|
93
|
+
postObject(request, reply, store);
|
|
94
|
+
}
|
|
95
|
+
else {
|
|
96
|
+
deleteObjects(request, reply, store);
|
|
97
|
+
}
|
|
92
98
|
}
|
|
93
99
|
catch (err) {
|
|
94
100
|
handleError(err, reply);
|
|
@@ -175,7 +181,12 @@ export function registerS3Routes(app, store) {
|
|
|
175
181
|
try {
|
|
176
182
|
const key = getKey(request.params);
|
|
177
183
|
if (!key) {
|
|
178
|
-
|
|
184
|
+
if (isPostObjectRequest(request.headers["content-type"])) {
|
|
185
|
+
postObject(request, reply, store);
|
|
186
|
+
}
|
|
187
|
+
else {
|
|
188
|
+
deleteObjects(request, reply, store);
|
|
189
|
+
}
|
|
179
190
|
return;
|
|
180
191
|
}
|
|
181
192
|
const query = getQuery(request);
|
package/dist/s3/s3Router.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"s3Router.js","sourceRoot":"","sources":["../../src/s3/s3Router.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,OAAO,EAAE,MAAM,qBAAqB,CAAC;AAE9C,OAAO,EAAE,YAAY,EAAE,MAAM,2BAA2B,CAAC;AACzD,OAAO,EAAE,UAAU,EAAE,MAAM,yBAAyB,CAAC;AACrD,OAAO,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AACvD,OAAO,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AACvD,OAAO,EAAE,YAAY,EAAE,MAAM,2BAA2B,CAAC;AACzD,OAAO,EAAE,aAAa,EAAE,MAAM,4BAA4B,CAAC;AAC3D,OAAO,EAAE,SAAS,EAAE,MAAM,wBAAwB,CAAC;AACnD,OAAO,EAAE,SAAS,EAAE,MAAM,wBAAwB,CAAC;AACnD,OAAO,EAAE,YAAY,EAAE,MAAM,2BAA2B,CAAC;AACzD,OAAO,EAAE,UAAU,EAAE,MAAM,yBAAyB,CAAC;AACrD,OAAO,EAAE,qBAAqB,EAAE,MAAM,oCAAoC,CAAC;AAC3E,OAAO,EAAE,UAAU,EAAE,MAAM,yBAAyB,CAAC;AACrD,OAAO,EAAE,uBAAuB,EAAE,MAAM,sCAAsC,CAAC;AAC/E,OAAO,EAAE,oBAAoB,EAAE,MAAM,mCAAmC,CAAC;AACzE,OAAO,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AACvE,OAAO,EAAE,YAAY,EAAE,MAAM,2BAA2B,CAAC;
|
|
1
|
+
{"version":3,"file":"s3Router.js","sourceRoot":"","sources":["../../src/s3/s3Router.ts"],"names":[],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "fauxqs",
|
|
3
|
-
"version": "2.
|
|
3
|
+
"version": "2.1.0",
|
|
4
4
|
"description": "SNS/SQS/S3 emulator",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"scripts": {
|
|
@@ -43,6 +43,7 @@
|
|
|
43
43
|
"valibot": "^1.2.0"
|
|
44
44
|
},
|
|
45
45
|
"devDependencies": {
|
|
46
|
+
"@aws-sdk/s3-presigned-post": "^3.997.0",
|
|
46
47
|
"@aws-sdk/s3-request-presigner": "^3.993.0",
|
|
47
48
|
"@message-queue-toolkit/s3-payload-store": "^3.0.0",
|
|
48
49
|
"@types/node": "^25.2.3",
|