@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 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 ShortRule<T extends string> = T | `${T}:${string}`;
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>, (ShortRule<keyof typeof rules_d_exports["names"]> | Rule)[]>>): T;
42
- declare function precognition<T extends object>(ctx: Context<T>, checks: Partial<Record<NestedKeyOf<T>, (ShortRule<keyof typeof rules_d_exports["names"]> | Rule)[]>>): T;
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 { precognition, validate };
45
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguZC50cyIsIm5hbWVzIjpbXSwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiLCIuLi9zcmMvcnVsZXMudHMiLCIuLi9zcmMvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7VUFFaUI7OztTQUdSOztLQUdHLDhCQUE4QixPQUFPO0tBRTVDLGlDQUNELG9DQUNFLFVBQVU7S0FPWCwrQkFDRCxzQkFDRSxVQUFVLFdBQ0o7VUFFRixjQUFBOzs7S0FJRSxpQkFBaUIsVUFBVSw2QkFDbkMsVUFBVSx5QkFFUixVQUFVLDBEQUdMLFdBQVcsTUFBTSxFQUFFLHlCQUNiLFdBQVcsS0FBSyxHQUFHLFNBQVMsWUFBWSxFQUFFLFVBQzdDLHNCQUNFLDRDQUVFLHdDQUNlLFlBQVk7Ozs7Y0NwQ2hDOzs7Ozs7Ozs7Ozs7Ozs7O2lCQWlCRyxnQkFBZ0IsU0FBUyx1QkFBdUIsU0FBUyxLQUFLO2lCQXVLOUQsdUJBQXVCLDBCQUEwQixLQUFLOzs7aUJDdEx0RCxnQ0FDVCxXQUNHLFFBQVEsT0FBTyxZQUFZLEtBQUssdUJBQXVCLDRCQUFrQixZQUNoRjtpQkEyQ2Esb0NBQ1QsUUFBUSxZQUNMLFFBQVEsT0FBTyxZQUFZLEtBQUssdUJBQXVCLDRCQUFrQixZQUNoRiJ9
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 [ruleName, params] = rule.includes(":") ? rule.split(":", 2) : [rule, ""];
41
- switch (ruleName) {
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, ...params.split(","));
46
- case "max": return maxRule(value, ...params.split(", "));
47
- case "between": return betweenRule(value, ...params.split(", "));
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 "regex": return regexRule(value, params);
52
- case "in": return inRule(value, ...params.split(", "));
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 ${ruleName}`,
69
+ message: `Unknown rule ${name}`,
56
70
  value
57
71
  };
58
72
  }
59
73
  }
60
- function minRule(value, ...params) {
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 result.value === "string") result.check = result.value.length >= minValue;
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, ...params) {
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 result.value === "string") {
84
- result.check = result.value.length <= maxValue;
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, ...params) {
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 result.value === "string") {
103
- result.check = result.value.length >= minValue && result.value.length <= maxValue;
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("^[^s@]+@[^s@]+.[^s@]+$", result.value);
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("^(http|https):\\/\\/[\\w\\-_]+(\\.[\\w\\-_]+)+([\\w\\-\\.,@?^=%&:/~\\+#]*[\\w\\-\\@?^=%&/~\\+#])?$", result.value);
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 result.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);
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, ...params) {
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 result.value === "string") {
147
- result.value = result.value.trim();
148
- result.check = util.matches(regex, result.value);
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
- const result = {
154
- check: false,
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 requiredRule(value, ..._params) {
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 result.value === "string") result.check = result.value.length > 0;
169
- if (isArray(result.value)) result.check = result.value.length > 0;
170
- if (typeof result.value === "number") result.check = true;
171
- if (result.check === false) result.message = ":attribute is required";
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, ..._params) {
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, ..._params) {
208
+ function sometimesRule(value) {
182
209
  const result = {
183
210
  check: true,
184
211
  message: "",
185
212
  value
186
213
  };
187
- if (typeof result.value === "undefined") return result;
188
- return requiredRule(value, ..._params);
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 hasErrors = false;
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
- hasErrors = true;
210
- errorMessages.push(result.message);
211
- util.appendError(result.message, "ValidationError", null, {
212
- path,
213
- value
214
- });
215
- if (rule === "required") skip = true;
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 (hasErrors) util.error(errorMessages[0], "ValidationError");
219
- return obj;
352
+ if (!error.msg) return obj;
353
+ util.error(error.msg, error.errorType, error.data, error.errorInfo);
220
354
  }
221
- function precognition(ctx, checks) {
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 { precognition, validate };
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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"}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@sot1986/appsync-precognition",
3
3
  "type": "module",
4
- "version": "0.0.2",
4
+ "version": "0.0.4",
5
5
  "description": "JavaScript resolver validation utilities for AWS AppSync",
6
6
  "author": "sot1986",
7
7
  "license": "MIT",