Define Schema
Referensi: Repositori GitHub Typebox
Sebuah kontrak adalah fungsi yang menerima satu input dan mengembalikan nilai yang sudah bersih. Define membangunnya dari dua bagian, sebuah transform yang membentuk output dan guard opsional yang menolak input sebelum transform berjalan.
Bentuk Define
Define(transform, guard?) mengembalikan sebuah kontrak:
import { Define } from '@neabyte/deserve'
// Hanya transform, tanpa guard
const Trim = Define((body: { name: string }) => ({
name: body.name.trim()
}))Transform menormalkan nilai, memangkas string, membuat email jadi huruf kecil, atau memaksa angka. Transform berjalan sebagai badan kontrak setelah input dipercaya.
Transform juga memiliki bentuk output. Guard yang lolos tidak membuang key tambahan, jadi field tak dikenal dari client tetap ada kecuali transform menghilangkannya. Mengembalikan object baru hanya dengan field yang diinginkan menjaga input kejutan keluar dari data tervalidasi:
import { Define } from '@neabyte/deserve'
// Output hanya menyimpan field yang disebut
const NewUser = Define((body: { name: string; role: string }) => ({
name: body.name.trim()
}))Di sini client yang mengirim role: 'admin' mendapati nilainya dibuang, karena transform tidak pernah menyalinnya.
Urutan Operasi
Memanggil sebuah kontrak menjalankan empat langkah dalam urutan tetap, dan transform hanya pernah melihat input yang sudah lolos setiap guard:
- Input string yang lebih panjang dari 10000 karakter ditolak sebelum hal lain.
- Input object dibekukan dalam (deep frozen) agar guard tidak bisa memutasinya.
- Setiap guard berjalan berurutan, melempar pada kegagalan pertama.
- Transform berjalan dan mengembalikan nilai yang sudah bersih.
Kontrak tanpa guard langsung lompat ke transform, jadi transform harus memercayai inputnya atau melakukan pemeriksaan sendiri.

Guard Menentukan Lolos Atau Gagal
Sebuah guard memeriksa input mentah dan mengembalikan keputusan:
truesaat input lolos.- Sebuah
stringuntuk satu alasan kegagalan. - Sebuah
string[]untuk beberapa alasan kegagalan sekaligus.

import { Define } from '@neabyte/deserve'
// Guard menolak nama kosong
const NewUser = Define(
(body: { name: string }) => ({ name: body.name.trim() }),
(body) => (body.name.trim().length > 0 ? true : 'name must not be empty')
)Guard yang mengembalikan alasan membuat kontrak melempar, dan validator mengubah lemparan itu menjadi 422 yang membawa alasan persis tersebut. Jalur dari sebuah alasan menuju respons ada di Membaca Data Tervalidasi.
Memeriksa Bentuk Lebih Dulu
Sebuah guard menerima input mentah, yang bisa null, sebuah array, atau nilai JSON apa pun yang dikirim client. Mengambil sebuah field pada bentuk yang salah akan melempar di dalam guard sebelum aturannya berjalan, jadi pemeriksaan bentuk datang lebih dulu:
import { Define } from '@neabyte/deserve'
// Pastikan object sebelum membaca field
function isRecord(value: unknown): value is Record<string, unknown> {
return value !== null && typeof value === 'object' && !Array.isArray(value)
}
const NewUser = Define(
(body: { name: string }) => ({ name: body.name.trim() }),
(body) => {
if (!isRecord(body)) {
return 'body must be a JSON object'
}
return typeof body['name'] === 'string' ? true : 'name must be a string'
}
)Lemparan di dalam guard tetap menjadi 422, tidak pernah 500, jadi pemeriksaan bentuk yang terlewat gagal dengan aman alih-alih menjatuhkan request.
Melaporkan Beberapa Field Sekaligus
Mengembalikan sebuah array melaporkan setiap field yang rusak dalam satu respons alih-alih satu per satu:
import { Define } from '@neabyte/deserve'
// Kumpulkan tiap kegagalan ke satu array
const NewUser = Define(
(body: { name: string; age: number }) => body,
(body) => {
const reasons: string[] = []
if (body.name.trim().length === 0) {
reasons.push('name must not be empty')
}
if (body.age < 18) {
reasons.push('age must be at least 18')
}
return reasons.length === 0 ? true : reasons
}
)Menyusun Beberapa Guard
Argumen kedua juga menerima array guard. Guard berjalan berurutan dan kontrak melempar pada yang pertama gagal, jadi guard berikutnya tidak pernah melihat input yang sudah ditolak guard sebelumnya:
import { Define } from '@neabyte/deserve'
// Cek bentuk dulu, aturan bisnis kedua
function hasFields(body: { from: string; to: string }): true | string {
return body.from && body.to ? true : 'from and to are required'
}
function distinctAccounts(body: { from: string; to: string }): true | string {
return body.from !== body.to ? true : 'from and to must differ'
}
const Transfer = Define(
(body: { from: string; to: string }) => body,
[hasFields, distinctAccounts]
)Memisahkan pemeriksaan bentuk dari aturan bisnis menjaga tiap guard tetap kecil dan membuat aturan lintas-field bisa berasumsi field-nya sudah ada.
Pengaman Bawaan
Batas string dan pembekuan dari Urutan Operasi berjalan otomatis, jadi sebuah kontrak tidak pernah membuang waktu pada payload raksasa dan sebuah guard tidak pernah memutasi nilai yang diperiksanya. Satu aturan lagi menjaga model waktunya:
- Guard async ditolak, karena validasi tetap sinkron dan dapat diprediksi.
Aturan ini berasal dari Typebox sendiri dan berlaku untuk setiap kontrak, baik yang berjalan lewat Middleware Validator maupun panggilan Validator.check langsung.
Langkah Berikutnya
- Middleware Validator - menghubungkan kontrak ke sumber request.
- Membaca Data Tervalidasi - membaca output transform di handler.
- Pola Lanjutan - menyusun guard dan mengatur urutan kegagalannya.
- Ringkasan Validasi - bagaimana semua bagian saling terhubung.