yuheng/doc/fastify-docs/docs/Fluent-Schema.md
2025-03-19 15:54:28 +08:00

3.3 KiB
Raw Blame History

Fastify

Fluent Schema

验证和序列化一文中,我们列明了使用 JSON schema 验证输入、优化输出时所有可用的参数。

现在,你可以使用 fluent-json-schema 来更简单地设置 JSON schema并且复用常量。

基本设置

const S = require("fluent-json-schema");

// 你可以使用如下的一个对象,或查询数据库来获取数据
const MY_KEYS = {
  KEY1: "ONE",
  KEY2: "TWO",
};

const bodyJsonSchema = S.object()
  .prop("someKey", S.string())
  .prop("someOtherKey", S.number())
  .prop("requiredKey", S.array().maxItems(3).items(S.integer()).required())
  .prop("nullableKey", S.mixed([S.TYPES.NUMBER, S.TYPES.NULL]))
  .prop("multipleTypesKey", S.mixed([S.TYPES.BOOLEAN, S.TYPES.NUMBER]))
  .prop(
    "multipleRestrictedTypesKey",
    S.oneOf([S.string().maxLength(5), S.number().minimum(10)]),
  )
  .prop("enumKey", S.enum(Object.values(MY_KEYS)))
  .prop("notTypeKey", S.not(S.array()));

const queryStringJsonSchema = S.object()
  .prop("name", S.string())
  .prop("excitement", S.integer());

const paramsJsonSchema = S.object()
  .prop("par1", S.string())
  .prop("par2", S.integer());

const headersJsonSchema = S.object().prop("x-foo", S.string().required());

const schema = {
  body: bodyJsonSchema,
  querystring: queryStringJsonSchema, // (或) query: queryStringJsonSchema
  params: paramsJsonSchema,
  headers: headersJsonSchema,
};

fastify.post("/the/url", { schema }, handler);

复用

使用 fluent-json-schema,你可以简单且程序化地处理 schema并通过 addSchema() 来复用它们。 正如验证和序列化一文所述,有两种方法来引用 schema。

以下是一些例子:

使用$ref:引用外部的 schema。

const addressSchema = S.object()
  .id("#address")
  .prop("line1")
  .required()
  .prop("line2")
  .prop("country")
  .required()
  .prop("city")
  .required()
  .prop("zipcode")
  .required();

const commonSchemas = S.object()
  .id("https://fastify/demo")
  .definition("addressSchema", addressSchema)
  .definition("otherSchema", otherSchema); // 你可以任意添加需要的 schema

fastify.addSchema(commonSchemas);

const bodyJsonSchema = S.object()
  .prop("residence", S.ref("https://fastify/demo#address"))
  .required()
  .prop("office", S.ref("https://fastify/demo#/definitions/addressSchema"))
  .required();

const schema = { body: bodyJsonSchema };

fastify.post("/the/url", { schema }, handler);

替换方式:在验证阶段之前,使用共用 schema 替换某些字段。

const sharedAddressSchema = {
  $id: "sharedAddress",
  type: "object",
  required: ["line1", "country", "city", "zipcode"],
  properties: {
    line1: { type: "string" },
    line2: { type: "string" },
    country: { type: "string" },
    city: { type: "string" },
    zipcode: { type: "string" },
  },
};
fastify.addSchema(sharedAddressSchema);

const bodyJsonSchema = {
  type: "object",
  properties: {
    vacation: "sharedAddress#",
  },
};

const schema = { body: bodyJsonSchema };

fastify.post("/the/url", { schema }, handler);

特别注意:你可以在 fastify.addSchema 方法里混用 $ref替换方式