@danceroutine/tango-resources 1.11.2 → 1.11.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.
Files changed (49) hide show
  1. package/README.md +2 -2
  2. package/dist/CursorPaginator-B_8MhYZY.js +195 -0
  3. package/dist/CursorPaginator-B_8MhYZY.js.map +1 -0
  4. package/dist/CursorPaginator-CfeMQCdJ.d.ts +177 -0
  5. package/dist/OffsetPaginator-CaycvxJU.js +188 -0
  6. package/dist/OffsetPaginator-CaycvxJU.js.map +1 -0
  7. package/dist/context/index.d.ts +2 -0
  8. package/dist/context/index.js +2 -0
  9. package/dist/context-euBQvNRT.js +65 -0
  10. package/dist/context-euBQvNRT.js.map +1 -0
  11. package/dist/filters/index.d.ts +2 -0
  12. package/dist/filters/index.js +2 -0
  13. package/dist/filters-46d2Nr5C.js +287 -0
  14. package/dist/filters-46d2Nr5C.js.map +1 -0
  15. package/dist/index-Ac94YL5S.d.ts +24 -0
  16. package/dist/index-BJJalUDB.d.ts +54 -0
  17. package/dist/index-Bg6TtnmQ.d.ts +175 -0
  18. package/dist/index-C55GDIOn.d.ts +222 -0
  19. package/dist/index-CiIB-1Ac.d.ts +123 -0
  20. package/dist/index-DkJtxvKu.d.ts +164 -0
  21. package/dist/index.d.ts +10 -12
  22. package/dist/index.js +10 -1013
  23. package/dist/inferModelFieldParsers-2irv7j1T.js +70 -0
  24. package/dist/inferModelFieldParsers-2irv7j1T.js.map +1 -0
  25. package/dist/pagination/index.d.ts +3 -0
  26. package/dist/pagination/index.js +15 -0
  27. package/dist/pagination/index.js.map +1 -0
  28. package/dist/paginators/index.d.ts +9 -0
  29. package/dist/paginators/index.js +12 -0
  30. package/dist/paginators/index.js.map +1 -0
  31. package/dist/resource/index.d.ts +3 -0
  32. package/dist/resource/index.js +7 -0
  33. package/dist/resource/index.js.map +1 -0
  34. package/dist/serializer/index.d.ts +2 -0
  35. package/dist/serializer/index.js +2 -0
  36. package/dist/serializer-RSwlXWls.js +305 -0
  37. package/dist/serializer-RSwlXWls.js.map +1 -0
  38. package/dist/view/index.d.ts +2 -1
  39. package/dist/view/index.js +1 -1
  40. package/dist/{view-C9B5Lln3.js → view-CYdJAO4t.js} +7 -314
  41. package/dist/view-CYdJAO4t.js.map +1 -0
  42. package/dist/viewset/index.d.ts +9 -0
  43. package/dist/viewset/index.js +2 -0
  44. package/dist/viewset-C9j-2U29.js +227 -0
  45. package/dist/viewset-C9j-2U29.js.map +1 -0
  46. package/package.json +21 -17
  47. package/dist/index-D6sfTSEj.d.ts +0 -902
  48. package/dist/index.js.map +0 -1
  49. package/dist/view-C9B5Lln3.js.map +0 -1
@@ -1,316 +1,9 @@
1
1
  import { t as __exportAll } from "./chunk-D7D4PA-g.js";
