Transformers
Refer to tRPC's documentation on transformers for more in-depth information.
You are able to serialize the request body (for POST, PATCH, etc. requests) and response data for all requests. The transformers need to be added both to the server and the client.
TIP
Since only request.body will be serialized by the client during POST, PATCH, etc. requests, the server will only de-serialize the body on those requests.
Possible TODO: Add a custom query parameter for transformed json. Read implementation notes here.
Using superjson
SuperJSON allows us to transparently use primitives such as Date
/Map
/Set
s between the server and client that are not normally serializable with JSON.stringify
and JSON.parse
.
This means that you can send and receive this data without having to create your own encoding scheme.
TIP
All response data can be serialized/deserialized by both the client and server.
Only request bodies can be serialized. This means that you should not send Date
, Map
, Set
, etc. within the request query
or params
, since these are sent as part of the URL.
Steps
1. Install superjson
npm install superjson
pnpm add superjson
yarn add superjson
bun add superjson
2. Add the transformPlugin
to your Elysia server application
import { Elysia, t } from 'elysia'
import { edenPlugin } from '@ap0nia/eden-react-query/server'
import SuperJSON from 'superjson'
const app = new Elysia()
.use(edenPlugin({ transformer: SuperJSON }))
.get('/a', () => 'A')
.get('/b', () => 'B')
export type App = typeof app
3. Create a client with links, e.g. httpLink()
, httpBatchLink()
, etc
TypeScript will guide you to where you need to add
transformer
as soon as you've added it your server application via thetransformPlugin
helper.
import { createEdenTreatyReactQuery, httpLink } from '@ap0nia/eden-react-query'
import SuperJSON from 'superjson'
import type { App } from './server'
const eden = createEdenTreatyReactQuery<App>()
export const client = eden.createClient({
links: [
httpLink({
domain: 'http://localhost:3000',
transformer: SuperJSON,
}),
],
})
Different transformers for upload and download
If a transformer should only be used for one direction or different transformers should be used for upload and download (e.g., for performance reasons), you can provide individual transformers for upload and download. Make sure you use the same combined transformer everywhere.
Steps
Here superjson is used for uploading and devalue for downloading data because devalue is a lot faster but insecure to use on the server.
1. Install dependencies
npm install superjson devalue
pnpm add superjson devalue
yarn add superjson devalue
bun add superjson devalue
2. Create the transformer
import { uneval } from 'devalue'
import SuperJSON from 'superjson'
import type { DataTransformerOptions } from '@ap0nia/eden-react-query'
export const transformer: DataTransformerOptions = {
input: SuperJSON,
output: {
serialize: (object) => uneval(object),
// This `eval` only ever happens on the **client**
deserialize: (object) => eval(`(${object})`),
},
}
3. Add the transformer via the transformPlugin
to your Elysia.js server application
import { createEdenTreatyReactQuery, httpLink } from '@ap0nia/eden-react-query'
import { transformer } from './transformer'
export const eden = createEdenTreatyReactQuery()
export const client = eden.createClient({
links: [httpLink({ transformer })],
})
DataTransformer
interface
export interface DataTransformer {
serialize(object: any): any
deserialize(object: any): any
}
interface InputDataTransformer extends DataTransformer {
/**
* This function runs **on the client** before sending the data to the server.
*/
serialize(object: any): any
/**
* This function runs **on the server** to transform the data before it is passed to the resolver
*/
deserialize(object: any): any
}
interface OutputDataTransformer extends DataTransformer {
/**
* This function runs **on the server** before sending the data to the client.
*/
serialize(object: any): any
/**
* This function runs **only on the client** to transform the data sent from the server.
*/
deserialize(object: any): any
}
export interface CombinedDataTransformer {
/**
* Specify how the data sent from the client to the server should be transformed.
*/
input: InputDataTransformer
/**
* Specify how the data sent from the server to the client should be transformed.
*/
output: OutputDataTransformer
}
Transforming Query for GET Requests
WARNING
This does not exist yet!
If you use a plugin like this, it is possible that your REST API is not "compliant" based on typical standards. It is probably best to use a string transformer in your schema, or other processor on the string values, instead of expecting special input.
A big appeal of Elysia to me was that everything was REST and very easily documented and accessible by others.
INFO
Right now, query values are basically restricted to only string
GET requests. Applying this new strategy would enable transformed query input to be provided to GET requests.
Example: original query is a=valueA&b=valueB
Maybe we can use a custom, special query key like transformed_query
which accepts encoded JSON. Then when the transformPlugin
or edenPlugin
receive a request with that parameter, it will de-serialize the JSON input and merge it into the request query params.
const query = { a: 'valueA', b: 'valueB' }
const base = 'https://localhost.com:3000'
const regularUrl = `${base}?${new URLSearchParams(query)}`
const transformedUrl = `${base}?transformed_query=${encodeURIComponent(JSON.stringify(query))}`
import { getDataTransformer, type DataTransformerOptions } from '@ap0nia/eden-react-query'
import { Elysia } from 'elysia'
import SuperJSON from 'superjson'
const defaultTransformQueryKey = 'transformed_query'
export type TransformQueryOptions = {
transformer: DataTransformerOptions
queryKey?: 'string'
}
function edenQueryTransformQueryPlugin(options: TransformQueryOptions) {
const transformer = getDataTransformer(options.transformer)
const queryKey = options.queryKey ?? defaultTransformQueryKey
const plugin = (app: AnyElysia) => {
app.onParse(async (context) => {
if (typeof context.query[queryKey] === 'string') {
try {
const rawJsonQuery = JSON.parse(context.query[queryKey])
const deserializedQuery = transformer.input.deserialize(rawJsonQuery)
Object.assign(context.query, deserializedQuery)
} catch {
// noop
}
}
// Do other stuff, i.e. transform the request body if it exists.
if (context.contentType !== 'application/json') return
const json = await context.request.json()
return await resolvedTransformer.input.deserialize(json)
})
}
return plugin
}