@superblocksteam/sabs-client 0.0.1-demo-databricks-deploy

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.
@@ -0,0 +1,341 @@
1
+ import {
2
+ BadRequestError,
3
+ InternalServerError,
4
+ NotFoundError,
5
+ ConflictError,
6
+ UnauthorizedError,
7
+ ForbiddenError,
8
+ HttpError,
9
+ createErrorFromStatusCode,
10
+ createClientError,
11
+ createNetworkError
12
+ } from './errors';
13
+
14
+ describe('errors', () => {
15
+ describe('HttpError for unhandled status codes', () => {
16
+ test('creates error with correct properties', () => {
17
+ const status = 418;
18
+ const message = 'I am a teapot';
19
+
20
+ const error = new HttpError(status, message);
21
+
22
+ expect(error).toBeInstanceOf(Error);
23
+ expect(error).toBeInstanceOf(HttpError);
24
+ expect(error.name).toBe('Error');
25
+ expect(error.message).toBe(message);
26
+ expect(error.status).toBe(status);
27
+ expect(error.title).toBeUndefined(); // HttpError doesn't set title by default
28
+ });
29
+
30
+ test('has consistent interface with other error types', () => {
31
+ const error = new HttpError(422, 'Test message');
32
+
33
+ expect(error.message).toBe('Test message');
34
+ expect(error.status).toBe(422);
35
+
36
+ // Should have the same interface as other HttpError types
37
+ expect('status' in error).toBe(true);
38
+ expect('message' in error).toBe(true);
39
+ });
40
+
41
+ test('has proper stack trace', () => {
42
+ const error = new HttpError(500, 'Test error');
43
+ expect(error.stack).toBeDefined();
44
+ expect(error.stack).toContain('Test error');
45
+ });
46
+ });
47
+
48
+ describe('createErrorFromStatusCode', () => {
49
+ test('creates BadRequestError for status 400', () => {
50
+ const message = 'Invalid request';
51
+ const error = createErrorFromStatusCode(400, message);
52
+
53
+ expect(error).toBeInstanceOf(BadRequestError);
54
+ expect(error.message).toBe('SABS API Error (400): Invalid request');
55
+ });
56
+
57
+ test('creates UnauthorizedError for status 401', () => {
58
+ const message = 'Invalid credentials';
59
+ const error = createErrorFromStatusCode(401, message);
60
+
61
+ expect(error).toBeInstanceOf(UnauthorizedError);
62
+ expect(error.message).toBe('SABS API Error (401): Invalid credentials');
63
+ });
64
+
65
+ test('creates ForbiddenError for status 403', () => {
66
+ const message = 'Access denied';
67
+ const error = createErrorFromStatusCode(403, message);
68
+
69
+ expect(error).toBeInstanceOf(ForbiddenError);
70
+ expect(error.message).toBe('SABS API Error (403): Access denied');
71
+ });
72
+
73
+ test('creates NotFoundError for status 404', () => {
74
+ const message = 'Resource not found';
75
+ const error = createErrorFromStatusCode(404, message);
76
+
77
+ expect(error).toBeInstanceOf(NotFoundError);
78
+ expect(error.message).toBe('SABS API Error (404): Resource not found');
79
+ });
80
+
81
+ test('creates ConflictError for status 409', () => {
82
+ const message = 'Resource conflict';
83
+ const error = createErrorFromStatusCode(409, message);
84
+
85
+ expect(error).toBeInstanceOf(ConflictError);
86
+ expect(error.message).toBe('SABS API Error (409): Resource conflict');
87
+ });
88
+
89
+ test.each([500, 502, 503, 504])('creates InternalServerError for status %s', (statusCode) => {
90
+ const message = 'Server error';
91
+ const error = createErrorFromStatusCode(statusCode, message);
92
+
93
+ expect(error).toBeInstanceOf(InternalServerError);
94
+ expect(error.message).toBe(`SABS API Error (${statusCode}): Server error`);
95
+ });
96
+
97
+ test('creates HttpError for unhandled status codes', () => {
98
+ const statusCode = 418; // I'm a teapot
99
+ const message = 'Teapot error';
100
+
101
+ const error = createErrorFromStatusCode(statusCode, message);
102
+
103
+ expect(error).toBeInstanceOf(HttpError);
104
+ expect(error.message).toBe('SABS API Error (418): Teapot error');
105
+
106
+ if (error instanceof HttpError) {
107
+ expect(error.status).toBe(statusCode);
108
+ }
109
+ });
110
+
111
+ test('handles edge case status codes', () => {
112
+ // Test some edge cases
113
+ const testCases = [
114
+ { status: 100, expected: HttpError },
115
+ { status: 200, expected: HttpError },
116
+ { status: 300, expected: HttpError },
117
+ { status: 451, expected: HttpError }, // Unavailable For Legal Reasons
118
+ { status: 999, expected: HttpError }
119
+ ];
120
+
121
+ testCases.forEach(({ status, expected }) => {
122
+ const error = createErrorFromStatusCode(status, 'Test message');
123
+ expect(error).toBeInstanceOf(expected);
124
+ });
125
+ });
126
+
127
+ test('all error types have consistent HttpError interface', () => {
128
+ const testCases = [
129
+ { statusCode: 400, expectedType: BadRequestError, expectedTitle: 'Bad request' },
130
+ { statusCode: 500, expectedType: InternalServerError, expectedTitle: 'Internal server error' },
131
+ { statusCode: 418, expectedType: HttpError, expectedTitle: undefined }
132
+ ];
133
+
134
+ testCases.forEach(({ statusCode, expectedType, expectedTitle }) => {
135
+ const error = createErrorFromStatusCode(statusCode, 'Test message');
136
+
137
+ expect(error).toBeInstanceOf(expectedType);
138
+ expect(error.message).toBe(`SABS API Error (${statusCode}): Test message`);
139
+
140
+ // All error types should have consistent HttpError interface
141
+ expect('status' in error).toBe(true);
142
+ if ('status' in error) {
143
+ expect(error.status).toBe(statusCode);
144
+ }
145
+ if ('title' in error) {
146
+ expect(error.title).toBe(expectedTitle);
147
+ }
148
+ });
149
+ });
150
+
151
+ test('returns standard error types for known status codes', () => {
152
+ const testCases = [
153
+ { status: 400, expectedType: BadRequestError },
154
+ { status: 401, expectedType: UnauthorizedError },
155
+ { status: 403, expectedType: ForbiddenError },
156
+ { status: 404, expectedType: NotFoundError },
157
+ { status: 409, expectedType: ConflictError },
158
+ { status: 500, expectedType: InternalServerError },
159
+ { status: 502, expectedType: InternalServerError },
160
+ { status: 503, expectedType: InternalServerError },
161
+ { status: 504, expectedType: InternalServerError }
162
+ ];
163
+
164
+ testCases.forEach(({ status, expectedType }) => {
165
+ const error = createErrorFromStatusCode(status, 'Test message');
166
+
167
+ expect(error).toBeInstanceOf(Error);
168
+ expect(error).toBeInstanceOf(expectedType);
169
+ expect(error.name).toBe('Error'); // Shared library errors use 'Error' as name
170
+ expect(error.message).toBe(`SABS API Error (${status}): Test message`);
171
+ });
172
+ });
173
+ });
174
+
175
+ describe('createClientError', () => {
176
+ test('creates BadRequestError with client error prefix', () => {
177
+ const message = 'Missing required parameter';
178
+ const error = createClientError(message);
179
+
180
+ expect(error).toBeInstanceOf(BadRequestError);
181
+ expect(error.message).toBe('SABS Client Error: Missing required parameter');
182
+ });
183
+
184
+ test('handles empty message', () => {
185
+ const error = createClientError('');
186
+ expect(error.message).toBe('SABS Client Error: ');
187
+ });
188
+
189
+ test('handles special characters in message', () => {
190
+ const message = 'Invalid characters: <>&"\'';
191
+ const error = createClientError(message);
192
+ expect(error.message).toBe(`SABS Client Error: ${message}`);
193
+ });
194
+ });
195
+
196
+ describe('createNetworkError', () => {
197
+ test('creates InternalServerError with network error prefix', () => {
198
+ const message = 'Connection refused';
199
+ const error = createNetworkError(message);
200
+
201
+ expect(error).toBeInstanceOf(InternalServerError);
202
+ expect(error.message).toBe('SABS Network Error: Connection refused');
203
+ });
204
+
205
+ test('includes original error message when provided', () => {
206
+ const message = 'Request failed';
207
+ const originalError = new Error('ECONNREFUSED');
208
+ const error = createNetworkError(message, originalError);
209
+
210
+ expect(error).toBeInstanceOf(InternalServerError);
211
+ expect(error.message).toBe('SABS Network Error: Request failed (caused by: ECONNREFUSED)');
212
+ });
213
+
214
+ test('handles original error without message', () => {
215
+ const message = 'Network issue';
216
+ const originalError = new Error(); // Empty message
217
+ const error = createNetworkError(message, originalError);
218
+
219
+ expect(error.message).toBe('SABS Network Error: Network issue (caused by: )');
220
+ });
221
+
222
+ test('handles null original error', () => {
223
+ const message = 'Unknown network error';
224
+ const error = createNetworkError(message, undefined);
225
+
226
+ expect(error.message).toBe('SABS Network Error: Unknown network error');
227
+ });
228
+ });
229
+
230
+ describe('error inheritance and instanceof checks', () => {
231
+ test('all error types are instances of Error', () => {
232
+ const errors = [
233
+ createErrorFromStatusCode(400, 'Bad Request'),
234
+ createErrorFromStatusCode(401, 'Unauthorized'),
235
+ createErrorFromStatusCode(403, 'Forbidden'),
236
+ createErrorFromStatusCode(404, 'Not Found'),
237
+ createErrorFromStatusCode(409, 'Conflict'),
238
+ createErrorFromStatusCode(500, 'Internal Server Error'),
239
+ createErrorFromStatusCode(418, 'Teapot'),
240
+ createClientError('Client error'),
241
+ createNetworkError('Network error'),
242
+ new HttpError(422, 'Custom error')
243
+ ];
244
+
245
+ errors.forEach((error) => {
246
+ expect(error).toBeInstanceOf(Error);
247
+ expect(error.message).toBeDefined();
248
+ expect(typeof error.message).toBe('string');
249
+ expect(error.name).toBeDefined();
250
+ });
251
+ });
252
+
253
+ test('error types can be distinguished via instanceof', () => {
254
+ const badRequestError = createErrorFromStatusCode(400, 'Bad Request');
255
+ const unauthorizedError = createErrorFromStatusCode(401, 'Unauthorized');
256
+ const notFoundError = createErrorFromStatusCode(404, 'Not Found');
257
+ const httpError = createErrorFromStatusCode(418, 'Teapot');
258
+
259
+ // All errors extend HttpError, so they're all instances of HttpError
260
+ expect(badRequestError).toBeInstanceOf(BadRequestError);
261
+ expect(badRequestError).toBeInstanceOf(HttpError);
262
+ expect(badRequestError).not.toBeInstanceOf(UnauthorizedError);
263
+
264
+ expect(unauthorizedError).toBeInstanceOf(UnauthorizedError);
265
+ expect(unauthorizedError).toBeInstanceOf(HttpError);
266
+ expect(unauthorizedError).not.toBeInstanceOf(BadRequestError);
267
+
268
+ expect(notFoundError).toBeInstanceOf(NotFoundError);
269
+ expect(notFoundError).toBeInstanceOf(HttpError);
270
+ expect(notFoundError).not.toBeInstanceOf(BadRequestError);
271
+
272
+ expect(httpError).toBeInstanceOf(HttpError);
273
+ expect(httpError).not.toBeInstanceOf(BadRequestError);
274
+ });
275
+ });
276
+
277
+ describe('error serialization and JSON handling', () => {
278
+ test('errors can be converted to JSON for logging', () => {
279
+ const error = new HttpError(422, 'Test error');
280
+
281
+ // Test that error properties are accessible for logging
282
+ const logObject = {
283
+ message: error.message,
284
+ name: error.name,
285
+ status: error.status,
286
+ title: error.title
287
+ };
288
+
289
+ expect(logObject.message).toBe('Test error');
290
+ expect(logObject.name).toBe('Error');
291
+ expect(logObject.status).toBe(422);
292
+ expect(logObject.title).toBeUndefined();
293
+ });
294
+
295
+ test('shared error types have basic error properties', () => {
296
+ const error = createErrorFromStatusCode(400, 'Validation failed');
297
+
298
+ const logObject = {
299
+ message: error.message,
300
+ name: error.name,
301
+ stack: error.stack
302
+ };
303
+
304
+ expect(logObject.message).toBe('SABS API Error (400): Validation failed');
305
+ expect(logObject.name).toBeDefined();
306
+ expect(logObject.stack).toBeDefined();
307
+ });
308
+ });
309
+
310
+ describe('edge cases and error conditions', () => {
311
+ test('handles undefined and null inputs gracefully', () => {
312
+ // These should not throw, but create errors with undefined/null content
313
+ expect(() => createErrorFromStatusCode(400, undefined as any)).not.toThrow();
314
+ expect(() => createErrorFromStatusCode(400, null as any)).not.toThrow();
315
+ expect(() => createClientError(undefined as any)).not.toThrow();
316
+ expect(() => createNetworkError(undefined as any)).not.toThrow();
317
+ });
318
+
319
+ test('handles very large status codes', () => {
320
+ const error = createErrorFromStatusCode(999999, 'Large status code');
321
+ expect(error).toBeInstanceOf(HttpError);
322
+ if (error instanceof HttpError) {
323
+ expect(error.status).toBe(999999);
324
+ }
325
+ });
326
+
327
+ test('handles negative status codes', () => {
328
+ const error = createErrorFromStatusCode(-1, 'Negative status code');
329
+ expect(error).toBeInstanceOf(HttpError);
330
+ if (error instanceof HttpError) {
331
+ expect(error.status).toBe(-1);
332
+ }
333
+ });
334
+
335
+ test('handles very long error messages', () => {
336
+ const longMessage = 'x'.repeat(10000);
337
+ const error = createErrorFromStatusCode(400, longMessage);
338
+ expect(error.message).toContain(longMessage);
339
+ });
340
+ });
341
+ });
package/src/errors.ts ADDED
@@ -0,0 +1,60 @@
1
+ import {
2
+ BadRequestError,
3
+ InternalServerError,
4
+ NotFoundError,
5
+ ConflictError,
6
+ UnauthorizedError,
7
+ ForbiddenError,
8
+ HttpError
9
+ } from '@superblocksteam/shared';
10
+
11
+ // Re-export all error types from @superblocksteam/shared for convenience
12
+ export { BadRequestError, InternalServerError, NotFoundError, ConflictError, UnauthorizedError, ForbiddenError, HttpError };
13
+
14
+ /**
15
+ * Internal factory function to create appropriate error types based on HTTP status codes
16
+ * @internal
17
+ */
18
+ export function createErrorFromStatusCode(statusCode: number, message: string): Error {
19
+ const baseMessage = `SABS API Error (${statusCode}): ${message}`;
20
+
21
+ switch (statusCode) {
22
+ case 400:
23
+ return new BadRequestError(baseMessage);
24
+ case 401:
25
+ return new UnauthorizedError(baseMessage);
26
+ case 403:
27
+ return new ForbiddenError(baseMessage);
28
+ case 404:
29
+ return new NotFoundError(baseMessage);
30
+ case 409:
31
+ return new ConflictError(baseMessage);
32
+ case 500:
33
+ case 502:
34
+ case 503:
35
+ case 504:
36
+ return new InternalServerError(baseMessage);
37
+ default:
38
+ // For any other status codes, use the generic HttpError
39
+ return new HttpError(statusCode, baseMessage);
40
+ }
41
+ }
42
+
43
+ /**
44
+ * Internal function to create a client-side error (treated as BadRequestError with 400 status)
45
+ * @internal
46
+ */
47
+ export function createClientError(message: string): BadRequestError {
48
+ return new BadRequestError(`SABS Client Error: ${message}`);
49
+ }
50
+
51
+ /**
52
+ * Internal function to create a network/unknown error (treated as InternalServerError with 500 status)
53
+ * @internal
54
+ */
55
+ export function createNetworkError(message: string, originalError?: Error): InternalServerError {
56
+ const errorMessage = originalError
57
+ ? `SABS Network Error: ${message} (caused by: ${originalError.message})`
58
+ : `SABS Network Error: ${message}`;
59
+ return new InternalServerError(errorMessage);
60
+ }
package/src/index.ts ADDED
@@ -0,0 +1,2 @@
1
+ export * from './sabs';
2
+ export { BadRequestError, InternalServerError, NotFoundError, ConflictError, UnauthorizedError, ForbiddenError, HttpError } from './errors';