2
- import { HttpErrorFactory, NotFoundError, TangoRequest, TangoResponse } from "@danceroutine/tango-core";
3
- import { Q, QueryResult } from "@danceroutine/tango-orm";
4
- import { z } from "zod";
5
- //#region src/context/RequestContext.ts
6
- /**
7
- * Normalized request context passed through the framework adapter into viewset methods.
8
- * Generic over the user type so consumers can plug in their own auth infrastructure.
9
- */
10
- var RequestContext = class RequestContext {
11
- request;
12
- user;
13
- params;
14
- static BRAND = "tango.resources.request_context";
15
- __tangoBrand = RequestContext.BRAND;
16
- state = /* @__PURE__ */ new Map();
17
- constructor(request, user = null, params = {}) {
18
- this.request = request;
19
- this.user = user;
20
- this.params = params;
21
- }
22
- /**
23
- * Narrow an unknown value to `RequestContext`.
24
- */
25
- static isRequestContext(value) {
26
- return typeof value === "object" && value !== null && value.__tangoBrand === RequestContext.BRAND;
27
- }
28
- /**
29
- * Construct a context with optional user payload.
30
- */
31
- static create(request, user) {
32
- return new RequestContext(TangoRequest.isTangoRequest(request) ? request : new TangoRequest(request), user ?? null);
33
- }
34
- /**
35
- * Store arbitrary per-request state for downstream middleware/handlers.
36
- */
37
- setState(key, value) {
38
- this.state.set(key, value);
39
- }
40
- /**
41
- * Retrieve previously stored request state.
42
- */
43
- getState(key) {
44
- return this.state.get(key);
45
- }
46
- /**
47
- * Check whether a state key has been set.
48
- */
49
- hasState(key) {
50
- return this.state.has(key);
51
- }
52
- /**
53
- * Clone the context, including route params and request-local state.
54
- */
55
- clone() {
56
- const cloned = new RequestContext(this.request, this.user, { ...this.params });
57
- cloned.state = new Map(this.state);
58
- return cloned;
59
- }
60
- };
61
- //#endregion
62
- //#region src/context/index.ts
63
- var context_exports = /* @__PURE__ */ __exportAll({ RequestContext: () => RequestContext });
64
- //#endregion
65
- //#region src/pagination/BasePaginator.ts
66
- var BasePaginator = class {
67
- resolveQueryResultRows(rows) {
68
- if (QueryResult.isQueryResult(rows)) return rows.toArray();
69
- return [...rows];
70
- }
71
- };
72
- //#endregion
73
- //#region src/pagination/OffsetPaginationInput.ts
74
- const OffsetPaginationInput = z.object({
75
- limit: z.coerce.number().int().min(1).default(25).transform((value) => Math.min(value, 100)),
76
- offset: z.coerce.number().int().min(0).default(0),
77
- page: z.coerce.number().int().min(1).optional()
78
- });
79
- //#endregion
80
- //#region src/paginators/OffsetPaginator.ts
81
- var OffsetPage = class OffsetPage {
82
- results;
83
- pageNumber;
84
- perPage;
85
- totalCount;
86
- static BRAND = "tango.resources.offset_page";
87
- __tangoBrand = OffsetPage.BRAND;
88
- constructor(results, pageNumber, perPage, totalCount) {
89
- this.results = results;
90
- this.pageNumber = pageNumber;
91
- this.perPage = perPage;
92
- this.totalCount = totalCount;
93
- }
94
- static isOffsetPage(value) {
95
- return typeof value === "object" && value !== null && value.__tangoBrand === OffsetPage.BRAND;
96
- }
97
- /** Whether a next page exists based on known total count. */
98
- hasNext() {
99
- if (this.totalCount === void 0) return false;
100
- return this.endIndex() < this.totalCount;
101
- }
102
- /** Whether a previous page exists. */
103
- hasPrevious() {
104
- return this.pageNumber > 1;
105
- }
106
- /** The next page number, if available. */
107
- nextPageNumber() {
108
- return this.hasNext() ? this.pageNumber + 1 : null;
109
- }
110
- /** The previous page number, if available. */
111
- previousPageNumber() {
112
- return this.hasPrevious() ? this.pageNumber - 1 : null;
113
- }
114
- /** Zero-based start index of this page in the full result set. */
115
- startIndex() {
116
- return (this.pageNumber - 1) * this.perPage;
117
- }
118
- /** Exclusive end index of this page in the full result set. */
119
- endIndex() {
120
- return this.startIndex() + this.results.length;
121
- }
122
- };
123
- /**
124
- * Offset/limit paginator modelled after DRF's LimitOffsetPagination.
125
- * Handles parsing limit/offset/page from URL query params and building
126
- * the paginated response envelope with next/previous links.
127
- *
128
- * @example
129
- * ```typescript
130
- * const paginator = new OffsetPaginator(queryset);
131
- * const { limit, offset } = paginator.parseParams(searchParams);
132
- * const results = await queryset.limit(limit).offset(offset).fetchAll();
133
- * const response = paginator.getPaginatedResponse(results, totalCount);
134
- * ```
135
- */
136
- var OffsetPaginator = class OffsetPaginator extends BasePaginator {
137
- queryset;
138
- perPage;
139
- static BRAND = "tango.resources.offset_paginator";
140
- __tangoBrand = OffsetPaginator.BRAND;
141
- limit = 25;
142
- offset = 0;
143
- constructor(queryset, perPage = 25) {
144
- super();
145
- this.queryset = queryset;
146
- this.perPage = perPage;
147
- this.limit = perPage;
148
- }
149
- /**
150
- * Narrow an unknown value to `OffsetPaginator`.
151
- */
152
- static isOffsetPaginator(value) {
153
- return typeof value === "object" && value !== null && value.__tangoBrand === OffsetPaginator.BRAND;
154
- }
155
- /**
156
- * Parse limit, offset, and page from Tango query params.
157
- * If `page` is provided, it's converted to an offset.
158
- * Stores parsed values for use by getPaginatedResponse.
159
- */
160
- parse(params) {
161
- const input = {
162
- limit: params.get("limit") ?? void 0,
163
- offset: params.get("offset") ?? void 0,
164
- page: params.get("page") ?? void 0
165
- };
166
- const parsed = OffsetPaginationInput.parse(input);
167
- if (parsed.page) parsed.offset = (parsed.page - 1) * parsed.limit;
168
- this.limit = parsed.limit;
169
- this.offset = parsed.offset;
170
- }
171
- /**
172
- * Parse params and return `{ limit, offset }` for compatibility callers.
173
- */
174
- parseParams(params) {
175
- this.parse(params);
176
- return {
177
- limit: this.limit,
178
- offset: this.offset
179
- };
180
- }
181
- /**
182
- * Build a DRF-style paginated response with count, next, and previous links.
183
- * Uses the limit/offset stored from the most recent parseParams call.
184
- */
185
- needsTotalCount() {
186
- return true;
187
- }
188
- toResponse(results, context) {
189
- const totalCount = context?.totalCount;
190
- const response = { results: this.resolveQueryResultRows(results) };
191
- if (totalCount !== void 0) {
192
- response.count = totalCount;
193
- if (this.offset + this.limit < totalCount) response.next = this.buildPageLink(this.offset + this.limit, context?.params);
194
- if (this.offset > 0) {
195
- const prevOffset = Math.max(0, this.offset - this.limit);
196
- response.previous = this.buildPageLink(prevOffset, context?.params);
197
- }
198
- }
199
- return response;
200
- }
201
- /**
202
- * Backward-compatible alias for `toResponse`.
203
- */
204
- getPaginatedResponse(results, totalCount, params) {
205
- return this.toResponse(results, {
206
- totalCount,
207
- params
208
- });
209
- }
210
- /**
211
- * Apply current limit/offset to a queryset.
212
- */
213
- apply(queryset) {
214
- return queryset.limit(this.limit).offset(this.offset);
215
- }
216
- /**
217
- * Fetch a 1-based page number from the bound queryset.
218
- */
219
- async paginate(page) {
220
- return this.getPage(page);
221
- }
222
- /**
223
- * Fetch a 1-based page and return page metadata.
224
- */
225
- async getPage(page) {
226
- const offset = (page - 1) * this.perPage;
227
- const results = await this.queryset.offset(offset).limit(this.perPage).fetch();
228
- const totalCount = await this.count();
229
- return new OffsetPage(this.resolveQueryResultRows(results), page, this.perPage, totalCount);
230
- }
231
- /**
232
- * Count total rows for the current queryset state.
233
- */
234
- async count() {
235
- return this.queryset.count();
236
- }
237
- buildPageLink(offset, params) {
238
- if (!params) return `?limit=${this.limit}&offset=${offset}`;
239
- return params.withValues({
240
- limit: this.limit,
241
- offset,
242
- page: null
243
- }).toRelativeURL();
244
- }
245
- };
246
- //#endregion
247
- //#region src/filters/inferModelFieldParsers.ts
248
- function normalizeParserTokens(raw) {
249
- const normalized = (Array.isArray(raw) ? raw : String(raw).split(",")).map((value) => value.trim());
250
- return normalized.every((value) => value.length > 0) ? normalized : [];
251
- }
252
- function createBooleanParser() {
253
- return (raw) => {
254
- const values = normalizeParserTokens(raw);
255
- if (values.length === 0) return;
256
- const parsed = values.map((value) => {
257
- const normalized = value.toLowerCase();
258
- if (normalized === "true" || normalized === "1") return true;
259
- if (normalized === "false" || normalized === "0") return false;
260
- return null;
261
- });
262
- if (parsed.some((value) => value === null)) return;
263
- return parsed.length === 1 ? parsed[0] : parsed;
264
- };
265
- }
266
- function createIntegerParser() {
267
- return (raw) => {
268
- const values = normalizeParserTokens(raw);
269
- if (values.length === 0) return;
270
- const parsed = values.map(Number);
271
- if (parsed.some((value) => !Number.isInteger(value))) return;
272
- return parsed.length === 1 ? parsed[0] : parsed;
273
- };
274
- }
275
- function createTimestampParser() {
276
- return (raw) => {
277
- const values = normalizeParserTokens(raw);
278
- if (values.length === 0) return;
279
- const parsed = values.map((value) => {
280
- const date = new Date(value);
281
- return Number.isNaN(date.getTime()) ? null : date;
282
- });
283
- if (parsed.some((value) => value === null)) return;
284
- return parsed.length === 1 ? parsed[0] : parsed;
285
- };
286
- }
287
- /**
288
- * Infer resource-level query-value parsers from Tango model metadata.
289
- *
290
- * Parsers are inferred conservatively from field metadata so HTTP query filters
291
- * can be coerced into typed ORM inputs without framework-specific glue.
292
- */
293
- function inferModelFieldParsers(model) {
294
- const metadata = model.metadata;
295
- if (!metadata) return {};
296
- const parsers = {};
297
- for (const field of metadata.fields) switch (field.type) {
298
- case "bool":
299
- parsers[field.name] = createBooleanParser();
300
- break;
301
- case "serial":
302
- case "int":
303
- case "bigint":
304
- parsers[field.name] = createIntegerParser();
305
- break;
306
- case "timestamptz":
307
- parsers[field.name] = createTimestampParser();
308
- break;
309
- default: break;
310
- }
311
- return parsers;
312
- }
313
- //#endregion
2
+ import "./context-euBQvNRT.js";
3
+ import { t as OffsetPaginator } from "./OffsetPaginator-CaycvxJU.js";
4
+ import { t as inferModelFieldParsers } from "./inferModelFieldParsers-2irv7j1T.js";
5
+ import { HttpErrorFactory, NotFoundError, TangoResponse } from "@danceroutine/tango-core";
6
+ import { Q } from "@danceroutine/tango-orm";
314
7
  //#region src/view/APIView.ts
