How to filter outgoing request body #1042
-
In our app (like most apps I'm sure) we need to pass Date values between the client and server. As there might be other special types of objects that need to be passed around, I've set up a system for serializing these kinds of classes safely, using an object with a special field (that wouldn't ever be used otherwise): {
"@type": "date",
"time": 124804183002
} Server to ClientFor server-to-client responses, the server filters its data and transforms dates into objects of this form. Then, when received by the client, I've overridden This works great! Client to ServerHowever, when going from client to server, I'd like to transform Date objects into objects of this form automatically as well. I have a function for doing this (it accepts any object, and deeply converts dates to this format). Now I'm looking to inject it into the Api class generated by As far as I can tell, the individual endpoint methods call That is, there doesn't appear to be a safe place to inject my function for filtering outgoing data (unless I override Our currrent build command looks something like, npx swagger-typescript-api -p ./api/swagger.json -o ./api -n api.ts How can we modify our build process arguments such that we can somehow inject a filter function before JSON stringifying the data going up? Or, otherwise, what's the proper method for adding our own value for The other issue here is dealing with TypeScript, which is currently claiming that these dates are string fields when - because of this transform process - they are in fact real dates. I'd like to resolve the above issue first, though, as it's a bit more critical. TIA! |
Beta Was this translation helpful? Give feedback.
Replies: 1 comment
-
Ok, I think I figured this out! The power of I started by switching from a command-line build process to a script. I now have a command that looks like, Then I used the This allowed me to resolve the first issue - making the uploaded / downloaded data automatically get filtered - by then simply passing functions to the new const apiService = new Api({
// ...
requestBodyFilter: (body) => encode(body),
responseBodyFilter: (body) => decode(body),
// ...
}); Lastly, to fix the typing, I used a trick directly provided by the docs to specify that the "date-time" string values should be given a "Date" type, instead of a plain "string". This is done in the import path from "node:path";
import { generateApi } from "swagger-typescript-api";
generateApi({
input: path.resolve(process.cwd(), "./api/swagger.json"),
output: path.resolve(process.cwd(), "./api"),
name: "api.ts",
primitiveTypeConstructs: (struct) => ({
...struct,
string: {
...(typeof struct.string === "string"
? { $default: struct.string }
: struct.string),
"date-time": "Date",
},
}),
templates: path.resolve(process.cwd(), "./api/templates"),
}); Hopefully this helps someone! It was a bit of work, but it seems to be running smoothly now. |
Beta Was this translation helpful? Give feedback.
Ok, I think I figured this out!
The power of
swagger-typescript-api
primarily exists in its ability to allow us to override templates. It took a lot of reconfiguration and work, but I think it's working very well now.I started by switching from a command-line build process to a script. I now have a command that looks like,
tsx ./api/generateApi.ts
which then importsgenerateApi
fromswagger-typescript-api
and calls it.Then I used the
generateApi
templates
parameter to provision a custom template for theHttpClient
class (http-client.ejs
) by copy/pasting the template from theswagger-typescript-api
package'stemplates/base/http-clients/fetch-http-client.ejs
file. In the copied template, …