@sot1986/appsync-precognition 0.0.2 → 0.0.4
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/dist/index.d.ts +13 -28
- package/dist/index.js +204 -64
- package/package.json +1 -1
package/dist/index.d.ts
CHANGED
|
@@ -1,45 +1,30 @@
|
|
|
1
1
|
import { Context } from "@aws-appsync/utils";
|
|
2
2
|
|
|
3
|
-
//#region rolldown:runtime
|
|
4
|
-
//#endregion
|
|
5
3
|
//#region src/types.d.ts
|
|
6
4
|
interface Rule<T = unknown> {
|
|
7
5
|
check: boolean;
|
|
8
6
|
message: string;
|
|
9
7
|
value: T;
|
|
10
8
|
}
|
|
11
|
-
type
|
|
9
|
+
type FullRule = "required" | "nullable" | "sometimes" | "number" | "boolean" | "string" | "array" | "object" | "date" | ["min", number] | ["max", number] | ["between", number, number] | "email" | "url" | "uuid" | "ulid" | ["regex", string] | ["in", ...(string | number | boolean | null)[]] | ["notIn", ...(string | number | boolean | null)[]] | ["after", string] | ["before", string] | ["afterOrEqual", string] | ["beforeOrEqual", string];
|
|
12
10
|
type ArrayKeys<T extends unknown[]> = T extends [unknown, ...unknown[]] ? T extends Record<infer Index, unknown> ? Index extends `${number}` ? Index : never : never : `${number}`;
|
|
13
11
|
type ObjectKeys<T extends object> = T extends unknown[] ? ArrayKeys<T> : keyof T & string;
|
|
14
12
|
interface HasConstructor {
|
|
15
13
|
new (...args: unknown[]): unknown;
|
|
16
14
|
}
|
|
17
15
|
type NestedKeyOf<T> = T extends Record<infer Key, unknown> ? T extends HasConstructor ? never : T extends CallableFunction ? never : Key extends string | number ? (ObjectKeys<T> | (T[Key] extends object ? `${ObjectKeys<Pick<T, Key>>}.${NestedKeyOf<T[Key]>}` : T extends unknown[] ? T extends [unknown, ...unknown[]] ? never : T[number] extends object ? `${number}.${NestedKeyOf<T[number]>}` : never : never)) : never : never;
|
|
18
|
-
declare namespace rules_d_exports {
|
|
19
|
-
export { names, parse, requiredRule };
|
|
20
|
-
}
|
|
21
|
-
declare const names: {
|
|
22
|
-
readonly required: "required";
|
|
23
|
-
readonly nullable: "nullable";
|
|
24
|
-
readonly sometimes: "sometimes";
|
|
25
|
-
readonly min: "min";
|
|
26
|
-
readonly max: "max";
|
|
27
|
-
readonly between: "between";
|
|
28
|
-
readonly email: "email";
|
|
29
|
-
readonly url: "url";
|
|
30
|
-
readonly uuid: "uuid";
|
|
31
|
-
readonly regex: "regex";
|
|
32
|
-
readonly in: "in";
|
|
33
|
-
readonly notIn: "notIn";
|
|
34
|
-
readonly before: "before";
|
|
35
|
-
readonly after: "after";
|
|
36
|
-
};
|
|
37
|
-
declare function parse<T>(value: T, rule: ShortRule<keyof typeof names>): Rule<T>;
|
|
38
|
-
declare function requiredRule<T>(value: T, ..._params: string[]): Rule<T>;
|
|
39
16
|
//#endregion
|
|
40
17
|
//#region src/index.d.ts
|
|
41
|
-
declare function validate<T extends object>(obj: T, checks: Partial<Record<NestedKeyOf<T>, (
|
|
42
|
-
|
|
18
|
+
declare function validate<T extends object>(obj: T, checks: Partial<Record<NestedKeyOf<T>, (FullRule | Rule)[]>>, options?: {
|
|
19
|
+
trim?: boolean;
|
|
20
|
+
allowEmptyString?: boolean;
|
|
21
|
+
}): T;
|
|
22
|
+
declare function precognitiveValidation<T extends object>(ctx: Context<T>, checks: Partial<Record<NestedKeyOf<T>, (FullRule | Rule)[]>>, options?: {
|
|
23
|
+
trim?: boolean;
|
|
24
|
+
allowEmptyString?: boolean;
|
|
25
|
+
skipTo?: "END" | "NEXT";
|
|
26
|
+
}): T;
|
|
27
|
+
declare function formatAttributeName(path: string): string;
|
|
43
28
|
//#endregion
|
|
44
|
-
export {
|
|
45
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
29
|
+
export { formatAttributeName, precognitiveValidation, validate };
|
|
30
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguZC50cyIsIm5hbWVzIjpbXSwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiLCIuLi9zcmMvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOltdLCJtYXBwaW5ncyI6Ijs7O1VBRWlCOztFQUFqQixPQUFpQixFQUFBLE1BQUE7RUFNakIsS0FBWSxFQUhILENBR0c7QUF1Qkw7QUFHSCxLQTFCUSxRQUFBLEdBMEJSLFVBQUEsR0FBQSxVQUFBLEdBQUEsV0FBQSxHQUFBLFFBQUEsR0FBQSxTQUFBLEdBQUEsUUFBQSxHQUFBLE9BQUEsR0FBQSxRQUFBLEdBQUEsTUFBQSxHQUFBLENBQUEsS0FBQSxFQUFBLE1BQUEsQ0FBQSxHQUFBLENBQUEsS0FBQSxFQUFBLE1BQUEsQ0FBQSxHQUFBLENBQUEsU0FBQSxFQUFBLE1BQUEsRUFBQSxNQUFBLENBQUEsR0FBQSxPQUFBLEdBQUEsS0FBQSxHQUFBLE1BQUEsR0FBQSxNQUFBLEdBQUEsQ0FBQSxPQUFBLEVBQUEsTUFBQSxDQUFBLEdBQUEsQ0FBQSxJQUFBLEVBQUEsR0FBQSxDQUFBLE1BQUEsR0FBQSxNQUFBLEdBQUEsT0FBQSxHQUFBLElBQUEsQ0FBQSxFQUFBLENBQUEsR0FBQSxDQUFBLE9BQUEsRUFBQSxHQUFBLENBQUEsTUFBQSxHQUFBLE1BQUEsR0FBQSxPQUFBLEdBQUEsSUFBQSxDQUFBLEVBQUEsQ0FBQSxHQUFBLENBQUEsT0FBQSxFQUFBLE1BQUEsQ0FBQSxHQUFBLENBQUEsUUFBQSxFQUFBLE1BQUEsQ0FBQSxHQUFBLENBQUEsY0FBQSxFQUFBLE1BQUEsQ0FBQSxHQUFBLENBQUEsZUFBQSxFQUFBLE1BQUEsQ0FBQTtLQURDLFNBRUMsQ0FBQSxVQUFBLE9BQUEsRUFBQSxDQUFBLEdBREYsQ0FDRSxTQUFBLENBQUEsT0FBQSxFQUFBLEdBQUEsT0FBQSxFQUFBLENBQUEsR0FBQSxDQUFBLFNBQVUsTUFBVixDQUFBLEtBQUEsTUFBQSxFQUFBLE9BQUEsQ0FBQSxHQUFBLEtBQUEsU0FBQSxHQUFBLE1BQUEsRUFBQSxHQUFBLEtBQUEsR0FBQSxLQUFBLEdBQUEsS0FBQSxHQUFBLEdBQUEsTUFBQSxFQUFBO0tBT0QsVUFQVyxDQUFBLFVBQUEsTUFBQSxDQUFBLEdBUVosQ0FSWSxTQUFBLE9BQUEsRUFBQSxHQVNWLFNBVFUsQ0FTQSxDQVRBLENBQUEsR0FBQSxNQVVKLENBVkksR0FBQSxNQUFBO1VBWU4sY0FBQSxDQVpNO0VBQUEsS0FPWCxHQUFBLElBQUEsRUFBQSxPQUFBLEVBQUEsQ0FBQSxFQUFBLE9BQUE7O0FBRVcsS0FPSixXQVBJLENBQUEsQ0FBQSxDQUFBLEdBT2EsQ0FQYixTQU91QixNQVB2QixDQUFBLEtBQUEsSUFBQSxFQUFBLE9BQUEsQ0FBQSxHQVFaLENBUlksU0FRRixjQVJFLEdBQUEsS0FBQSxHQVVWLENBVlUsU0FVQSxnQkFWQSxHQUFBLEtBQUEsR0FBQSxHQUFBLFNBQUEsTUFBQSxHQUFBLE1BQUEsR0FBQSxDQWFMLFVBYkssQ0FhTSxDQWJOLENBQUEsR0FBQSxDQWFZLENBYlosQ0FhYyxHQWJkLENBQUEsU0FBQSxNQUFBLEdBQUEsR0FjQyxVQWRELENBY1ksSUFkWixDQWNpQixDQWRqQixFQWNvQixHQWRwQixDQUFBLENBQUEsSUFjNkIsV0FkN0IsQ0FjeUMsQ0FkekMsQ0FjMkMsR0FkM0MsQ0FBQSxDQUFBLEVBQUEsR0FlRixDQWZFLFNBQUEsT0FBQSxFQUFBLEdBZ0JBLENBaEJBLFNBQUEsQ0FBQSxPQUFBLEVBQUEsR0FBQSxPQUFBLEVBQUEsQ0FBQSxHQUFBLEtBQUEsR0FrQkUsQ0FsQkYsQ0FBQSxNQUFBLENBQUEsU0FBQSxNQUFBLEdBQUEsR0FBQSxNQUFBLElBbUJpQixXQW5CakIsQ0FtQjZCLENBbkI3QixDQUFBLE1BQUEsQ0FBQSxDQUFBLEVBQUEsR0FBQSxLQUFBLEdBQUEsS0FBQSxDQUFBLENBQUEsR0FBQSxLQUFBLEdBQUEsS0FBQTs7O2lCQ3RDQSxnQ0FDVCxXQUNHLFFBQVEsT0FBTyxZQUFZLEtBQUssV0FBVyxrQkR1QjlDO0VBN0JQLElBQWlCLENBQUEsRUFBQSxPQUFBO0VBTWpCLGdCQUFZLENBQUEsRUFBQSxPQUFBO0FBdUJMLENBQUEsQ0FBQSxFQ2xCSixDRG9CRTtBQUNELGlCQzhCWSxzQkQ5QlosQ0FBQSxVQUFBLE1BQUEsQ0FBQSxDQUFBLEdBQUEsRUMrQkcsT0QvQkgsQ0MrQlcsQ0QvQlgsQ0FBQSxFQUFBLE1BQUEsRUNnQ00sT0RoQ04sQ0NnQ2MsTURoQ2QsQ0NnQ3FCLFdEaENyQixDQ2dDaUMsQ0RoQ2pDLENBQUEsRUFBQSxDQ2dDc0MsUURoQ3RDLEdDZ0NpRCxJRGhDakQsQ0FBQSxFQUFBLENBQUEsQ0FBQSxFQUFBLE9BQ1ksQ0FEWixFQUFBO0VBQ0UsSUFBQSxDQUFBLEVBQUEsT0FBQTtFQUFVLGdCQUFBLENBQUEsRUFBQSxPQUFBO0VBQUEsTUFBQSxDQUFBLEVBQUEsS0FBQSxHQUFBLE1BQUE7QUFBQSxDQUFBLENBQUEsRUNxQ2IsQ0Q5QkU7QUFDRCxpQkNxRFksbUJBQUEsQ0RyRFosSUFBQSxFQUFBLE1BQUEsQ0FBQSxFQUFBLE1BQUEifQ==
|
package/dist/index.js
CHANGED
|
@@ -28,8 +28,10 @@ function precognitiveKeys(ctx) {
|
|
|
28
28
|
const keys = getHeader("Precognition-Validate-Only", ctx);
|
|
29
29
|
return keys ? keys.split(",").map((key) => key.trim()) : null;
|
|
30
30
|
}
|
|
31
|
-
function cleanString(value) {
|
|
31
|
+
function cleanString(value, options) {
|
|
32
|
+
if (options?.trim === false) return value;
|
|
32
33
|
let parsed = value.trim();
|
|
34
|
+
if (options?.allowEmptyString) return parsed;
|
|
33
35
|
if (parsed === "") parsed = null;
|
|
34
36
|
return parsed;
|
|
35
37
|
}
|
|
@@ -37,51 +39,61 @@ function cleanString(value) {
|
|
|
37
39
|
//#endregion
|
|
38
40
|
//#region src/rules.ts
|
|
39
41
|
function parse(value, rule) {
|
|
40
|
-
const [
|
|
41
|
-
switch (
|
|
42
|
+
const [name, ...params] = typeof rule === "string" ? [rule, void 0] : [rule[0], ...rule.slice(1)];
|
|
43
|
+
switch (name) {
|
|
42
44
|
case "required": return requiredRule(value);
|
|
43
45
|
case "nullable": return nullableRule(value);
|
|
44
46
|
case "sometimes": return sometimesRule(value);
|
|
45
|
-
case "min": return minRule(value,
|
|
46
|
-
case "max": return maxRule(value,
|
|
47
|
-
case "between": return betweenRule(value,
|
|
47
|
+
case "min": return minRule(value, params[0]);
|
|
48
|
+
case "max": return maxRule(value, params[0]);
|
|
49
|
+
case "between": return betweenRule(value, params[0], params[1]);
|
|
48
50
|
case "email": return emailRule(value);
|
|
49
51
|
case "url": return urlRule(value);
|
|
50
52
|
case "uuid": return uuidRule(value);
|
|
51
|
-
case "
|
|
52
|
-
case "
|
|
53
|
+
case "ulid": return ulidRule(value);
|
|
54
|
+
case "regex": return regexRule(value, params[0]);
|
|
55
|
+
case "in": return inRule(value, ...params);
|
|
56
|
+
case "notIn": return notInRule(value, ...params);
|
|
57
|
+
case "array": return arrayRule(value);
|
|
58
|
+
case "object": return objectRule(value);
|
|
59
|
+
case "boolean": return booleanRule(value);
|
|
60
|
+
case "number": return numberRule(value);
|
|
61
|
+
case "string": return stringRule(value);
|
|
62
|
+
case "date": return dateRule(value);
|
|
63
|
+
case "before": return beforeRule(value, params[0]);
|
|
64
|
+
case "after": return afterRule(value, params[0]);
|
|
65
|
+
case "beforeOrEqual": return beforeOrEqualRule(value, params[0]);
|
|
66
|
+
case "afterOrEqual": return afterOrEqualRule(value, params[0]);
|
|
53
67
|
default: return {
|
|
54
68
|
check: false,
|
|
55
|
-
message: `Unknown rule ${
|
|
69
|
+
message: `Unknown rule ${name}`,
|
|
56
70
|
value
|
|
57
71
|
};
|
|
58
72
|
}
|
|
59
73
|
}
|
|
60
|
-
function minRule(value,
|
|
61
|
-
const minValue = Number(params[0] ?? "0");
|
|
74
|
+
function minRule(value, minValue) {
|
|
62
75
|
const result = {
|
|
63
76
|
check: false,
|
|
64
77
|
message: `:attribute must be greater than or equal to ${minValue}`,
|
|
65
78
|
value
|
|
66
79
|
};
|
|
67
80
|
if (typeof value === "number") result.check = value >= minValue;
|
|
68
|
-
if (typeof
|
|
81
|
+
if (typeof value === "string") result.check = value.length >= minValue;
|
|
69
82
|
if (isArray(value)) {
|
|
70
83
|
result.check = value.length >= minValue;
|
|
71
84
|
result.message = `Array must contain at least ${minValue} elements`;
|
|
72
85
|
}
|
|
73
86
|
return result;
|
|
74
87
|
}
|
|
75
|
-
function maxRule(value,
|
|
76
|
-
const maxValue = Number(params[0] ?? "0");
|
|
88
|
+
function maxRule(value, maxValue) {
|
|
77
89
|
const result = {
|
|
78
90
|
check: false,
|
|
79
91
|
message: `:attribute must be less than or equal to ${maxValue}`,
|
|
80
92
|
value
|
|
81
93
|
};
|
|
82
94
|
if (typeof value === "number") result.check = value <= maxValue;
|
|
83
|
-
if (typeof
|
|
84
|
-
result.check =
|
|
95
|
+
if (typeof value === "string") {
|
|
96
|
+
result.check = value.length <= maxValue;
|
|
85
97
|
result.message = `String must contain at most ${maxValue} characters`;
|
|
86
98
|
}
|
|
87
99
|
if (isArray(value)) {
|
|
@@ -90,17 +102,15 @@ function maxRule(value, ...params) {
|
|
|
90
102
|
}
|
|
91
103
|
return result;
|
|
92
104
|
}
|
|
93
|
-
function betweenRule(value,
|
|
94
|
-
const minValue = Number(params[0] ?? "0");
|
|
95
|
-
const maxValue = Number(params[1] ?? "0");
|
|
105
|
+
function betweenRule(value, minValue, maxValue) {
|
|
96
106
|
const result = {
|
|
97
107
|
check: false,
|
|
98
108
|
message: `:attribute must be between ${minValue} and ${maxValue}`,
|
|
99
109
|
value
|
|
100
110
|
};
|
|
101
111
|
if (typeof value === "number") result.check = value >= minValue && value <= maxValue;
|
|
102
|
-
if (typeof
|
|
103
|
-
result.check =
|
|
112
|
+
if (typeof value === "string") {
|
|
113
|
+
result.check = value.length >= minValue && value.length <= maxValue;
|
|
104
114
|
result.message = `String must contain between ${minValue} and ${maxValue} characters`;
|
|
105
115
|
}
|
|
106
116
|
if (isArray(value)) {
|
|
@@ -115,7 +125,7 @@ function emailRule(value) {
|
|
|
115
125
|
message: ":attribute must be a valid email address",
|
|
116
126
|
value
|
|
117
127
|
};
|
|
118
|
-
if (typeof value === "string") result.check = util.matches("^[
|
|
128
|
+
if (typeof value === "string") result.check = util.matches("^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\\.[a-zA-Z]{2,}$", result.value);
|
|
119
129
|
return result;
|
|
120
130
|
}
|
|
121
131
|
function urlRule(value) {
|
|
@@ -124,7 +134,7 @@ function urlRule(value) {
|
|
|
124
134
|
message: ":attribute must be a valid URL",
|
|
125
135
|
value
|
|
126
136
|
};
|
|
127
|
-
if (typeof value === "string") result.check = util.matches("^(
|
|
137
|
+
if (typeof value === "string") result.check = util.matches("^https?:\\/\\/(www\\.)?[-a-zA-Z0-9@:%._\\+~#=]{1,256}\\.[a-zA-Z0-9()]{1,6}\\b([-a-zA-Z0-9()@:%_\\+.~#?&//=]*)$|^https?:\\/\\/(localhost|\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}\\.\\d{1,3})(:\\d+)?(\\/.*)?$", result.value);
|
|
128
138
|
return result;
|
|
129
139
|
}
|
|
130
140
|
function uuidRule(value) {
|
|
@@ -133,70 +143,189 @@ function uuidRule(value) {
|
|
|
133
143
|
message: ":attribute must be a valid UUID",
|
|
134
144
|
value
|
|
135
145
|
};
|
|
136
|
-
if (typeof
|
|
146
|
+
if (typeof value === "string") result.check = util.matches("^[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$", value);
|
|
147
|
+
return result;
|
|
148
|
+
}
|
|
149
|
+
function ulidRule(value) {
|
|
150
|
+
const result = {
|
|
151
|
+
check: false,
|
|
152
|
+
message: ":attribute must be a valid ULID",
|
|
153
|
+
value
|
|
154
|
+
};
|
|
155
|
+
if (typeof value === "string") result.check = util.matches("^[0123456789ABCDEFGHJKMNPQRSTVWXYZ]{26}$", value);
|
|
137
156
|
return result;
|
|
138
157
|
}
|
|
139
|
-
function regexRule(value,
|
|
140
|
-
const regex = params[0] ?? "";
|
|
158
|
+
function regexRule(value, pattern) {
|
|
141
159
|
const result = {
|
|
142
160
|
check: false,
|
|
143
161
|
message: ":attribute must match the specified regular expression",
|
|
144
162
|
value
|
|
145
163
|
};
|
|
146
|
-
if (typeof
|
|
147
|
-
result.value =
|
|
148
|
-
result.check = util.matches(
|
|
164
|
+
if (typeof value === "string") {
|
|
165
|
+
result.value = value.trim();
|
|
166
|
+
result.check = util.matches(pattern, result.value);
|
|
149
167
|
}
|
|
150
168
|
return result;
|
|
151
169
|
}
|
|
152
170
|
function inRule(value, ...params) {
|
|
153
|
-
|
|
154
|
-
check:
|
|
171
|
+
return {
|
|
172
|
+
check: params.includes(value),
|
|
155
173
|
message: ":attribute must be one of the specified values",
|
|
156
174
|
value
|
|
157
175
|
};
|
|
158
|
-
if (typeof result.value === "string") result.check = params.includes(result.value);
|
|
159
|
-
if (typeof value === "number") result.check = params.map(Number).includes(value);
|
|
160
|
-
return result;
|
|
161
176
|
}
|
|
162
|
-
function
|
|
177
|
+
function notInRule(value, ...params) {
|
|
178
|
+
return {
|
|
179
|
+
check: !params.includes(value),
|
|
180
|
+
message: ":attribute must not be one of the specified values",
|
|
181
|
+
value
|
|
182
|
+
};
|
|
183
|
+
}
|
|
184
|
+
function requiredRule(value) {
|
|
163
185
|
const result = {
|
|
164
186
|
check: true,
|
|
165
187
|
message: ":attribute is required",
|
|
166
188
|
value
|
|
167
189
|
};
|
|
168
|
-
if (typeof
|
|
169
|
-
if (isArray(
|
|
170
|
-
if (typeof
|
|
171
|
-
if (
|
|
190
|
+
if (typeof value === "string") result.check = value.length > 0;
|
|
191
|
+
if (isArray(value)) result.check = value.length > 0;
|
|
192
|
+
if (typeof value === "number") result.check = true;
|
|
193
|
+
if (typeof value === "boolean") result.check = true;
|
|
194
|
+
if (typeof value === "object" && !result.value) {
|
|
195
|
+
result.message = ":attribute is not nullable";
|
|
196
|
+
result.check = false;
|
|
197
|
+
}
|
|
198
|
+
if (typeof value === "undefined") result.check = false;
|
|
172
199
|
return result;
|
|
173
200
|
}
|
|
174
|
-
function nullableRule(value
|
|
201
|
+
function nullableRule(value) {
|
|
175
202
|
return {
|
|
176
203
|
check: true,
|
|
177
204
|
message: "",
|
|
178
205
|
value
|
|
179
206
|
};
|
|
180
207
|
}
|
|
181
|
-
function sometimesRule(value
|
|
208
|
+
function sometimesRule(value) {
|
|
182
209
|
const result = {
|
|
183
210
|
check: true,
|
|
184
211
|
message: "",
|
|
185
212
|
value
|
|
186
213
|
};
|
|
187
|
-
if (typeof
|
|
188
|
-
|
|
214
|
+
if (typeof value === "undefined") return result;
|
|
215
|
+
if (typeof value === "object" && !result.value) {
|
|
216
|
+
result.message = ":attribute is not nullable";
|
|
217
|
+
result.check = false;
|
|
218
|
+
}
|
|
219
|
+
return requiredRule(value);
|
|
220
|
+
}
|
|
221
|
+
function arrayRule(value) {
|
|
222
|
+
const result = {
|
|
223
|
+
check: false,
|
|
224
|
+
message: ":attribute must be an array",
|
|
225
|
+
value
|
|
226
|
+
};
|
|
227
|
+
if (isArray(value)) result.check = true;
|
|
228
|
+
return result;
|
|
229
|
+
}
|
|
230
|
+
function objectRule(value) {
|
|
231
|
+
const result = {
|
|
232
|
+
check: false,
|
|
233
|
+
message: ":attribute must be an object",
|
|
234
|
+
value
|
|
235
|
+
};
|
|
236
|
+
if (typeof value === "object" && !isArray(result.value)) result.check = true;
|
|
237
|
+
return result;
|
|
238
|
+
}
|
|
239
|
+
function booleanRule(value) {
|
|
240
|
+
const result = {
|
|
241
|
+
check: false,
|
|
242
|
+
message: ":attribute must be a boolean",
|
|
243
|
+
value
|
|
244
|
+
};
|
|
245
|
+
if (typeof value === "boolean") result.check = true;
|
|
246
|
+
return result;
|
|
247
|
+
}
|
|
248
|
+
function numberRule(value) {
|
|
249
|
+
const result = {
|
|
250
|
+
check: false,
|
|
251
|
+
message: ":attribute must be a number",
|
|
252
|
+
value
|
|
253
|
+
};
|
|
254
|
+
if (typeof value === "number") result.check = true;
|
|
255
|
+
return result;
|
|
256
|
+
}
|
|
257
|
+
function stringRule(value) {
|
|
258
|
+
const result = {
|
|
259
|
+
check: false,
|
|
260
|
+
message: ":attribute must be a string",
|
|
261
|
+
value
|
|
262
|
+
};
|
|
263
|
+
if (typeof value === "string") result.check = true;
|
|
264
|
+
return result;
|
|
265
|
+
}
|
|
266
|
+
function dateRule(value) {
|
|
267
|
+
const result = {
|
|
268
|
+
check: false,
|
|
269
|
+
message: ":attribute must be a date",
|
|
270
|
+
value
|
|
271
|
+
};
|
|
272
|
+
if (typeof value === "string") result.check = util.matches("^\\d{4}-(0[1-9]|1[0-2])-(0[1-9]|[12]\\d|3[01])T([01]\\d|2[0-3]):[0-5]\\d:[0-5]\\d(\\.\\d{1,6})?Z$", result.value);
|
|
273
|
+
if (typeof value === "number") result.check = true;
|
|
274
|
+
return result;
|
|
275
|
+
}
|
|
276
|
+
function beforeRule(value, start) {
|
|
277
|
+
const result = {
|
|
278
|
+
check: false,
|
|
279
|
+
message: `:attribute must be before ${start}`,
|
|
280
|
+
value
|
|
281
|
+
};
|
|
282
|
+
const startValue = util.time.parseISO8601ToEpochMilliSeconds(start);
|
|
283
|
+
if (typeof value === "string") result.check = util.time.parseISO8601ToEpochMilliSeconds(value) < startValue;
|
|
284
|
+
if (typeof value === "number") result.check = value < startValue;
|
|
285
|
+
return result;
|
|
286
|
+
}
|
|
287
|
+
function afterRule(value, start) {
|
|
288
|
+
const result = {
|
|
289
|
+
check: false,
|
|
290
|
+
message: `:attribute must be after ${start}`,
|
|
291
|
+
value
|
|
292
|
+
};
|
|
293
|
+
const startValue = util.time.parseISO8601ToEpochMilliSeconds(start);
|
|
294
|
+
if (typeof value === "string") result.check = util.time.parseISO8601ToEpochMilliSeconds(value) > startValue;
|
|
295
|
+
if (typeof value === "number") result.check = value > startValue;
|
|
296
|
+
return result;
|
|
297
|
+
}
|
|
298
|
+
function beforeOrEqualRule(value, start) {
|
|
299
|
+
const result = {
|
|
300
|
+
check: false,
|
|
301
|
+
message: `:attribute must be before or equal to ${start}`,
|
|
302
|
+
value
|
|
303
|
+
};
|
|
304
|
+
const startValue = util.time.parseISO8601ToEpochMilliSeconds(start);
|
|
305
|
+
if (typeof value === "string") result.check = util.time.parseISO8601ToEpochMilliSeconds(value) <= startValue;
|
|
306
|
+
if (typeof value === "number") result.check = value <= startValue;
|
|
307
|
+
return result;
|
|
308
|
+
}
|
|
309
|
+
function afterOrEqualRule(value, start) {
|
|
310
|
+
const result = {
|
|
311
|
+
check: false,
|
|
312
|
+
message: `:attribute must be after or equal to ${start}`,
|
|
313
|
+
value
|
|
314
|
+
};
|
|
315
|
+
const startValue = util.time.parseISO8601ToEpochMilliSeconds(start);
|
|
316
|
+
if (typeof value === "string") result.check = util.time.parseISO8601ToEpochMilliSeconds(value) >= startValue;
|
|
317
|
+
if (typeof value === "number") result.check = value >= startValue;
|
|
318
|
+
return result;
|
|
189
319
|
}
|
|
190
320
|
|
|
191
321
|
//#endregion
|
|
192
322
|
//#region src/index.ts
|
|
193
|
-
function validate(obj, checks) {
|
|
194
|
-
let
|
|
195
|
-
const errorMessages = [];
|
|
323
|
+
function validate(obj, checks, options) {
|
|
324
|
+
let error = {};
|
|
196
325
|
Object.keys(checks).forEach((path) => {
|
|
197
326
|
let value = getNestedValue(obj, path);
|
|
198
327
|
if (typeof value === "string") {
|
|
199
|
-
value = cleanString(value);
|
|
328
|
+
value = cleanString(value, options);
|
|
200
329
|
setNestedValue(obj, path, value);
|
|
201
330
|
}
|
|
202
331
|
let skip = false;
|
|
@@ -204,26 +333,31 @@ function validate(obj, checks) {
|
|
|
204
333
|
if (skip) return;
|
|
205
334
|
if (rule === "nullable" && value === null) skip = true;
|
|
206
335
|
if (rule === "sometimes" && typeof value === "undefined") skip = true;
|
|
207
|
-
const result = typeof rule === "string" ? parse(value, rule) : { ...rule };
|
|
336
|
+
const result = typeof rule === "string" || isArray(rule) ? parse(value, rule) : { ...rule };
|
|
208
337
|
if (result.check) return;
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
338
|
+
if (error.msg) util.appendError(error.msg, error.errorType, error.data, error.errorInfo);
|
|
339
|
+
result.message = result.message.replace(":attribute", formatAttributeName(path));
|
|
340
|
+
error = {
|
|
341
|
+
msg: result.message,
|
|
342
|
+
errorType: "ValidationError",
|
|
343
|
+
data: null,
|
|
344
|
+
errorInfo: {
|
|
345
|
+
path,
|
|
346
|
+
value
|
|
347
|
+
}
|
|
348
|
+
};
|
|
349
|
+
skip = true;
|
|
216
350
|
});
|
|
217
351
|
});
|
|
218
|
-
if (
|
|
219
|
-
|
|
352
|
+
if (!error.msg) return obj;
|
|
353
|
+
util.error(error.msg, error.errorType, error.data, error.errorInfo);
|
|
220
354
|
}
|
|
221
|
-
function
|
|
222
|
-
if (!isPrecognitiveRequest(ctx)) return validate(ctx.args, checks);
|
|
355
|
+
function precognitiveValidation(ctx, checks, options) {
|
|
356
|
+
if (!isPrecognitiveRequest(ctx)) return validate(ctx.args, checks, options);
|
|
223
357
|
const validationKeys = precognitiveKeys(ctx);
|
|
224
358
|
util.http.addResponseHeader("Precognition", "true");
|
|
225
359
|
if (!validationKeys) {
|
|
226
|
-
validate(ctx.args, checks);
|
|
360
|
+
validate(ctx.args, checks, options);
|
|
227
361
|
util.http.addResponseHeader("Precognition-Success", "true");
|
|
228
362
|
runtime.earlyReturn(null);
|
|
229
363
|
}
|
|
@@ -232,11 +366,17 @@ function precognition(ctx, checks) {
|
|
|
232
366
|
validationKeys.forEach((key) => {
|
|
233
367
|
precognitionChecks[key] = checks[key];
|
|
234
368
|
});
|
|
235
|
-
validate(ctx.args, precognitionChecks);
|
|
369
|
+
validate(ctx.args, precognitionChecks, options);
|
|
236
370
|
util.http.addResponseHeader("Precognition-Success", "true");
|
|
237
|
-
runtime.earlyReturn(null);
|
|
371
|
+
runtime.earlyReturn(null, { skipTo: options?.skipTo ?? "END" });
|
|
372
|
+
}
|
|
373
|
+
function formatAttributeName(path) {
|
|
374
|
+
return path.split(".").reduce((acc, part) => {
|
|
375
|
+
if (util.matches("^d+$", part)) return acc;
|
|
376
|
+
return acc ? `${acc} ${part.toLowerCase()}` : part.toLowerCase();
|
|
377
|
+
}, "");
|
|
238
378
|
}
|
|
239
379
|
|
|
240
380
|
//#endregion
|
|
241
|
-
export {
|
|
242
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.js","names":["parsed: string | null","result: Rule<T>","errorMessages: string[]","rules.parse"],"sources":["../src/utils.ts","../src/rules.ts","../src/index.ts"],"sourcesContent":["import type { Context } from '@aws-appsync/utils'\nimport type { NestedKeyOf } from './types'\nimport { util } from '@aws-appsync/utils'\n\nexport function isString(value: unknown): value is string {\n  return typeof value === 'string'\n}\n\nexport function isArray(value: unknown): value is unknown[] {\n  if (typeof value === 'object' && !!value && Object.hasOwn(value, 'length')) {\n    return typeof (value as unknown[]).length === 'number'\n  }\n  return false\n}\n\nexport function getNestedValue<T extends object>(obj: T, path: NestedKeyOf<T>): any {\n  return path.split('.').reduce<unknown>((current, key) => util.matches('^\\d+$', key)\n    ? (current as unknown[])[Number(key)]\n    : (current as Record<string, unknown>)[key], obj)\n}\n\nexport function setNestedValue<T extends object>(obj: T, path: NestedKeyOf<T>, value: unknown): void {\n  const keys = path.split('.')\n  if (keys.length === 1) {\n    obj[keys[0] as keyof typeof obj] = value as any\n    return\n  }\n  const lastKey = keys.pop() as string\n  const parentObject = getNestedValue(obj, keys.join('.') as NestedKeyOf<T>)\n  if (typeof parentObject === 'object' && !!parentObject) {\n    parentObject[lastKey] = value\n  }\n}\n\nexport function getHeader(name: string, ctx: Context): string | null {\n  return Object.entries(ctx.request.headers)\n    .reduce((prev, [key, value]) => typeof prev === 'string'\n      ? prev\n      : (key.toLowerCase() === name.toLowerCase() && typeof value === 'string'\n          ? value\n          : null), null as string | null)\n}\n\nexport function isPrecognitiveRequest(ctx: Context): boolean {\n  return getHeader('precognition', ctx) === 'true'\n}\n\nexport function precognitiveKeys(ctx: Context): string[] | null {\n  const keys = getHeader('Precognition-Validate-Only', ctx)\n  return keys ? keys.split(',').map(key => key.trim()) : null\n}\n\nexport function cleanString(value: string): string | null {\n  let parsed: string | null = value.trim()\n  if (parsed === '')\n    parsed = null\n  return parsed\n}\n","import type { Rule, ShortRule } from './types'\nimport { util } from '@aws-appsync/utils'\nimport { isArray } from './utils'\n\nexport const names = {\n  required: 'required',\n  nullable: 'nullable',\n  sometimes: 'sometimes',\n  min: 'min',\n  max: 'max',\n  between: 'between',\n  email: 'email',\n  url: 'url',\n  uuid: 'uuid',\n  regex: 'regex',\n  in: 'in',\n  notIn: 'notIn',\n  before: 'before',\n  after: 'after',\n} as const\n\nexport function parse<T>(value: T, rule: ShortRule<keyof typeof names>): Rule<T> {\n  const [ruleName, params] = rule.includes(':') ? rule.split(':', 2) : [rule, '']\n\n  switch (ruleName as keyof typeof names) {\n    case 'required':\n      return requiredRule(value)\n    case 'nullable':\n      return nullableRule(value)\n    case 'sometimes':\n      return sometimesRule(value)\n    case 'min':\n      return minRule(value, ...params.split(','))\n    case 'max':\n      return maxRule(value, ...params.split(', '))\n    case 'between':\n      return betweenRule(value, ...params.split(', '))\n    case 'email':\n      return emailRule(value)\n    case 'url':\n      return urlRule(value)\n    case 'uuid':\n      return uuidRule(value)\n    case 'regex':\n      return regexRule(value, params)\n    case 'in':\n      return inRule(value, ...params.split(', '))\n    default:\n      return { check: false, message: `Unknown rule ${ruleName}`, value }\n  }\n}\n\nfunction minRule<T>(value: T, ...params: string[]): Rule<T> {\n  const minValue = Number(params[0] ?? '0')\n  const result: Rule<T> = {\n    check: false,\n    message: `:attribute must be greater than or equal to ${minValue}`,\n    value,\n  }\n  if (typeof value === 'number') {\n    result.check = value >= minValue\n  }\n  if (typeof result.value === 'string') {\n    result.check = (result.value).length >= minValue\n  }\n  if (isArray(value)) {\n    result.check = value.length >= minValue\n    result.message = `Array must contain at least ${minValue} elements`\n  }\n  return result\n}\n\nfunction maxRule<T>(value: T, ...params: string[]): Rule<T> {\n  const maxValue = Number(params[0] ?? '0')\n  const result: Rule<T> = {\n    check: false,\n    message: `:attribute must be less than or equal to ${maxValue}`,\n    value,\n  }\n\n  if (typeof value === 'number') {\n    result.check = value <= maxValue\n  }\n  if (typeof result.value === 'string') {\n    result.check = result.value.length <= maxValue\n    result.message = `String must contain at most ${maxValue} characters`\n  }\n  if (isArray(value)) {\n    result.check = value.length <= maxValue\n    result.message = `Array must contain at most ${maxValue} elements`\n  }\n  return result\n}\n\nfunction betweenRule<T>(value: T, ...params: string[]): Rule<T> {\n  const minValue = Number(params[0] ?? '0')\n  const maxValue = Number(params[1] ?? '0')\n  const result: Rule<T> = {\n    check: false,\n    message: `:attribute must be between ${minValue} and ${maxValue}`,\n    value,\n  }\n  if (typeof value === 'number') {\n    result.check = value >= minValue && value <= maxValue\n  }\n  if (typeof result.value === 'string') {\n    result.check = result.value.length >= minValue && (result.value).length <= maxValue\n    result.message = `String must contain between ${minValue} and ${maxValue} characters`\n  }\n  if (isArray(value)) {\n    result.check = value.length >= minValue && value.length <= maxValue\n    result.message = `Array must contain between ${minValue} and ${maxValue} elements`\n  }\n  return result\n}\n\nfunction emailRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must be a valid email address',\n    value,\n  }\n\n  if (typeof value === 'string') {\n    result.check = util.matches('^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$', result.value as string)\n  }\n  return result\n}\n\nfunction urlRule<T>(value: T): Rule<T> {\n  const result = {\n    check: false,\n    message: ':attribute must be a valid URL',\n    value,\n  }\n  if (typeof value === 'string') {\n    result.check = util.matches(\n      '^(http|https):\\\\/\\\\/[\\\\w\\\\-_]+(\\\\.[\\\\w\\\\-_]+)+([\\\\w\\\\-\\\\.,@?^=%&:/~\\\\+#]*[\\\\w\\\\-\\\\@?^=%&/~\\\\+#])?$',\n      result.value as string,\n    )\n  }\n  return result\n}\n\nfunction uuidRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must be a valid UUID',\n    value,\n  }\n  if (typeof result.value === 'string') {\n    result.check = util.matches(\n      '^[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$',\n      value as string,\n    )\n  }\n  return result\n}\n\nfunction regexRule<T>(value: T, ...params: string[]): Rule<T> {\n  const regex = params[0] ?? ''\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must match the specified regular expression',\n    value,\n  }\n  if (typeof result.value === 'string') {\n    result.value = result.value.trim() as T\n    result.check = util.matches(regex, result.value as string)\n  }\n  return result\n}\n\nfunction inRule<T>(value: T, ...params: string[]): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must be one of the specified values',\n    value,\n  }\n  if (typeof result.value === 'string') {\n    result.check = params.includes(result.value)\n  }\n  if (typeof value === 'number') {\n    result.check = params.map(Number).includes(value)\n  }\n  return result\n}\n\nexport function requiredRule<T>(value: T, ..._params: string[]): Rule<T> {\n  const result: Rule<T> = {\n    check: true,\n    message: ':attribute is required',\n    value,\n  }\n  if (typeof result.value === 'string') {\n    result.check = result.value.length > 0\n  }\n  if (isArray(result.value)) {\n    result.check = result.value.length > 0\n  }\n  if (typeof result.value === 'number') {\n    result.check = true\n  }\n  if (result.check === false) {\n    result.message = ':attribute is required'\n  }\n  return result\n}\n\nfunction nullableRule<T>(value: T, ..._params: string[]): Rule<T> {\n  const result: Rule<T> = {\n    check: true,\n    message: '',\n    value,\n  }\n  return result\n}\n\nfunction sometimesRule<T>(value: T, ..._params: string[]): Rule<T> {\n  const result: Rule<T> = {\n    check: true,\n    message: '',\n    value,\n  }\n  if (typeof result.value === 'undefined') {\n    return result\n  }\n  return requiredRule(value, ..._params)\n}\n","import type { Context } from '@aws-appsync/utils'\nimport type { NestedKeyOf, Rule, ShortRule } from './types'\nimport { runtime, util } from '@aws-appsync/utils'\nimport * as rules from './rules'\nimport { cleanString, getNestedValue, isPrecognitiveRequest, precognitiveKeys, setNestedValue } from './utils'\n\nexport function validate<T extends object>(\n  obj: T,\n  checks: Partial<Record<NestedKeyOf<T>, (ShortRule<keyof typeof rules['names']> | Rule)[]>>,\n): T {\n  let hasErrors = false\n  const errorMessages: string[] = []\n\n  Object.keys(checks).forEach((path) => {\n    let value = getNestedValue(obj, path as NestedKeyOf<T>)\n    if (typeof value === 'string') {\n      value = cleanString(value)\n      setNestedValue(obj, path as NestedKeyOf<T>, value)\n    }\n\n    let skip = false\n    checks[path as NestedKeyOf<T>]?.forEach((rule) => {\n      if (skip) {\n        return\n      }\n\n      if (rule === 'nullable' && value === null) {\n        skip = true\n      }\n      if (rule === 'sometimes' && typeof value === 'undefined') {\n        skip = true\n      }\n\n      const result = (typeof rule === 'string') ? rules.parse(value, rule) : { ...rule }\n      if (result.check)\n        return\n      hasErrors = true\n      errorMessages.push(result.message)\n      util.appendError(result.message, 'ValidationError', null, { path, value })\n      if (rule === 'required') {\n        skip = true\n      }\n    })\n  })\n\n  if (hasErrors) {\n    util.error(errorMessages[0], 'ValidationError')\n  }\n\n  return obj\n}\n\nexport function precognition<T extends object>(\n  ctx: Context<T>,\n  checks: Partial<Record<NestedKeyOf<T>, (ShortRule<keyof typeof rules['names']> | Rule)[]>>,\n): T {\n  if (!isPrecognitiveRequest(ctx)) {\n    return validate(ctx.args, checks)\n  }\n  const validationKeys = precognitiveKeys(ctx)\n  util.http.addResponseHeader('Precognition', 'true')\n\n  if (!validationKeys) {\n    validate(ctx.args, checks)\n    util.http.addResponseHeader('Precognition-Success', 'true')\n    runtime.earlyReturn(null)\n  }\n\n  util.http.addResponseHeader('Precognition-Validate-Only', validationKeys.join(','))\n  const precognitionChecks = {} as Partial<typeof checks>\n  validationKeys.forEach((key) => {\n    precognitionChecks[key as NestedKeyOf<T>] = checks[key as NestedKeyOf<T>]\n  })\n\n  validate(ctx.args, precognitionChecks)\n  util.http.addResponseHeader('Precognition-Success', 'true')\n  runtime.earlyReturn(null)\n}\n"],"mappings":"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"}
|
|
381
|
+
export { formatAttributeName, precognitiveValidation, validate };
|
|
382
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.js","names":["parsed: string | null","result: Rule<T>","error: { msg?: string, errorType?: string, data?: any, errorInfo?: any }","rules.parse"],"sources":["../src/utils.ts","../src/rules.ts","../src/index.ts"],"sourcesContent":["import type { Context } from '@aws-appsync/utils'\nimport type { NestedKeyOf } from './types'\nimport { util } from '@aws-appsync/utils'\n\nexport function isString(value: unknown): value is string {\n  return typeof value === 'string'\n}\n\nexport function isArray(value: unknown): value is unknown[] {\n  if (typeof value === 'object' && !!value && Object.hasOwn(value, 'length')) {\n    return typeof (value as unknown[]).length === 'number'\n  }\n  return false\n}\n\nexport function getNestedValue<T extends object>(obj: T, path: NestedKeyOf<T>): any {\n  return path.split('.').reduce<unknown>((current, key) => util.matches('^\\d+$', key)\n    ? (current as unknown[])[Number(key)]\n    : (current as Record<string, unknown>)[key], obj)\n}\n\nexport function setNestedValue<T extends object>(obj: T, path: NestedKeyOf<T>, value: unknown): void {\n  const keys = path.split('.')\n  if (keys.length === 1) {\n    obj[keys[0] as keyof typeof obj] = value as any\n    return\n  }\n  const lastKey = keys.pop() as string\n  const parentObject = getNestedValue(obj, keys.join('.') as NestedKeyOf<T>)\n  if (typeof parentObject === 'object' && !!parentObject) {\n    parentObject[lastKey] = value\n  }\n}\n\nexport function getHeader(name: string, ctx: Context): string | null {\n  return Object.entries(ctx.request.headers)\n    .reduce((prev, [key, value]) => typeof prev === 'string'\n      ? prev\n      : (key.toLowerCase() === name.toLowerCase() && typeof value === 'string'\n          ? value\n          : null), null as string | null)\n}\n\nexport function isPrecognitiveRequest(ctx: Context): boolean {\n  return getHeader('precognition', ctx) === 'true'\n}\n\nexport function precognitiveKeys(ctx: Context): string[] | null {\n  const keys = getHeader('Precognition-Validate-Only', ctx)\n  return keys ? keys.split(',').map(key => key.trim()) : null\n}\n\nexport function cleanString(value: string, options?: {\n  trim?: boolean\n  allowEmptyString?: boolean\n}): string | null {\n  if (options?.trim === false)\n    return value\n\n  let parsed: string | null = value.trim()\n\n  if (options?.allowEmptyString)\n    return parsed\n\n  if (parsed === '')\n    parsed = null\n\n  return parsed\n}\n","import type { FullRule, Rule } from './types'\nimport { util } from '@aws-appsync/utils'\nimport { isArray } from './utils'\n\nexport function parse<T>(value: T, rule: FullRule): Rule<T> {\n  const [name, ...params] = typeof rule === 'string'\n    ? [rule, undefined]\n    : [rule[0], ...rule.slice(1)]\n\n  switch (name) {\n    case 'required':\n      return requiredRule(value)\n    case 'nullable':\n      return nullableRule(value)\n    case 'sometimes':\n      return sometimesRule(value)\n    case 'min':\n      return minRule(value, (params[0]! as number))\n    case 'max':\n      return maxRule(value, (params[0] as number))\n    case 'between':\n      return betweenRule(value, (params[0] as number), params[1] as number)\n    case 'email':\n      return emailRule(value)\n    case 'url':\n      return urlRule(value)\n    case 'uuid':\n      return uuidRule(value)\n    case 'ulid':\n      return ulidRule(value)\n    case 'regex':\n      return regexRule(value, params[0] as string)\n    case 'in':\n      return inRule(value, ...params)\n    case 'notIn':\n      return notInRule(value, ...params)\n    case 'array':\n      return arrayRule(value)\n    case 'object':\n      return objectRule(value)\n    case 'boolean':\n      return booleanRule(value)\n    case 'number':\n      return numberRule(value)\n    case 'string':\n      return stringRule(value)\n    case 'date':\n      return dateRule(value)\n    case 'before':\n      return beforeRule(value, params[0] as string)\n    case 'after':\n      return afterRule(value, params[0] as string)\n    case 'beforeOrEqual':\n      return beforeOrEqualRule(value, params[0] as string)\n    case 'afterOrEqual':\n      return afterOrEqualRule(value, params[0] as string)\n    default:\n      return { check: false, message: `Unknown rule ${name}`, value }\n  }\n}\n\nfunction minRule<T>(value: T, minValue: number): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: `:attribute must be greater than or equal to ${minValue}`,\n    value,\n  }\n  if (typeof value === 'number') {\n    result.check = value >= minValue\n  }\n  if (typeof value === 'string') {\n    result.check = value.length >= minValue\n  }\n  if (isArray(value)) {\n    result.check = value.length >= minValue\n    result.message = `Array must contain at least ${minValue} elements`\n  }\n  return result\n}\n\nfunction maxRule<T>(value: T, maxValue: number): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: `:attribute must be less than or equal to ${maxValue}`,\n    value,\n  }\n\n  if (typeof value === 'number') {\n    result.check = value <= maxValue\n  }\n  if (typeof value === 'string') {\n    result.check = value.length <= maxValue\n    result.message = `String must contain at most ${maxValue} characters`\n  }\n  if (isArray(value)) {\n    result.check = value.length <= maxValue\n    result.message = `Array must contain at most ${maxValue} elements`\n  }\n  return result\n}\n\nfunction betweenRule<T>(value: T, minValue: number, maxValue: number): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: `:attribute must be between ${minValue} and ${maxValue}`,\n    value,\n  }\n  if (typeof value === 'number') {\n    result.check = value >= minValue && value <= maxValue\n  }\n  if (typeof value === 'string') {\n    result.check = value.length >= minValue && value.length <= maxValue\n    result.message = `String must contain between ${minValue} and ${maxValue} characters`\n  }\n  if (isArray(value)) {\n    result.check = value.length >= minValue && value.length <= maxValue\n    result.message = `Array must contain between ${minValue} and ${maxValue} elements`\n  }\n  return result\n}\n\nfunction emailRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must be a valid email address',\n    value,\n  }\n\n  if (typeof value === 'string') {\n    result.check = util.matches('^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\\\\.[a-zA-Z]{2,}$', result.value as string)\n  }\n  return result\n}\n\nfunction urlRule<T>(value: T): Rule<T> {\n  const result = {\n    check: false,\n    message: ':attribute must be a valid URL',\n    value,\n  }\n  if (typeof value === 'string') {\n    result.check = util.matches(\n      '^https?:\\\\/\\\\/(www\\\\.)?[-a-zA-Z0-9@:%._\\\\+~#=]{1,256}\\\\.[a-zA-Z0-9()]{1,6}\\\\b([-a-zA-Z0-9()@:%_\\\\+.~#?&//=]*)$|^https?:\\\\/\\\\/(localhost|\\\\d{1,3}\\\\.\\\\d{1,3}\\\\.\\\\d{1,3}\\\\.\\\\d{1,3})(:\\\\d+)?(\\\\/.*)?$',\n      result.value as string,\n    )\n  }\n  return result\n}\n\nfunction uuidRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must be a valid UUID',\n    value,\n  }\n  if (typeof value === 'string') {\n    result.check = util.matches(\n      '^[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$',\n      value as string,\n    )\n  }\n  return result\n}\n\nfunction ulidRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must be a valid ULID',\n    value,\n  }\n  if (typeof value === 'string') {\n    // ULID format: 26 characters, base32 encoded (0-9, A-Z excluding I, L, O, U)\n    result.check = util.matches(\n      '^[0123456789ABCDEFGHJKMNPQRSTVWXYZ]{26}$',\n      value as string,\n    )\n  }\n  return result\n}\n\nfunction regexRule<T>(value: T, pattern: string): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must match the specified regular expression',\n    value,\n  }\n  if (typeof value === 'string') {\n    result.value = value.trim() as T\n    result.check = util.matches(pattern, result.value as string)\n  }\n  return result\n}\n\nfunction inRule<T>(value: T, ...params: unknown[]): Rule<T> {\n  return {\n    check: params.includes(value),\n    message: ':attribute must be one of the specified values',\n    value,\n  }\n}\n\nfunction notInRule<T>(value: T, ...params: unknown[]): Rule<T> {\n  return {\n    check: !params.includes(value),\n    message: ':attribute must not be one of the specified values',\n    value,\n  }\n}\n\nexport function requiredRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: true,\n    message: ':attribute is required',\n    value,\n  }\n  if (typeof value === 'string') {\n    result.check = value.length > 0\n  }\n  if (isArray(value)) {\n    result.check = value.length > 0\n  }\n  if (typeof value === 'number') {\n    result.check = true\n  }\n  if (typeof value === 'boolean') {\n    result.check = true\n  }\n  if (typeof value === 'object' && !result.value) {\n    result.message = ':attribute is not nullable'\n    result.check = false\n  }\n  if (typeof value === 'undefined') {\n    result.check = false\n  }\n  return result\n}\n\nfunction nullableRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: true,\n    message: '',\n    value,\n  }\n  return result\n}\n\nfunction sometimesRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: true,\n    message: '',\n    value,\n  }\n  if (typeof value === 'undefined') {\n    return result\n  }\n  if (typeof value === 'object' && !result.value) {\n    result.message = ':attribute is not nullable'\n    result.check = false\n  }\n  return requiredRule(value)\n}\n\nfunction arrayRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must be an array',\n    value,\n  }\n  if (isArray(value)) {\n    result.check = true\n  }\n  return result\n}\n\nfunction objectRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must be an object',\n    value,\n  }\n  if (typeof value === 'object' && !isArray(result.value)) {\n    result.check = true\n  }\n  return result\n}\n\nfunction booleanRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must be a boolean',\n    value,\n  }\n  if (typeof value === 'boolean') {\n    result.check = true\n  }\n  return result\n}\n\nfunction numberRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must be a number',\n    value,\n  }\n  if (typeof value === 'number') {\n    result.check = true\n  }\n  return result\n}\n\nfunction stringRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must be a string',\n    value,\n  }\n  if (typeof value === 'string') {\n    result.check = true\n  }\n  return result\n}\n\nfunction dateRule<T>(value: T): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: ':attribute must be a date',\n    value,\n  }\n  if (typeof value === 'string') {\n    result.check = util.matches(\n      '^\\\\d{4}-(0[1-9]|1[0-2])-(0[1-9]|[12]\\\\d|3[01])T([01]\\\\d|2[0-3]):[0-5]\\\\d:[0-5]\\\\d(\\\\.\\\\d{1,6})?Z$',\n      result.value as string,\n    )\n  }\n  if (typeof value === 'number') {\n    result.check = true\n  }\n  return result\n}\n\nfunction beforeRule<T>(value: T, start: string): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: `:attribute must be before ${start}`,\n    value,\n  }\n  const startValue = util.time.parseISO8601ToEpochMilliSeconds(start)\n\n  if (typeof value === 'string') {\n    const date = util.time.parseISO8601ToEpochMilliSeconds(value)\n    result.check = date < startValue\n  }\n\n  if (typeof value === 'number') {\n    result.check = value < startValue\n  }\n  return result\n}\n\nfunction afterRule<T>(value: T, start: string): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: `:attribute must be after ${start}`,\n    value,\n  }\n  const startValue = util.time.parseISO8601ToEpochMilliSeconds(start)\n\n  if (typeof value === 'string') {\n    const date = util.time.parseISO8601ToEpochMilliSeconds(value)\n    result.check = date > startValue\n  }\n\n  if (typeof value === 'number') {\n    result.check = value > startValue\n  }\n  return result\n}\n\nfunction beforeOrEqualRule<T>(value: T, start: string): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: `:attribute must be before or equal to ${start}`,\n    value,\n  }\n  const startValue = util.time.parseISO8601ToEpochMilliSeconds(start)\n\n  if (typeof value === 'string') {\n    const date = util.time.parseISO8601ToEpochMilliSeconds(value)\n    result.check = date <= startValue\n  }\n\n  if (typeof value === 'number') {\n    result.check = value <= startValue\n  }\n  return result\n}\n\nfunction afterOrEqualRule<T>(value: T, start: string): Rule<T> {\n  const result: Rule<T> = {\n    check: false,\n    message: `:attribute must be after or equal to ${start}`,\n    value,\n  }\n  const startValue = util.time.parseISO8601ToEpochMilliSeconds(start)\n\n  if (typeof value === 'string') {\n    const date = util.time.parseISO8601ToEpochMilliSeconds(value)\n    result.check = date >= startValue\n  }\n\n  if (typeof value === 'number') {\n    result.check = value >= startValue\n  }\n  return result\n}\n","import type { Context } from '@aws-appsync/utils'\nimport type { FullRule, NestedKeyOf, Rule } from './types'\nimport { runtime, util } from '@aws-appsync/utils'\nimport * as rules from './rules'\nimport { cleanString, getNestedValue, isArray, isPrecognitiveRequest, precognitiveKeys, setNestedValue } from './utils'\n\nexport function validate<T extends object>(\n  obj: T,\n  checks: Partial<Record<NestedKeyOf<T>, (FullRule | Rule)[]>>,\n  options?: {\n    trim?: boolean\n    allowEmptyString?: boolean\n  },\n): T {\n  let error: { msg?: string, errorType?: string, data?: any, errorInfo?: any } = {}\n\n  Object.keys(checks).forEach((path) => {\n    let value = getNestedValue(obj, path as NestedKeyOf<T>)\n    if (typeof value === 'string') {\n      value = cleanString(value, options)\n      setNestedValue(obj, path as NestedKeyOf<T>, value)\n    }\n\n    let skip = false\n    checks[path as NestedKeyOf<T>]?.forEach((rule) => {\n      if (skip) {\n        return\n      }\n\n      if (rule === 'nullable' && value === null)\n        skip = true\n\n      if (rule === 'sometimes' && typeof value === 'undefined')\n        skip = true\n\n      const result = (typeof rule === 'string' || isArray(rule))\n        ? rules.parse(value, rule)\n        : { ...rule }\n\n      if (result.check)\n        return\n\n      if (error.msg)\n        util.appendError(error.msg, error.errorType, error.data, error.errorInfo)\n\n      result.message = result.message.replace(':attribute', formatAttributeName(path))\n      error = {\n        msg: result.message,\n        errorType: 'ValidationError',\n        data: null,\n        errorInfo: { path, value },\n      }\n\n      skip = true\n    })\n  })\n\n  if (!error.msg) {\n    return obj\n  }\n\n  util.error(error.msg, error.errorType, error.data, error.errorInfo)\n}\n\nexport function precognitiveValidation<T extends object>(\n  ctx: Context<T>,\n  checks: Partial<Record<NestedKeyOf<T>, (FullRule | Rule)[]>>,\n  options?: {\n    trim?: boolean\n    allowEmptyString?: boolean\n    skipTo?: 'END' | 'NEXT'\n  },\n): T {\n  if (!isPrecognitiveRequest(ctx)) {\n    return validate(ctx.args, checks, options)\n  }\n  const validationKeys = precognitiveKeys(ctx)\n  util.http.addResponseHeader('Precognition', 'true')\n\n  if (!validationKeys) {\n    validate(ctx.args, checks, options)\n    util.http.addResponseHeader('Precognition-Success', 'true')\n    runtime.earlyReturn(null)\n  }\n\n  util.http.addResponseHeader('Precognition-Validate-Only', validationKeys.join(','))\n  const precognitionChecks = {} as Partial<typeof checks>\n  validationKeys.forEach((key) => {\n    precognitionChecks[key as NestedKeyOf<T>] = checks[key as NestedKeyOf<T>]\n  })\n\n  validate(ctx.args, precognitionChecks, options)\n  util.http.addResponseHeader('Precognition-Success', 'true')\n  runtime.earlyReturn(null, { skipTo: options?.skipTo ?? 'END' })\n}\n\nexport function formatAttributeName(path: string): string {\n  return path.split('.').reduce((acc, part) => {\n    if (util.matches('^\\d+$', part)) {\n      return acc\n    }\n    return acc ? `${acc} ${part.toLowerCase()}` : part.toLowerCase()\n  }, '')\n}\n"],"mappings":"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"}
|