315
8
  /**
316
9
  * Lightweight class-based request dispatcher for non-model API endpoints.
@@ -725,6 +418,6 @@ var view_exports = /* @__PURE__ */ __exportAll({
725
418
  UpdateModelMixin: () => UpdateModelMixin
726
419
  });
727
420
  //#endregion
728
- export { OffsetPaginator as _, ListCreateAPIView as a, context_exports as b, ListAPIView as c, RetrieveModelMixin as d, CreateModelMixin as f, inferModelFieldParsers as g, APIView as h, RetrieveUpdateAPIView as i, DestroyModelMixin as l, GenericAPIView as m, RetrieveUpdateDestroyAPIView as n, RetrieveAPIView as o, ListModelMixin as p, RetrieveDestroyAPIView as r, CreateAPIView as s, view_exports as t, UpdateModelMixin as u, OffsetPaginationInput as v, RequestContext as x, BasePaginator as y };
421
+ export { ListCreateAPIView as a, ListAPIView as c, RetrieveModelMixin as d, CreateModelMixin as f, APIView as h, RetrieveUpdateAPIView as i, DestroyModelMixin as l, GenericAPIView as m, RetrieveUpdateDestroyAPIView as n, RetrieveAPIView as o, ListModelMixin as p, RetrieveDestroyAPIView as r, CreateAPIView as s, view_exports as t, UpdateModelMixin as u };
729
422
 
730
- //# sourceMappingURL=view-C9B5Lln3.js.map
423
+ //# sourceMappingURL=view-CYdJAO4t.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"view-CYdJAO4t.js","names":[],"sources":["../src/view/APIView.ts","../src/view/GenericAPIView.ts","../src/view/mixins/ListModelMixin.ts","../src/view/mixins/CreateModelMixin.ts","../src/view/mixins/RetrieveModelMixin.ts","../src/view/mixins/UpdateModelMixin.ts","../src/view/mixins/DestroyModelMixin.ts","../src/view/generics/ListAPIView.ts","../src/view/generics/CreateAPIView.ts","../src/view/generics/RetrieveAPIView.ts","../src/view/generics/ListCreateAPIView.ts","../src/view/generics/RetrieveUpdateAPIView.ts","../src/view/generics/RetrieveDestroyAPIView.ts","../src/view/generics/RetrieveUpdateDestroyAPIView.ts","../src/view/index.ts"],"sourcesContent":["import { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../context/index';\n\nexport type APIViewMethod = 'GET' | 'POST' | 'PUT' | 'PATCH' | 'DELETE';\n\ntype APIViewMethodHandler = (ctx: RequestContext) => Promise<TangoResponse>;\n\n/**\n * Lightweight class-based request dispatcher for non-model API endpoints.\n */\nexport abstract class APIView {\n static readonly BRAND = 'tango.resources.api_view' as const;\n readonly __tangoBrand: typeof APIView.BRAND = APIView.BRAND;\n\n /**\n * Narrow an unknown value to `APIView`.\n */\n static isAPIView(value: unknown): value is APIView {\n return (\n typeof value === 'object' &&\n value !== null &&\n (value as { __tangoBrand?: unknown }).__tangoBrand === APIView.BRAND\n );\n }\n\n /**\n * Dispatch the request to the handler for the current HTTP method.\n */\n async dispatch(ctx: RequestContext): Promise<TangoResponse> {\n const method = normalizeMethod(ctx.request.method);\n if (!method) {\n return this.httpMethodNotAllowed();\n }\n\n const handler = this.getMethodHandler(method);\n return handler(ctx);\n }\n\n getAllowedMethods(): readonly APIViewMethod[] {\n const allowed: APIViewMethod[] = [];\n if (this.get !== APIView.prototype.get) {\n allowed.push('GET');\n }\n if (this.post !== APIView.prototype.post) {\n allowed.push('POST');\n }\n if (this.put !== APIView.prototype.put) {\n allowed.push('PUT');\n }\n if (this.patch !== APIView.prototype.patch) {\n allowed.push('PATCH');\n }\n if (this.delete !== APIView.prototype.delete) {\n allowed.push('DELETE');\n }\n return allowed;\n }\n\n protected get(_ctx: RequestContext): Promise<TangoResponse> {\n return Promise.resolve(this.httpMethodNotAllowed());\n }\n\n protected post(_ctx: RequestContext): Promise<TangoResponse> {\n return Promise.resolve(this.httpMethodNotAllowed());\n }\n\n protected put(_ctx: RequestContext): Promise<TangoResponse> {\n return Promise.resolve(this.httpMethodNotAllowed());\n }\n\n protected patch(_ctx: RequestContext): Promise<TangoResponse> {\n return Promise.resolve(this.httpMethodNotAllowed());\n }\n\n protected delete(_ctx: RequestContext): Promise<TangoResponse> {\n return Promise.resolve(this.httpMethodNotAllowed());\n }\n\n protected httpMethodNotAllowed(): TangoResponse {\n return TangoResponse.methodNotAllowed(this.getAllowedMethods());\n }\n\n private getMethodHandler(method: APIViewMethod): APIViewMethodHandler {\n if (method === 'GET') {\n return (ctx) => this.get(ctx);\n }\n if (method === 'POST') {\n return (ctx) => this.post(ctx);\n }\n if (method === 'PUT') {\n return (ctx) => this.put(ctx);\n }\n if (method === 'PATCH') {\n return (ctx) => this.patch(ctx);\n }\n return (ctx) => this.delete(ctx);\n }\n}\n\nfunction normalizeMethod(method: string): APIViewMethod | null {\n const upper = method.toUpperCase();\n if (upper === 'GET') {\n return 'GET';\n }\n if (upper === 'POST') {\n return 'POST';\n }\n if (upper === 'PUT') {\n return 'PUT';\n }\n if (upper === 'PATCH') {\n return 'PATCH';\n }\n if (upper === 'DELETE') {\n return 'DELETE';\n }\n return null;\n}\n","import { HttpErrorFactory, TangoResponse, type JsonValue, NotFoundError } from '@danceroutine/tango-core';\nimport { Q, type FilterInput, type ManagerLike, type QuerySet } from '@danceroutine/tango-orm';\nimport type { OffsetPaginatedResponse, Paginator } from '../pagination/index';\nimport { OffsetPaginator } from '../paginators/OffsetPaginator';\nimport { APIView } from './APIView';\nimport { RequestContext } from '../context/index';\nimport type { FilterSet } from '../filters/index';\nimport { inferModelFieldParsers } from '../filters/inferModelFieldParsers';\nimport type { GenericAPIViewOpenAPIDescription } from '../resource/index';\nimport type { AnyModelSerializer, SerializerOutput } from '../serializer/index';\nimport type { ResourceModelLike } from '../resource/index';\n\ntype SearchFieldRef<TModel extends Record<string, unknown>> = Extract<keyof TModel, string> | string;\n\nexport interface GenericAPIViewConfig<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> {\n serializer: TSerializer;\n filters?: FilterSet<TModel>;\n orderingFields?: (keyof TModel)[];\n searchFields?: SearchFieldRef<TModel>[];\n lookupField?: keyof TModel;\n lookupParam?: string;\n paginatorFactory?: (queryset: QuerySet<TModel>) => Paginator<TModel, SerializerOutput<TSerializer>>;\n}\n\n/**\n * Generic API base class that centralizes query/build/validation helpers.\n */\nexport abstract class GenericAPIView<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends APIView {\n protected readonly serializerClass: TSerializer;\n protected readonly filters?: FilterSet<TModel>;\n protected readonly orderingFields: readonly (keyof TModel)[];\n protected readonly searchFields: readonly SearchFieldRef<TModel>[];\n protected readonly lookupField?: keyof TModel;\n protected readonly lookupParam: string;\n protected readonly paginatorFactory?: (\n queryset: QuerySet<TModel>\n ) => Paginator<TModel, SerializerOutput<TSerializer>>;\n private serializer?: InstanceType<TSerializer>;\n\n constructor(config: GenericAPIViewConfig<TModel, TSerializer>) {\n super();\n this.serializerClass = config.serializer;\n this.filters = config.filters;\n this.orderingFields = config.orderingFields ?? [];\n this.searchFields = config.searchFields ?? [];\n this.lookupField = config.lookupField;\n this.lookupParam = config.lookupParam ?? 'id';\n this.paginatorFactory = config.paginatorFactory;\n }\n\n /**\n * Return the serializer class that owns this resource contract.\n */\n getSerializerClass(): TSerializer {\n return this.serializerClass;\n }\n\n /**\n * Return the serializer instance for the current resource.\n */\n getSerializer(): InstanceType<TSerializer> {\n if (!this.serializer) {\n this.serializer = new this.serializerClass() as InstanceType<TSerializer>;\n }\n\n return this.serializer;\n }\n\n /**\n * Describe the public HTTP contract that this resource contributes to OpenAPI generation.\n */\n describeOpenAPI(): GenericAPIViewOpenAPIDescription<TModel, TSerializer> {\n const model = this.requireModelMetadata();\n return {\n model,\n outputSchema: this.getOutputSchema(),\n createSchema: this.getCreateSchema(),\n updateSchema: this.getUpdateSchema(),\n searchFields: this.searchFields,\n orderingFields: this.orderingFields,\n lookupField: this.lookupField ?? this.getLookupFieldFromMetadata(model),\n lookupParam: this.lookupParam,\n allowedMethods: this.getAllowedMethods(),\n usesDefaultOffsetPagination: !this.paginatorFactory,\n };\n }\n\n protected getManager(): ManagerLike<TModel> {\n return this.getSerializer().getManager();\n }\n\n protected getOutputSchema(): TSerializer['outputSchema'] {\n return this.getSerializer().getOutputSchema() as TSerializer['outputSchema'];\n }\n\n protected getCreateSchema(): TSerializer['createSchema'] {\n return this.getSerializer().getCreateSchema() as TSerializer['createSchema'];\n }\n\n protected getUpdateSchema(): TSerializer['updateSchema'] {\n return this.getSerializer().getUpdateSchema() as TSerializer['updateSchema'];\n }\n\n protected getLookupField(): keyof TModel {\n return this.lookupField ?? (this.getManager().meta.pk as keyof TModel);\n }\n\n protected getLookupValue(ctx: RequestContext): string | null {\n const value = ctx.params[this.lookupParam]?.trim();\n return value || null;\n }\n\n protected getPaginator(queryset: QuerySet<TModel>): Paginator<TModel, SerializerOutput<TSerializer>> {\n if (this.paginatorFactory) {\n return this.paginatorFactory(queryset);\n }\n return new OffsetPaginator<TModel>(queryset) as Paginator<TModel, SerializerOutput<TSerializer>>;\n }\n\n protected async performList(ctx: RequestContext): Promise<TangoResponse> {\n try {\n const params = ctx.request.queryParams;\n const baseQueryset = this.getManager().query();\n const paginator = this.getPaginator(baseQueryset);\n paginator.parse(params);\n\n let qs = baseQueryset;\n\n if (this.filters) {\n const filterInputs = this.filters\n .withFieldParsers(inferModelFieldParsers(this.getSerializer().getModel()))\n .apply(params);\n if (filterInputs.length > 0) {\n qs = qs.filter(Q.and(...filterInputs));\n }\n }\n\n const search = params.getSearch();\n if (search && this.searchFields.length > 0) {\n const searchFilters: FilterInput<TModel>[] = this.searchFields.map((field) => {\n const lookup = `${String(field)}__icontains`;\n return { [lookup]: search } as FilterInput<TModel>;\n });\n qs = qs.filter(Q.or(...searchFilters));\n }\n\n const ordering = params.getOrdering();\n if (ordering.length > 0) {\n const orderTokens = ordering.filter((field) => {\n const cleanField = field.startsWith('-') ? field.slice(1) : field;\n return this.orderingFields.includes(cleanField as keyof TModel);\n });\n if (orderTokens.length > 0) {\n qs = qs.orderBy(...orderTokens.map((token) => token as keyof TModel | `-${string & keyof TModel}`));\n }\n }\n\n const paginatedQueryset = paginator.apply(qs);\n const resultPromise = paginatedQueryset.fetch();\n const totalCountPromise = paginator.needsTotalCount()\n ? qs.count()\n : Promise.resolve<number | undefined>(undefined);\n const [result, totalCount] = await Promise.all([resultPromise, totalCountPromise]);\n const serializer = this.getSerializer();\n const rows = await serializer.serializeMany(result.items);\n const response = paginator.toResponse(rows as SerializerOutput<TSerializer>[], {\n totalCount,\n }) as OffsetPaginatedResponse<SerializerOutput<TSerializer>>;\n\n return TangoResponse.json(response as unknown as JsonValue, { status: 200 });\n } catch (error) {\n return this.handleError(error);\n }\n }\n\n protected async performCreate(ctx: RequestContext): Promise<TangoResponse> {\n try {\n const body = await ctx.request.json();\n const result = await this.getSerializer().create(body);\n\n return TangoResponse.created(undefined, result as JsonValue);\n } catch (error) {\n return this.handleError(error);\n }\n }\n\n protected async performRetrieve(ctx: RequestContext): Promise<TangoResponse> {\n try {\n const value = this.getLookupValue(ctx);\n if (!value) {\n throw new NotFoundError('Lookup parameter was not provided.');\n }\n\n const lookupField = this.getLookupField();\n const filterByLookup = { [lookupField]: value } as FilterInput<TModel>;\n const result = await this.getManager().query().filter(filterByLookup).fetchOne();\n if (!result) {\n throw new NotFoundError(\n `No ${this.getManager().meta.table} record found for ${String(lookupField)}=${value}.`\n );\n }\n\n return TangoResponse.json((await this.getSerializer().serialize(result)) as JsonValue, { status: 200 });\n } catch (error) {\n return this.handleError(error);\n }\n }\n\n protected async performUpdate(ctx: RequestContext): Promise<TangoResponse> {\n try {\n const value = this.getLookupValue(ctx);\n if (!value) {\n throw new NotFoundError('Lookup parameter was not provided.');\n }\n\n const body = await ctx.request.json();\n const result = await this.getSerializer().update(value as TModel[keyof TModel], body);\n\n return TangoResponse.json(result as JsonValue, { status: 200 });\n } catch (error) {\n return this.handleError(error);\n }\n }\n\n protected async performDestroy(ctx: RequestContext): Promise<TangoResponse> {\n try {\n const value = this.getLookupValue(ctx);\n if (!value) {\n throw new NotFoundError('Lookup parameter was not provided.');\n }\n\n await this.getManager().delete(value as TModel[keyof TModel]);\n return TangoResponse.noContent();\n } catch (error) {\n return this.handleError(error);\n }\n }\n\n protected handleError(error: unknown): TangoResponse {\n const httpError = HttpErrorFactory.toHttpError(error);\n return TangoResponse.json(httpError.body as JsonValue, { status: httpError.status });\n }\n\n private requireModelMetadata(): ResourceModelLike<TModel> & {\n metadata: NonNullable<ResourceModelLike<TModel>['metadata']>;\n } {\n const model = this.getSerializer().getModel();\n\n if (!model.metadata) {\n throw new Error('OpenAPI generation requires Tango model metadata on GenericAPIView models.');\n }\n\n return model as ResourceModelLike<TModel> & {\n metadata: NonNullable<ResourceModelLike<TModel>['metadata']>;\n };\n }\n\n private getLookupFieldFromMetadata(\n model: ResourceModelLike<TModel> & {\n metadata: NonNullable<ResourceModelLike<TModel>['metadata']>;\n }\n ): keyof TModel {\n const primaryKeyField = model.metadata.fields.find((field) => field.primaryKey);\n\n if (!primaryKeyField) {\n throw new Error('OpenAPI generation requires a primary key field in Tango model metadata.');\n }\n\n return primaryKeyField.name as keyof TModel;\n }\n}\n","import { GenericAPIView } from '../GenericAPIView';\nimport type { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../../context/index';\nimport type { AnyModelSerializer } from '../../serializer/index';\n\n/**\n * Mixin that wires `GET` requests to the generic list implementation.\n */\nexport abstract class ListModelMixin<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends GenericAPIView<TModel, TSerializer> {\n protected list(ctx: RequestContext): Promise<TangoResponse> {\n return this.performList(ctx);\n }\n\n protected override get(ctx: RequestContext): Promise<TangoResponse> {\n return this.list(ctx);\n }\n}\n","import { GenericAPIView } from '../GenericAPIView';\nimport type { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../../context/index';\nimport type { AnyModelSerializer } from '../../serializer/index';\n\n/**\n * Mixin that wires `POST` requests to the generic create implementation.\n */\nexport abstract class CreateModelMixin<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends GenericAPIView<TModel, TSerializer> {\n protected create(ctx: RequestContext): Promise<TangoResponse> {\n return this.performCreate(ctx);\n }\n\n protected override post(ctx: RequestContext): Promise<TangoResponse> {\n return this.create(ctx);\n }\n}\n","import { GenericAPIView } from '../GenericAPIView';\nimport type { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../../context/index';\nimport type { AnyModelSerializer } from '../../serializer/index';\n\n/**\n * Mixin that wires `GET` requests to the generic retrieve implementation.\n */\nexport abstract class RetrieveModelMixin<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends GenericAPIView<TModel, TSerializer> {\n protected retrieve(ctx: RequestContext): Promise<TangoResponse> {\n return this.performRetrieve(ctx);\n }\n\n protected override get(ctx: RequestContext): Promise<TangoResponse> {\n return this.retrieve(ctx);\n }\n}\n","import { GenericAPIView } from '../GenericAPIView';\nimport type { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../../context/index';\nimport type { AnyModelSerializer } from '../../serializer/index';\n\n/**\n * Mixin that wires `PUT` and `PATCH` requests to the generic update implementation.\n */\nexport abstract class UpdateModelMixin<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends GenericAPIView<TModel, TSerializer> {\n protected update(ctx: RequestContext): Promise<TangoResponse> {\n return this.performUpdate(ctx);\n }\n\n protected override put(ctx: RequestContext): Promise<TangoResponse> {\n return this.update(ctx);\n }\n\n protected override patch(ctx: RequestContext): Promise<TangoResponse> {\n return this.update(ctx);\n }\n}\n","import { GenericAPIView } from '../GenericAPIView';\nimport type { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../../context/index';\nimport type { AnyModelSerializer } from '../../serializer/index';\n\n/**\n * Mixin that wires `DELETE` requests to the generic destroy implementation.\n */\nexport abstract class DestroyModelMixin<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends GenericAPIView<TModel, TSerializer> {\n protected destroy(ctx: RequestContext): Promise<TangoResponse> {\n return this.performDestroy(ctx);\n }\n\n protected override delete(ctx: RequestContext): Promise<TangoResponse> {\n return this.destroy(ctx);\n }\n}\n","import { ListModelMixin } from '../mixins/ListModelMixin';\nimport type { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../../context/index';\nimport type { AnyModelSerializer } from '../../serializer/index';\n\n/**\n * Generic API view for endpoints that only expose a list operation.\n */\nexport abstract class ListAPIView<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends ListModelMixin<TModel, TSerializer> {\n protected override get(ctx: RequestContext): Promise<TangoResponse> {\n return super.get(ctx);\n }\n}\n","import { CreateModelMixin } from '../mixins/CreateModelMixin';\nimport type { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../../context/index';\nimport type { AnyModelSerializer } from '../../serializer/index';\n\n/**\n * Generic API view for endpoints that only support resource creation.\n */\nexport abstract class CreateAPIView<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends CreateModelMixin<TModel, TSerializer> {\n protected override post(ctx: RequestContext): Promise<TangoResponse> {\n return super.post(ctx);\n }\n}\n","import { RetrieveModelMixin } from '../mixins/RetrieveModelMixin';\nimport type { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../../context/index';\nimport type { AnyModelSerializer } from '../../serializer/index';\n\n/**\n * Generic API view for endpoints that retrieve a single resource by lookup.\n */\nexport abstract class RetrieveAPIView<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends RetrieveModelMixin<TModel, TSerializer> {\n protected override get(ctx: RequestContext): Promise<TangoResponse> {\n return super.get(ctx);\n }\n}\n","import { GenericAPIView } from '../GenericAPIView';\nimport type { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../../context/index';\nimport type { AnyModelSerializer } from '../../serializer/index';\n\n/**\n * Generic API view for collection endpoints that list and create resources.\n */\nexport abstract class ListCreateAPIView<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends GenericAPIView<TModel, TSerializer> {\n protected override get(ctx: RequestContext): Promise<TangoResponse> {\n return this.performList(ctx);\n }\n\n protected override post(ctx: RequestContext): Promise<TangoResponse> {\n return this.performCreate(ctx);\n }\n}\n","import { GenericAPIView } from '../GenericAPIView';\nimport type { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../../context/index';\nimport type { AnyModelSerializer } from '../../serializer/index';\n\n/**\n * Generic API view for endpoints that retrieve and update a single resource.\n */\nexport abstract class RetrieveUpdateAPIView<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends GenericAPIView<TModel, TSerializer> {\n protected override get(ctx: RequestContext): Promise<TangoResponse> {\n return this.performRetrieve(ctx);\n }\n\n protected override put(ctx: RequestContext): Promise<TangoResponse> {\n return this.performUpdate(ctx);\n }\n\n protected override patch(ctx: RequestContext): Promise<TangoResponse> {\n return this.performUpdate(ctx);\n }\n}\n","import { GenericAPIView } from '../GenericAPIView';\nimport type { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../../context/index';\nimport type { AnyModelSerializer } from '../../serializer/index';\n\n/**\n * Generic API view for endpoints that retrieve and delete a single resource.\n */\nexport abstract class RetrieveDestroyAPIView<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends GenericAPIView<TModel, TSerializer> {\n protected override get(ctx: RequestContext): Promise<TangoResponse> {\n return this.performRetrieve(ctx);\n }\n\n protected override delete(ctx: RequestContext): Promise<TangoResponse> {\n return this.performDestroy(ctx);\n }\n}\n","import { GenericAPIView } from '../GenericAPIView';\nimport type { TangoResponse } from '@danceroutine/tango-core';\nimport { RequestContext } from '../../context/index';\nimport type { AnyModelSerializer } from '../../serializer/index';\n\n/**\n * Generic API view for full detail endpoints that retrieve, update, and delete.\n */\nexport abstract class RetrieveUpdateDestroyAPIView<\n TModel extends Record<string, unknown>,\n TSerializer extends AnyModelSerializer<TModel>,\n> extends GenericAPIView<TModel, TSerializer> {\n protected override get(ctx: RequestContext): Promise<TangoResponse> {\n return this.performRetrieve(ctx);\n }\n\n protected override put(ctx: RequestContext): Promise<TangoResponse> {\n return this.performUpdate(ctx);\n }\n\n protected override patch(ctx: RequestContext): Promise<TangoResponse> {\n return this.performUpdate(ctx);\n }\n\n protected override delete(ctx: RequestContext): Promise<TangoResponse> {\n return this.performDestroy(ctx);\n }\n}\n","/**\n * Domain boundary barrel: centralizes this subdomain's public contract.\n */\n\nexport { APIView, type APIViewMethod } from './APIView';\nexport { GenericAPIView, type GenericAPIViewConfig } from './GenericAPIView';\nexport type { GenericAPIViewOpenAPIDescription } from '../resource/index';\nexport {\n ListModelMixin,\n CreateModelMixin,\n RetrieveModelMixin,\n UpdateModelMixin,\n DestroyModelMixin,\n} from './mixins/index';\nexport {\n ListAPIView,\n CreateAPIView,\n RetrieveAPIView,\n ListCreateAPIView,\n RetrieveUpdateAPIView,\n RetrieveDestroyAPIView,\n RetrieveUpdateDestroyAPIView,\n} from './generics/index';\n"],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import { a as ModelViewSetConfig, c as ViewSetActionMethod, i as ModelViewSet, l as ViewSetActionScope, o as ResolvedViewSetActionDescriptor, r as ModelViewSetOpenAPIDescription, s as ViewSetActionDescriptor } from "../index-DkJtxvKu.js";
2
+
3
+ //#region src/viewset/index.d.ts
4
+ declare namespace index_d_exports {
5
+ export { ModelViewSet, ModelViewSetConfig, ModelViewSetOpenAPIDescription, ResolvedViewSetActionDescriptor, ViewSetActionDescriptor, ViewSetActionMethod, ViewSetActionScope };
6
+ }
7
+ //#endregion
8
+ export { ModelViewSet, type ModelViewSetConfig, type ModelViewSetOpenAPIDescription, type ResolvedViewSetActionDescriptor, type ViewSetActionDescriptor, type ViewSetActionMethod, type ViewSetActionScope, index_d_exports as t };
9
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1,2 @@
1
+ import { n as ModelViewSet } from "../viewset-C9j-2U29.js";
2
+ export { ModelViewSet };
@@ -0,0 +1,227 @@
1
+ import { t as __exportAll } from "./chunk-D7D4PA-g.js";
2
+ import { t as OffsetPaginator } from "./OffsetPaginator-CaycvxJU.js";
3
+ import { t as inferModelFieldParsers } from "./inferModelFieldParsers-2irv7j1T.js";
4
+ import { HttpErrorFactory, NotFoundError, TangoResponse } from "@danceroutine/tango-core";
5
+ import { Q } from "@danceroutine/tango-orm";
6
+ //#region src/viewset/ModelViewSet.ts
7
+ /**
8
+ * Base class for creating RESTful API viewsets with built-in CRUD operations.
9
+ * Provides list, retrieve, create, update, and delete methods with filtering,
10
+ * search, pagination, and ordering support.
11
+ */
12
+ var ModelViewSet = class ModelViewSet {
13
+ static BRAND = "tango.resources.model_view_set";
14
+ static actions = [];
15
+ __tangoBrand = ModelViewSet.BRAND;
16
+ serializerClass;
17
+ filters;
18
+ orderingFields;
19
+ searchFields;
20
+ paginatorFactory;
21
+ serializer;
22
+ constructor(config) {
23
+ this.serializerClass = config.serializer;
24
+ this.filters = config.filters;
25
+ this.orderingFields = config.orderingFields ?? [];
26
+ this.searchFields = config.searchFields ?? [];
27
+ this.paginatorFactory = config.paginatorFactory;
28
+ }
29
+ /**
30
+ * Return the custom action descriptors declared by a viewset or constructor.
31
+ */
32
+ static getActions(viewsetOrConstructor) {
33
+ const viewset = ModelViewSet.isModelViewSet(viewsetOrConstructor) ? viewsetOrConstructor : null;
34
+ const constructorValue = viewset ? viewset.constructor : viewsetOrConstructor;
35
+ return (Array.isArray(constructorValue.actions) ? constructorValue.actions : []).map((action) => ({
36
+ ...action,
37
+ path: viewset ? viewset.resolveActionPath(action) : ModelViewSet.resolvePathFromDescriptor(action.name, action.path)
38
+ }));
39
+ }
40
+ /**
41
+ * Narrow an unknown value to `ModelViewSet`.
42
+ */
43
+ static isModelViewSet(value) {
44
+ return typeof value === "object" && value !== null && value.__tangoBrand === ModelViewSet.BRAND;
45
+ }
46
+ /**
47
+ * Preserve literal action inference while validating the descriptor shape.
48
+ */
49
+ static defineViewSetActions(actions) {
50
+ return actions;
51
+ }
52
+ static resolvePathFromDescriptor(name, explicitPath) {
53
+ const normalized = (explicitPath?.trim() || ModelViewSet.toKebabCase(name)).replace(/^\/+|\/+$/g, "");
54
+ if (!normalized) throw new Error(`Invalid custom action path for '${name}'.`);
55
+ return normalized;
56
+ }
57
+ static toKebabCase(input) {
58
+ return input.replace(/([a-z0-9])([A-Z])/g, "$1-$2").replace(/[_\s]+/g, "-").toLowerCase();
59
+ }
60
+ /**
61
+ * Return the serializer class that owns this resource contract.
62
+ */
63
+ getSerializerClass() {
64
+ return this.serializerClass;
65
+ }
66
+ /**
67
+ * Return the serializer instance for the current resource.
68
+ */
69
+ getSerializer() {
70
+ if (!this.serializer) this.serializer = new this.serializerClass();
71
+ return this.serializer;
72
+ }
73
+ /**
74
+ * Describe the public HTTP contract that this resource contributes to OpenAPI generation.
75
+ */
76
+ describeOpenAPI() {
77
+ const model = this.requireModelMetadata();
78
+ return {
79
+ model,
80
+ outputSchema: this.getSerializer().getOutputSchema(),
81
+ createSchema: this.getSerializer().getCreateSchema(),
82
+ updateSchema: this.getSerializer().getUpdateSchema(),
83
+ searchFields: this.searchFields,
84
+ orderingFields: this.orderingFields,
85
+ lookupField: this.getLookupFieldFromMetadata(model),
86
+ lookupParam: "id",
87
+ allowedMethods: [
88
+ "GET",
89
+ "POST",
90
+ "PUT",
91
+ "PATCH",
92
+ "DELETE"
93
+ ],
94
+ usesDefaultOffsetPagination: !this.paginatorFactory,
95
+ actions: ModelViewSet.getActions(this)
96
+ };
97
+ }
98
+ /**
99
+ * List endpoint with filtering, search, ordering, and offset pagination.
100
+ */
101
+ async list(ctx) {
102
+ try {
103
+ const params = ctx.request.queryParams;
104
+ const baseQueryset = this.getManager().query();
105
+ const paginator = this.getPaginator(baseQueryset);
106
+ paginator.parse(params);
107
+ let qs = baseQueryset;
108
+ if (this.filters) {
109
+ const filterInputs = this.filters.withFieldParsers(inferModelFieldParsers(this.getSerializer().getModel())).apply(params);
110
+ if (filterInputs.length > 0) qs = qs.filter(Q.and(...filterInputs));
111
+ }
112
+ const search = params.getSearch();
113
+ if (search && this.searchFields.length > 0) {
114
+ const searchFilters = this.searchFields.map((field) => {
115
+ return { [`${String(field)}__icontains`]: search };
116
+ });
117
+ qs = qs.filter(Q.or(...searchFilters));
118
+ }
119
+ const ordering = params.getOrdering();
120
+ if (ordering.length > 0) {
121
+ const orderTokens = ordering.filter((field) => {
122
+ const cleanField = field.startsWith("-") ? field.slice(1) : field;
123
+ return this.orderingFields.includes(cleanField);
124
+ });
125
+ if (orderTokens.length > 0) qs = qs.orderBy(...orderTokens.map((token) => token));
126
+ }
127
+ const resultPromise = paginator.apply(qs).fetch();
128
+ const totalCountPromise = paginator.needsTotalCount() ? qs.count() : Promise.resolve(void 0);
129
+ const [result, totalCount] = await Promise.all([resultPromise, totalCountPromise]);
130
+ const rows = await this.getSerializer().serializeMany(result.items);
131
+ const response = paginator.toResponse(rows, { totalCount });
132
+ return TangoResponse.json(response, { status: 200 });
133
+ } catch (error) {
134
+ return this.handleError(error);
135
+ }
136
+ }
137
+ /**
138
+ * Retrieve endpoint for a single resource by id.
139
+ */
140
+ async retrieve(_ctx, id) {
141
+ try {
142
+ const manager = this.getManager();
143
+ const pk = manager.meta.pk;
144
+ const filterById = { [pk]: id };
145
+ const result = await manager.query().filter(filterById).fetchOne();
146
+ if (!result) throw new NotFoundError(`No ${manager.meta.table} record found for ${String(pk)}=${id}.`);
147
+ return TangoResponse.json(await this.getSerializer().serialize(result), { status: 200 });
148
+ } catch (error) {
149
+ return this.handleError(error);
150
+ }
151
+ }
152
+ /**
153
+ * Create endpoint: validate input, persist, and return serialized output.
154
+ */
155
+ async create(ctx) {
156
+ try {
157
+ const body = await ctx.request.json();
158
+ const result = await this.getSerializer().create(body);
159
+ return TangoResponse.created(void 0, result);
160
+ } catch (error) {
161
+ return this.handleError(error);
162
+ }
163
+ }
164
+ /**
165
+ * Update endpoint: validate partial payload and persist by id.
166
+ */
167
+ async update(ctx, id) {
168
+ try {
169
+ const body = await ctx.request.json();
170
+ const pkValue = id;
171
+ const result = await this.getSerializer().update(pkValue, body);
172
+ return TangoResponse.json(result, { status: 200 });
173
+ } catch (error) {
174
+ return this.handleError(error);
175
+ }
176
+ }
177
+ /**
178
+ * Destroy endpoint: delete a resource by id.
179
+ */
180
+ async destroy(_ctx, id) {
181
+ try {
182
+ const pkValue = id;
183
+ await this.getManager().delete(pkValue);
184
+ return TangoResponse.noContent();
185
+ } catch (error) {
186
+ return this.handleError(error);
187
+ }
188
+ }
189
+ getPaginator(queryset) {
190
+ if (this.paginatorFactory) return this.paginatorFactory(queryset);
191
+ return new OffsetPaginator(queryset);
192
+ }
193
+ getManager() {
194
+ return this.getSerializer().getManager();
195
+ }
196
+ /**
197
+ * Convert thrown errors into normalized HTTP responses.
198
+ */
199
+ handleError(error) {
200
+ const httpError = HttpErrorFactory.toHttpError(error);
201
+ return TangoResponse.json(httpError.body, { status: httpError.status });
202
+ }
203
+ /**
204
+ * Resolve route path segment(s) for a custom action.
205
+ * Override this in subclasses to customize path derivation globally.
206
+ */
207
+ resolveActionPath(action) {
208
+ return ModelViewSet.resolvePathFromDescriptor(action.name, action.path);
209
+ }
210
+ requireModelMetadata() {
211
+ const model = this.getSerializer().getModel();
212
+ if (!model.metadata) throw new Error("OpenAPI generation requires Tango model metadata on ModelViewSet models.");
213
+ return model;
214
+ }
215
+ getLookupFieldFromMetadata(model) {
216
+ const primaryKeyField = model.metadata.fields.find((field) => field.primaryKey);
217
+ if (!primaryKeyField) throw new Error("OpenAPI generation requires a primary key field in Tango model metadata.");
218
+ return primaryKeyField.name;
219
+ }
220
+ };
221
+ //#endregion
222
+ //#region src/viewset/index.ts
223
+ var viewset_exports = /* @__PURE__ */ __exportAll({ ModelViewSet: () => ModelViewSet });
224
+ //#endregion
225
+ export { ModelViewSet as n, viewset_exports as t };
226
+
227
+ //# sourceMappingURL=viewset-C9j-2U29.js.map