A type alias that represents the mode of a JSON schema; either 'validation' or 'serialization'.
For some types, the inputs to validation differ from the outputs of serialization. For example,
computed fields will only be present when serializing, and should not be provided when
validating. This flag provides a way to indicate whether you want the JSON schema required
for validation inputs, or that will be matched by serialization outputs.
This class generates JSON schemas based on configured parameters. The default schema dialect
is https://json-schema.org/draft/2020-12/schema.
The class uses by_alias to configure how fields with
multiple names are handled and ref_template to format reference names.
Attributes:
Name
Type
Description
schema_dialect
The JSON schema dialect used to generate the schema. See
Declaring a Dialect
in the JSON Schema documentation for more information about dialects.
Warnings to ignore when generating the schema. self.render_warning_message will
do nothing if its argument kind is in ignored_warning_kinds;
this value can be modified on subclasses to easily control which warnings are emitted.
by_alias
Whether to use field aliases when generating the schema.
ref_template
The format string used when generating reference names.
def__init__(self,by_alias:bool=True,ref_template:str=DEFAULT_REF_TEMPLATE):self.by_alias=by_aliasself.ref_template=ref_templateself.core_to_json_refs:dict[CoreModeRef,JsonRef]={}self.core_to_defs_refs:dict[CoreModeRef,DefsRef]={}self.defs_to_core_refs:dict[DefsRef,CoreModeRef]={}self.json_to_defs_refs:dict[JsonRef,DefsRef]={}self.definitions:dict[DefsRef,JsonSchemaValue]={}self._config_wrapper_stack=_config.ConfigWrapperStack(_config.ConfigWrapper({}))self._mode:JsonSchemaMode='validation'# The following includes a mapping of a fully-unique defs ref choice to a list of preferred# alternatives, which are generally simpler, such as only including the class name.# At the end of schema generation, we use these to produce a JSON schema with more human-readable# definitions, which would also work better in a generated OpenAPI client, etc.self._prioritized_defsref_choices:dict[DefsRef,list[DefsRef]]={}self._collision_counter:dict[str,int]=defaultdict(int)self._collision_index:dict[str,int]={}self._schema_type_to_method=self.build_schema_type_to_method()# When we encounter definitions we need to try to build them immediately# so that they are available schemas that reference them# But it's possible that CoreSchema was never going to be used# (e.g. because the CoreSchema that references short circuits is JSON schema generation without needing# the reference) so instead of failing altogether if we can't build a definition we# store the error raised and re-throw it if we end up needing that defself._core_defs_invalid_for_json_schema:dict[DefsRef,PydanticInvalidForJsonSchema]={}# This changes to True after generating a schema, to prevent issues caused by accidental re-use# of a single instance of a schema generatorself._used=False
This class just contains mappings from core_schema attribute names to the corresponding
JSON schema attribute names. While I suspect it is unlikely to be necessary, you can in
principle override this class in a subclass of GenerateJsonSchema (by inheriting from
GenerateJsonSchema.ValidationsMapping) to change these mappings.
defbuild_schema_type_to_method(self,)->dict[CoreSchemaOrFieldType,Callable[[CoreSchemaOrField],JsonSchemaValue]]:"""Builds a dictionary mapping fields to methods for generating JSON schemas. Returns: A dictionary containing the mapping of `CoreSchemaOrFieldType` to a handler method. Raises: TypeError: If no method has been defined for generating a JSON schema for a given pydantic core schema type. """mapping:dict[CoreSchemaOrFieldType,Callable[[CoreSchemaOrField],JsonSchemaValue]]={}core_schema_types:list[CoreSchemaOrFieldType]=_typing_extra.all_literal_values(CoreSchemaOrFieldType# type: ignore)forkeyincore_schema_types:method_name=f"{key.replace('-','_')}_schema"try:mapping[key]=getattr(self,method_name)exceptAttributeErrorase:# pragma: no coverraiseTypeError(f'No method for generating JsonSchema for core_schema.type={key!r} 'f'(expected: {type(self).__name__}.{method_name})')fromereturnmapping
Generates JSON schema definitions from a list of core schemas, pairing the generated definitions with a
mapping that links the input keys to the definition references.
The first element is a dictionary whose keys are tuples of JSON schema key type and JSON mode, and
whose values are the JSON schema corresponding to that pair of inputs. (These schemas may have
JsonRef references to definitions that are defined in the second returned element.)
The second element is a dictionary whose keys are definition references for the JSON schemas
from the first returned element, and whose values are the actual JSON schema definitions.
defgenerate_definitions(self,inputs:Sequence[tuple[JsonSchemaKeyT,JsonSchemaMode,core_schema.CoreSchema]])->tuple[dict[tuple[JsonSchemaKeyT,JsonSchemaMode],JsonSchemaValue],dict[DefsRef,JsonSchemaValue]]:"""Generates JSON schema definitions from a list of core schemas, pairing the generated definitions with a mapping that links the input keys to the definition references. Args: inputs: A sequence of tuples, where: - The first element is a JSON schema key type. - The second element is the JSON mode: either 'validation' or 'serialization'. - The third element is a core schema. Returns: A tuple where: - The first element is a dictionary whose keys are tuples of JSON schema key type and JSON mode, and whose values are the JSON schema corresponding to that pair of inputs. (These schemas may have JsonRef references to definitions that are defined in the second returned element.) - The second element is a dictionary whose keys are definition references for the JSON schemas from the first returned element, and whose values are the actual JSON schema definitions. Raises: PydanticUserError: Raised if the JSON schema generator has already been used to generate a JSON schema. """ifself._used:raisePydanticUserError('This JSON schema generator has already been used to generate a JSON schema. 'f'You must create a new instance of {type(self).__name__} to generate a new JSON schema.',code='json-schema-already-used',)forkey,mode,schemaininputs:self._mode=modeself.generate_inner(schema)definitions_remapping=self._build_definitions_remapping()json_schemas_map:dict[tuple[JsonSchemaKeyT,JsonSchemaMode],DefsRef]={}forkey,mode,schemaininputs:self._mode=modejson_schema=self.generate_inner(schema)json_schemas_map[(key,mode)]=definitions_remapping.remap_json_schema(json_schema)json_schema={'$defs':self.definitions}json_schema=definitions_remapping.remap_json_schema(json_schema)self._used=Truereturnjson_schemas_map,_sort_json_schema(json_schema['$defs'])# type: ignore
defgenerate(self,schema:CoreSchema,mode:JsonSchemaMode='validation')->JsonSchemaValue:"""Generates a JSON schema for a specified schema in a specified mode. Args: schema: A Pydantic model. mode: The mode in which to generate the schema. Defaults to 'validation'. Returns: A JSON schema representing the specified schema. Raises: PydanticUserError: If the JSON schema generator has already been used to generate a JSON schema. """self._mode=modeifself._used:raisePydanticUserError('This JSON schema generator has already been used to generate a JSON schema. 'f'You must create a new instance of {type(self).__name__} to generate a new JSON schema.',code='json-schema-already-used',)json_schema:JsonSchemaValue=self.generate_inner(schema)json_ref_counts=self.get_json_ref_counts(json_schema)# Remove the top-level $ref if present; note that the _generate method already ensures there are no sibling keysref=cast(JsonRef,json_schema.get('$ref'))whilerefisnotNone:# may need to unpack multiple levelsref_json_schema=self.get_schema_from_definitions(ref)ifjson_ref_counts[ref]>1orref_json_schemaisNone:# Keep the ref, but use an allOf to remove the top level $refjson_schema={'allOf':[{'$ref':ref}]}else:# "Unpack" the ref since this is the only referencejson_schema=ref_json_schema.copy()# copy to prevent recursive dict referencejson_ref_counts[ref]-=1ref=cast(JsonRef,json_schema.get('$ref'))self._garbage_collect_definitions(json_schema)definitions_remapping=self._build_definitions_remapping()ifself.definitions:json_schema['$defs']=self.definitionsjson_schema=definitions_remapping.remap_json_schema(json_schema)# For now, we will not set the $schema key. However, if desired, this can be easily added by overriding# this method and adding the following line after a call to super().generate(schema):# json_schema['$schema'] = self.schema_dialectself._used=Truereturn_sort_json_schema(json_schema)
defgenerate_inner(self,schema:CoreSchemaOrField)->JsonSchemaValue:# noqa: C901"""Generates a JSON schema for a given core schema. Args: schema: The given core schema. Returns: The generated JSON schema. """# If a schema with the same CoreRef has been handled, just return a reference to it# Note that this assumes that it will _never_ be the case that the same CoreRef is used# on types that should have different JSON schemasif'ref'inschema:core_ref=CoreRef(schema['ref'])# type: ignore[typeddict-item]core_mode_ref=(core_ref,self.mode)ifcore_mode_refinself.core_to_defs_refsandself.core_to_defs_refs[core_mode_ref]inself.definitions:return{'$ref':self.core_to_json_refs[core_mode_ref]}# Generate the JSON schema, accounting for the json_schema_override and core_schema_overridemetadata_handler=_core_metadata.CoreMetadataHandler(schema)defpopulate_defs(core_schema:CoreSchema,json_schema:JsonSchemaValue)->JsonSchemaValue:if'ref'incore_schema:core_ref=CoreRef(core_schema['ref'])# type: ignore[typeddict-item]defs_ref,ref_json_schema=self.get_cache_defs_ref_schema(core_ref)json_ref=JsonRef(ref_json_schema['$ref'])self.json_to_defs_refs[json_ref]=defs_ref# Replace the schema if it's not a reference to itself# What we want to avoid is having the def be just a ref to itself# which is what would happen if we blindly assigned anyifjson_schema.get('$ref',None)!=json_ref:self.definitions[defs_ref]=json_schemaself._core_defs_invalid_for_json_schema.pop(defs_ref,None)json_schema=ref_json_schemareturnjson_schemadefconvert_to_all_of(json_schema:JsonSchemaValue)->JsonSchemaValue:if'$ref'injson_schemaandlen(json_schema.keys())>1:# technically you can't have any other keys next to a "$ref"# but it's an easy mistake to make and not hard to correct automatically herejson_schema=json_schema.copy()ref=json_schema.pop('$ref')json_schema={'allOf':[{'$ref':ref}],**json_schema}returnjson_schemadefhandler_func(schema_or_field:CoreSchemaOrField)->JsonSchemaValue:"""Generate a JSON schema based on the input schema. Args: schema_or_field: The core schema to generate a JSON schema from. Returns: The generated JSON schema. Raises: TypeError: If an unexpected schema type is encountered. """# Generate the core-schema-type-specific bits of the schema generation:json_schema:JsonSchemaValue|None=Noneifself.mode=='serialization'and'serialization'inschema_or_field:ser_schema=schema_or_field['serialization']# type: ignorejson_schema=self.ser_schema(ser_schema)ifjson_schemaisNone:if_core_utils.is_core_schema(schema_or_field)or_core_utils.is_core_schema_field(schema_or_field):generate_for_schema_type=self._schema_type_to_method[schema_or_field['type']]json_schema=generate_for_schema_type(schema_or_field)else:raiseTypeError(f'Unexpected schema type: schema={schema_or_field}')if_core_utils.is_core_schema(schema_or_field):json_schema=populate_defs(schema_or_field,json_schema)json_schema=convert_to_all_of(json_schema)returnjson_schemacurrent_handler=_schema_generation_shared.GenerateJsonSchemaHandler(self,handler_func)forjs_modify_functioninmetadata_handler.metadata.get('pydantic_js_functions',()):defnew_handler_func(schema_or_field:CoreSchemaOrField,current_handler:GetJsonSchemaHandler=current_handler,js_modify_function:GetJsonSchemaFunction=js_modify_function,)->JsonSchemaValue:json_schema=js_modify_function(schema_or_field,current_handler)if_core_utils.is_core_schema(schema_or_field):json_schema=populate_defs(schema_or_field,json_schema)original_schema=current_handler.resolve_ref_schema(json_schema)ref=json_schema.pop('$ref',None)ifrefandjson_schema:original_schema.update(json_schema)returnoriginal_schemacurrent_handler=_schema_generation_shared.GenerateJsonSchemaHandler(self,new_handler_func)forjs_modify_functioninmetadata_handler.metadata.get('pydantic_js_annotation_functions',()):defnew_handler_func(schema_or_field:CoreSchemaOrField,current_handler:GetJsonSchemaHandler=current_handler,js_modify_function:GetJsonSchemaFunction=js_modify_function,)->JsonSchemaValue:json_schema=js_modify_function(schema_or_field,current_handler)if_core_utils.is_core_schema(schema_or_field):json_schema=populate_defs(schema_or_field,json_schema)json_schema=convert_to_all_of(json_schema)returnjson_schemacurrent_handler=_schema_generation_shared.GenerateJsonSchemaHandler(self,new_handler_func)json_schema=current_handler(schema)if_core_utils.is_core_schema(schema):json_schema=populate_defs(schema,json_schema)json_schema=convert_to_all_of(json_schema)returnjson_schema
defany_schema(self,schema:core_schema.AnySchema)->JsonSchemaValue:"""Generates a JSON schema that matches any value. Args: schema: The core schema. Returns: The generated JSON schema. """return{}
defnone_schema(self,schema:core_schema.NoneSchema)->JsonSchemaValue:"""Generates a JSON schema that matches `None`. Args: schema: The core schema. Returns: The generated JSON schema. """return{'type':'null'}
defbool_schema(self,schema:core_schema.BoolSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a bool value. Args: schema: The core schema. Returns: The generated JSON schema. """return{'type':'boolean'}
defint_schema(self,schema:core_schema.IntSchema)->JsonSchemaValue:"""Generates a JSON schema that matches an int value. Args: schema: The core schema. Returns: The generated JSON schema. """json_schema:dict[str,Any]={'type':'integer'}self.update_with_validations(json_schema,schema,self.ValidationsMapping.numeric)json_schema={k:vfork,vinjson_schema.items()ifvnotin{math.inf,-math.inf}}returnjson_schema
deffloat_schema(self,schema:core_schema.FloatSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a float value. Args: schema: The core schema. Returns: The generated JSON schema. """json_schema:dict[str,Any]={'type':'number'}self.update_with_validations(json_schema,schema,self.ValidationsMapping.numeric)json_schema={k:vfork,vinjson_schema.items()ifvnotin{math.inf,-math.inf}}returnjson_schema
defdecimal_schema(self,schema:core_schema.DecimalSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a decimal value. Args: schema: The core schema. Returns: The generated JSON schema. """json_schema=self.str_schema(core_schema.str_schema())ifself.mode=='validation':multiple_of=schema.get('multiple_of')le=schema.get('le')ge=schema.get('ge')lt=schema.get('lt')gt=schema.get('gt')json_schema={'anyOf':[self.float_schema(core_schema.float_schema(allow_inf_nan=schema.get('allow_inf_nan'),multiple_of=Noneifmultiple_ofisNoneelsefloat(multiple_of),le=NoneifleisNoneelsefloat(le),ge=NoneifgeisNoneelsefloat(ge),lt=NoneifltisNoneelsefloat(lt),gt=NoneifgtisNoneelsefloat(gt),)),json_schema,],}returnjson_schema
defstr_schema(self,schema:core_schema.StringSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a string value. Args: schema: The core schema. Returns: The generated JSON schema. """json_schema={'type':'string'}self.update_with_validations(json_schema,schema,self.ValidationsMapping.string)ifisinstance(json_schema.get('pattern'),Pattern):# TODO: should we add regex flags to the pattern?json_schema['pattern']=json_schema.get('pattern').pattern# type: ignorereturnjson_schema
defbytes_schema(self,schema:core_schema.BytesSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a bytes value. Args: schema: The core schema. Returns: The generated JSON schema. """json_schema={'type':'string','format':'base64url'ifself._config.ser_json_bytes=='base64'else'binary'}self.update_with_validations(json_schema,schema,self.ValidationsMapping.bytes)returnjson_schema
defdate_schema(self,schema:core_schema.DateSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a date value. Args: schema: The core schema. Returns: The generated JSON schema. """json_schema={'type':'string','format':'date'}self.update_with_validations(json_schema,schema,self.ValidationsMapping.date)returnjson_schema
deftime_schema(self,schema:core_schema.TimeSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a time value. Args: schema: The core schema. Returns: The generated JSON schema. """return{'type':'string','format':'time'}
defdatetime_schema(self,schema:core_schema.DatetimeSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a datetime value. Args: schema: The core schema. Returns: The generated JSON schema. """return{'type':'string','format':'date-time'}
deftimedelta_schema(self,schema:core_schema.TimedeltaSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a timedelta value. Args: schema: The core schema. Returns: The generated JSON schema. """ifself._config.ser_json_timedelta=='float':return{'type':'number'}return{'type':'string','format':'duration'}
defliteral_schema(self,schema:core_schema.LiteralSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a literal value. Args: schema: The core schema. Returns: The generated JSON schema. """expected=[v.valueifisinstance(v,Enum)elsevforvinschema['expected']]# jsonify the expected valuesexpected=[to_jsonable_python(v)forvinexpected]result:dict[str,Any]={'enum':expected}iflen(expected)==1:result['const']=expected[0]types={type(e)foreinexpected}iftypes=={str}:result['type']='string'eliftypes=={int}:result['type']='integer'eliftypes=={float}:result['type']='numeric'eliftypes=={bool}:result['type']='boolean'eliftypes=={list}:result['type']='array'eliftypes=={type(None)}:result['type']='null'returnresult
defenum_schema(self,schema:core_schema.EnumSchema)->JsonSchemaValue:"""Generates a JSON schema that matches an Enum value. Args: schema: The core schema. Returns: The generated JSON schema. """enum_type=schema['cls']description=Noneifnotenum_type.__doc__elseinspect.cleandoc(enum_type.__doc__)if(description=='An enumeration.'):# This is the default value provided by enum.EnumMeta.__new__; don't use itdescription=Noneresult:dict[str,Any]={'title':enum_type.__name__,'description':description}result={k:vfork,vinresult.items()ifvisnotNone}expected=[to_jsonable_python(v.value)forvinschema['members']]result['enum']=expectediflen(expected)==1:result['const']=expected[0]types={type(e)foreinexpected}ifisinstance(enum_type,str)ortypes=={str}:result['type']='string'elifisinstance(enum_type,int)ortypes=={int}:result['type']='integer'elifisinstance(enum_type,float)ortypes=={float}:result['type']='numeric'eliftypes=={bool}:result['type']='boolean'eliftypes=={list}:result['type']='array'returnresult
defis_instance_schema(self,schema:core_schema.IsInstanceSchema)->JsonSchemaValue:"""Handles JSON schema generation for a core schema that checks if a value is an instance of a class. Unless overridden in a subclass, this raises an error. Args: schema: The core schema. Returns: The generated JSON schema. """returnself.handle_invalid_for_json_schema(schema,f'core_schema.IsInstanceSchema ({schema["cls"]})')
defis_subclass_schema(self,schema:core_schema.IsSubclassSchema)->JsonSchemaValue:"""Handles JSON schema generation for a core schema that checks if a value is a subclass of a class. For backwards compatibility with v1, this does not raise an error, but can be overridden to change this. Args: schema: The core schema. Returns: The generated JSON schema. """# Note: This is for compatibility with V1; you can override if you want different behavior.return{}
defcallable_schema(self,schema:core_schema.CallableSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a callable value. Unless overridden in a subclass, this raises an error. Args: schema: The core schema. Returns: The generated JSON schema. """returnself.handle_invalid_for_json_schema(schema,'core_schema.CallableSchema')
deflist_schema(self,schema:core_schema.ListSchema)->JsonSchemaValue:"""Returns a schema that matches a list schema. Args: schema: The core schema. Returns: The generated JSON schema. """items_schema={}if'items_schema'notinschemaelseself.generate_inner(schema['items_schema'])json_schema={'type':'array','items':items_schema}self.update_with_validations(json_schema,schema,self.ValidationsMapping.array)returnjson_schema
@deprecated('`tuple_positional_schema` is deprecated. Use `tuple_schema` instead.',category=None)@finaldeftuple_positional_schema(self,schema:core_schema.TupleSchema)->JsonSchemaValue:"""Replaced by `tuple_schema`."""warnings.warn('`tuple_positional_schema` is deprecated. Use `tuple_schema` instead.',PydanticDeprecatedSince26,stacklevel=2,)returnself.tuple_schema(schema)
@deprecated('`tuple_variable_schema` is deprecated. Use `tuple_schema` instead.',category=None)@finaldeftuple_variable_schema(self,schema:core_schema.TupleSchema)->JsonSchemaValue:"""Replaced by `tuple_schema`."""warnings.warn('`tuple_variable_schema` is deprecated. Use `tuple_schema` instead.',PydanticDeprecatedSince26,stacklevel=2,)returnself.tuple_schema(schema)
deftuple_schema(self,schema:core_schema.TupleSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a tuple schema e.g. `Tuple[int, str, bool]` or `Tuple[int, ...]`. Args: schema: The core schema. Returns: The generated JSON schema. """json_schema:JsonSchemaValue={'type':'array'}if'variadic_item_index'inschema:variadic_item_index=schema['variadic_item_index']ifvariadic_item_index>0:json_schema['minItems']=variadic_item_indexjson_schema['prefixItems']=[self.generate_inner(item)foriteminschema['items_schema'][:variadic_item_index]]ifvariadic_item_index+1==len(schema['items_schema']):# if the variadic item is the last item, then represent it faithfullyjson_schema['items']=self.generate_inner(schema['items_schema'][variadic_item_index])else:# otherwise, 'items' represents the schema for the variadic# item plus the suffix, so just allow anything for simplicity# for nowjson_schema['items']=Trueelse:prefixItems=[self.generate_inner(item)foriteminschema['items_schema']]ifprefixItems:json_schema['prefixItems']=prefixItemsjson_schema['minItems']=len(prefixItems)json_schema['maxItems']=len(prefixItems)self.update_with_validations(json_schema,schema,self.ValidationsMapping.array)returnjson_schema
defset_schema(self,schema:core_schema.SetSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a set schema. Args: schema: The core schema. Returns: The generated JSON schema. """returnself._common_set_schema(schema)
deffrozenset_schema(self,schema:core_schema.FrozenSetSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a frozenset schema. Args: schema: The core schema. Returns: The generated JSON schema. """returnself._common_set_schema(schema)
defgenerator_schema(self,schema:core_schema.GeneratorSchema)->JsonSchemaValue:"""Returns a JSON schema that represents the provided GeneratorSchema. Args: schema: The schema. Returns: The generated JSON schema. """items_schema={}if'items_schema'notinschemaelseself.generate_inner(schema['items_schema'])json_schema={'type':'array','items':items_schema}self.update_with_validations(json_schema,schema,self.ValidationsMapping.array)returnjson_schema
defdict_schema(self,schema:core_schema.DictSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a dict schema. Args: schema: The core schema. Returns: The generated JSON schema. """json_schema:JsonSchemaValue={'type':'object'}keys_schema=self.generate_inner(schema['keys_schema']).copy()if'keys_schema'inschemaelse{}keys_pattern=keys_schema.pop('pattern',None)values_schema=self.generate_inner(schema['values_schema']).copy()if'values_schema'inschemaelse{}values_schema.pop('title',None)# don't give a title to the additionalPropertiesifvalues_schemaorkeys_patternisnotNone:# don't add additionalProperties if it's emptyifkeys_patternisNone:json_schema['additionalProperties']=values_schemaelse:json_schema['patternProperties']={keys_pattern:values_schema}self.update_with_validations(json_schema,schema,self.ValidationsMapping.object)returnjson_schema
deffunction_before_schema(self,schema:core_schema.BeforeValidatorFunctionSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a function-before schema. Args: schema: The core schema. Returns: The generated JSON schema. """returnself._function_schema(schema)
deffunction_after_schema(self,schema:core_schema.AfterValidatorFunctionSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a function-after schema. Args: schema: The core schema. Returns: The generated JSON schema. """returnself._function_schema(schema)
deffunction_plain_schema(self,schema:core_schema.PlainValidatorFunctionSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a function-plain schema. Args: schema: The core schema. Returns: The generated JSON schema. """returnself._function_schema(schema)
deffunction_wrap_schema(self,schema:core_schema.WrapValidatorFunctionSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a function-wrap schema. Args: schema: The core schema. Returns: The generated JSON schema. """returnself._function_schema(schema)
defdefault_schema(self,schema:core_schema.WithDefaultSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema with a default value. Args: schema: The core schema. Returns: The generated JSON schema. """json_schema=self.generate_inner(schema['schema'])if'default'notinschema:returnjson_schemadefault=schema['default']# Note: if you want to include the value returned by the default_factory,# override this method and replace the code above with:# if 'default' in schema:# default = schema['default']# elif 'default_factory' in schema:# default = schema['default_factory']()# else:# return json_schema# we reflect the application of custom plain, no-info serializers to defaults for# json schemas viewed in serialization mode# TODO: improvements along with https://github.com/pydantic/pydantic/issues/8208# TODO: improve type safety hereifself.mode=='serialization':if((ser_schema:=schema['schema'].get('serialization',{}))and(ser_func:=ser_schema.get('function'))andser_schema.get('type')=='function-plain'# type: ignoreandser_schema.get('info_arg')isFalse# type: ignore):default=ser_func(default)# type: ignoretry:encoded_default=self.encode_default(default)exceptpydantic_core.PydanticSerializationError:self.emit_warning('non-serializable-default',f'Default value {default} is not JSON serializable; excluding default from JSON schema',)# Return the inner schema, as though there was no defaultreturnjson_schemaif'$ref'injson_schema:# Since reference schemas do not support child keys, we wrap the reference schema in a single-case allOf:return{'allOf':[json_schema],'default':encoded_default}else:json_schema['default']=encoded_defaultreturnjson_schema
defnullable_schema(self,schema:core_schema.NullableSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that allows null values. Args: schema: The core schema. Returns: The generated JSON schema. """null_schema={'type':'null'}inner_json_schema=self.generate_inner(schema['schema'])ifinner_json_schema==null_schema:returnnull_schemaelse:# Thanks to the equality check against `null_schema` above, I think 'oneOf' would also be valid here;# I'll use 'anyOf' for now, but it could be changed it if it would work better with some external toolingreturnself.get_flattened_anyof([inner_json_schema,null_schema])
defunion_schema(self,schema:core_schema.UnionSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that allows values matching any of the given schemas. Args: schema: The core schema. Returns: The generated JSON schema. """generated:list[JsonSchemaValue]=[]choices=schema['choices']forchoiceinchoices:# choice will be a tuple if an explicit label was providedchoice_schema=choice[0]ifisinstance(choice,tuple)elsechoicetry:generated.append(self.generate_inner(choice_schema))exceptPydanticOmit:continueexceptPydanticInvalidForJsonSchemaasexc:self.emit_warning('skipped-choice',exc.message)iflen(generated)==1:returngenerated[0]returnself.get_flattened_anyof(generated)
Generates a JSON schema that matches a schema that allows values matching any of the given schemas, where
the schemas are tagged with a discriminator field that indicates which schema should be used to validate
the value.
deftagged_union_schema(self,schema:core_schema.TaggedUnionSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that allows values matching any of the given schemas, where the schemas are tagged with a discriminator field that indicates which schema should be used to validate the value. Args: schema: The core schema. Returns: The generated JSON schema. """generated:dict[str,JsonSchemaValue]={}fork,vinschema['choices'].items():ifisinstance(k,Enum):k=k.valuetry:# Use str(k) since keys must be strings for json; while not technically correct,# it's the closest that can be represented in valid JSONgenerated[str(k)]=self.generate_inner(v).copy()exceptPydanticOmit:continueexceptPydanticInvalidForJsonSchemaasexc:self.emit_warning('skipped-choice',exc.message)one_of_choices=_deduplicate_schemas(generated.values())json_schema:JsonSchemaValue={'oneOf':one_of_choices}# This reflects the v1 behavior; TODO: we should make it possible to exclude OpenAPI stuff from the JSON schemaopenapi_discriminator=self._extract_discriminator(schema,one_of_choices)ifopenapi_discriminatorisnotNone:json_schema['discriminator']={'propertyName':openapi_discriminator,'mapping':{k:v.get('$ref',v)fork,vingenerated.items()},}returnjson_schema
Generates a JSON schema that matches a core_schema.ChainSchema.
When generating a schema for validation, we return the validation JSON schema for the first step in the chain.
For serialization, we return the serialization JSON schema for the last step in the chain.
defchain_schema(self,schema:core_schema.ChainSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a core_schema.ChainSchema. When generating a schema for validation, we return the validation JSON schema for the first step in the chain. For serialization, we return the serialization JSON schema for the last step in the chain. Args: schema: The core schema. Returns: The generated JSON schema. """step_index=0ifself.mode=='validation'else-1# use first step for validation, last for serializationreturnself.generate_inner(schema['steps'][step_index])
deflax_or_strict_schema(self,schema:core_schema.LaxOrStrictSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that allows values matching either the lax schema or the strict schema. Args: schema: The core schema. Returns: The generated JSON schema. """# TODO: Need to read the default value off of model config or whateveruse_strict=schema.get('strict',False)# TODO: replace this default False# If your JSON schema fails to generate it is probably# because one of the following two branches failed.ifuse_strict:returnself.generate_inner(schema['strict_schema'])else:returnself.generate_inner(schema['lax_schema'])
defjson_or_python_schema(self,schema:core_schema.JsonOrPythonSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that allows values matching either the JSON schema or the Python schema. The JSON schema is used instead of the Python schema. If you want to use the Python schema, you should override this method. Args: schema: The core schema. Returns: The generated JSON schema. """returnself.generate_inner(schema['json_schema'])
deftyped_dict_schema(self,schema:core_schema.TypedDictSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a typed dict. Args: schema: The core schema. Returns: The generated JSON schema. """total=schema.get('total',True)named_required_fields:list[tuple[str,bool,CoreSchemaField]]=[(name,self.field_is_required(field,total),field)forname,fieldinschema['fields'].items()ifself.field_is_present(field)]ifself.mode=='serialization':named_required_fields.extend(self._name_required_computed_fields(schema.get('computed_fields',[])))cls=_get_typed_dict_cls(schema)config=_get_typed_dict_config(cls)withself._config_wrapper_stack.push(config):json_schema=self._named_required_fields_schema(named_required_fields)json_schema_extra=config.get('json_schema_extra')extra=schema.get('extra_behavior')ifextraisNone:extra=config.get('extra','ignore')ifclsisnotNone:title=config.get('title')orcls.__name__json_schema=self._update_class_schema(json_schema,title,extra,cls,json_schema_extra)else:ifextra=='forbid':json_schema['additionalProperties']=Falseelifextra=='allow':json_schema['additionalProperties']=Truereturnjson_schema
deftyped_dict_field_schema(self,schema:core_schema.TypedDictField)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a typed dict field. Args: schema: The core schema. Returns: The generated JSON schema. """returnself.generate_inner(schema['schema'])
defdataclass_field_schema(self,schema:core_schema.DataclassField)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a dataclass field. Args: schema: The core schema. Returns: The generated JSON schema. """returnself.generate_inner(schema['schema'])
defmodel_field_schema(self,schema:core_schema.ModelField)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a model field. Args: schema: The core schema. Returns: The generated JSON schema. """returnself.generate_inner(schema['schema'])
defcomputed_field_schema(self,schema:core_schema.ComputedField)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a computed field. Args: schema: The core schema. Returns: The generated JSON schema. """returnself.generate_inner(schema['return_schema'])
defmodel_schema(self,schema:core_schema.ModelSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a model. Args: schema: The core schema. Returns: The generated JSON schema. """# We do not use schema['model'].model_json_schema() here# because it could lead to inconsistent refs handling, etc.cls=cast('type[BaseModel]',schema['cls'])config=cls.model_configtitle=config.get('title')withself._config_wrapper_stack.push(config):json_schema=self.generate_inner(schema['schema'])json_schema_extra=config.get('json_schema_extra')ifcls.__pydantic_root_model__:root_json_schema_extra=cls.model_fields['root'].json_schema_extraifjson_schema_extraandroot_json_schema_extra:raiseValueError('"model_config[\'json_schema_extra\']" and "Field.json_schema_extra" on "RootModel.root"'' field must not be set simultaneously')ifroot_json_schema_extra:json_schema_extra=root_json_schema_extrajson_schema=self._update_class_schema(json_schema,title,config.get('extra',None),cls,json_schema_extra)returnjson_schema
defresolve_schema_to_update(self,json_schema:JsonSchemaValue)->JsonSchemaValue:"""Resolve a JsonSchemaValue to the non-ref schema if it is a $ref schema. Args: json_schema: The schema to resolve. Returns: The resolved schema. """if'$ref'injson_schema:schema_to_update=self.get_schema_from_definitions(JsonRef(json_schema['$ref']))ifschema_to_updateisNone:raiseRuntimeError(f'Cannot update undefined schema for $ref={json_schema["$ref"]}')returnself.resolve_schema_to_update(schema_to_update)else:schema_to_update=json_schemareturnschema_to_update
defmodel_fields_schema(self,schema:core_schema.ModelFieldsSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a model's fields. Args: schema: The core schema. Returns: The generated JSON schema. """named_required_fields:list[tuple[str,bool,CoreSchemaField]]=[(name,self.field_is_required(field,total=True),field)forname,fieldinschema['fields'].items()ifself.field_is_present(field)]ifself.mode=='serialization':named_required_fields.extend(self._name_required_computed_fields(schema.get('computed_fields',[])))json_schema=self._named_required_fields_schema(named_required_fields)extras_schema=schema.get('extras_schema',None)ifextras_schemaisnotNone:schema_to_update=self.resolve_schema_to_update(json_schema)schema_to_update['additionalProperties']=self.generate_inner(extras_schema)returnjson_schema
deffield_is_present(self,field:CoreSchemaField)->bool:"""Whether the field should be included in the generated JSON schema. Args: field: The schema for the field itself. Returns: `True` if the field should be included in the generated JSON schema, `False` otherwise. """ifself.mode=='serialization':# If you still want to include the field in the generated JSON schema,# override this method and return Truereturnnotfield.get('serialization_exclude')elifself.mode=='validation':returnTrueelse:assert_never(self.mode)
Whether the field should be marked as required in the generated JSON schema.
(Note that this is irrelevant if the field is not present in the JSON schema.).
Only applies to TypedDictFields.
Indicates if the TypedDict this field belongs to is total, in which case any fields that don't
explicitly specify required=False are required.
deffield_is_required(self,field:core_schema.ModelField|core_schema.DataclassField|core_schema.TypedDictField,total:bool,)->bool:"""Whether the field should be marked as required in the generated JSON schema. (Note that this is irrelevant if the field is not present in the JSON schema.). Args: field: The schema for the field itself. total: Only applies to `TypedDictField`s. Indicates if the `TypedDict` this field belongs to is total, in which case any fields that don't explicitly specify `required=False` are required. Returns: `True` if the field should be marked as required in the generated JSON schema, `False` otherwise. """ifself.mode=='serialization'andself._config.json_schema_serialization_defaults_required:returnnotfield.get('serialization_exclude')else:iffield['type']=='typed-dict-field':returnfield.get('required',total)else:returnfield['schema']['type']!='default'
defdataclass_args_schema(self,schema:core_schema.DataclassArgsSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a dataclass's constructor arguments. Args: schema: The core schema. Returns: The generated JSON schema. """named_required_fields:list[tuple[str,bool,CoreSchemaField]]=[(field['name'],self.field_is_required(field,total=True),field)forfieldinschema['fields']ifself.field_is_present(field)]ifself.mode=='serialization':named_required_fields.extend(self._name_required_computed_fields(schema.get('computed_fields',[])))returnself._named_required_fields_schema(named_required_fields)
defdataclass_schema(self,schema:core_schema.DataclassSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a dataclass. Args: schema: The core schema. Returns: The generated JSON schema. """cls=schema['cls']config:ConfigDict=getattr(cls,'__pydantic_config__',cast('ConfigDict',{}))title=config.get('title')orcls.__name__withself._config_wrapper_stack.push(config):json_schema=self.generate_inner(schema['schema']).copy()json_schema_extra=config.get('json_schema_extra')json_schema=self._update_class_schema(json_schema,title,config.get('extra',None),cls,json_schema_extra)# Dataclass-specific handling of descriptionifis_dataclass(cls)andnothasattr(cls,'__pydantic_validator__'):# vanilla dataclass; don't use cls.__doc__ as it will contain the class signature by defaultdescription=Noneelse:description=Noneifcls.__doc__isNoneelseinspect.cleandoc(cls.__doc__)ifdescription:json_schema['description']=descriptionreturnjson_schema
defarguments_schema(self,schema:core_schema.ArgumentsSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a function's arguments. Args: schema: The core schema. Returns: The generated JSON schema. """metadata=_core_metadata.CoreMetadataHandler(schema).metadataprefer_positional=metadata.get('pydantic_js_prefer_positional_arguments')arguments=schema['arguments_schema']kw_only_arguments=[aforainargumentsifa.get('mode')=='keyword_only']kw_or_p_arguments=[aforainargumentsifa.get('mode')in{'positional_or_keyword',None}]p_only_arguments=[aforainargumentsifa.get('mode')=='positional_only']var_args_schema=schema.get('var_args_schema')var_kwargs_schema=schema.get('var_kwargs_schema')ifprefer_positional:positional_possible=notkw_only_argumentsandnotvar_kwargs_schemaifpositional_possible:returnself.p_arguments_schema(p_only_arguments+kw_or_p_arguments,var_args_schema)keyword_possible=notp_only_argumentsandnotvar_args_schemaifkeyword_possible:returnself.kw_arguments_schema(kw_or_p_arguments+kw_only_arguments,var_kwargs_schema)ifnotprefer_positional:positional_possible=notkw_only_argumentsandnotvar_kwargs_schemaifpositional_possible:returnself.p_arguments_schema(p_only_arguments+kw_or_p_arguments,var_args_schema)raisePydanticInvalidForJsonSchema('Unable to generate JSON schema for arguments validator with positional-only and keyword-only arguments')
defkw_arguments_schema(self,arguments:list[core_schema.ArgumentsParameter],var_kwargs_schema:CoreSchema|None)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a function's keyword arguments. Args: arguments: The core schema. Returns: The generated JSON schema. """properties:dict[str,JsonSchemaValue]={}required:list[str]=[]forargumentinarguments:name=self.get_argument_name(argument)argument_schema=self.generate_inner(argument['schema']).copy()argument_schema['title']=self.get_title_from_name(name)properties[name]=argument_schemaifargument['schema']['type']!='default':# This assumes that if the argument has a default value,# the inner schema must be of type WithDefaultSchema.# I believe this is true, but I am not 100% surerequired.append(name)json_schema:JsonSchemaValue={'type':'object','properties':properties}ifrequired:json_schema['required']=requiredifvar_kwargs_schema:additional_properties_schema=self.generate_inner(var_kwargs_schema)ifadditional_properties_schema:json_schema['additionalProperties']=additional_properties_schemaelse:json_schema['additionalProperties']=Falsereturnjson_schema
defp_arguments_schema(self,arguments:list[core_schema.ArgumentsParameter],var_args_schema:CoreSchema|None)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a function's positional arguments. Args: arguments: The core schema. Returns: The generated JSON schema. """prefix_items:list[JsonSchemaValue]=[]min_items=0forargumentinarguments:name=self.get_argument_name(argument)argument_schema=self.generate_inner(argument['schema']).copy()argument_schema['title']=self.get_title_from_name(name)prefix_items.append(argument_schema)ifargument['schema']['type']!='default':# This assumes that if the argument has a default value,# the inner schema must be of type WithDefaultSchema.# I believe this is true, but I am not 100% suremin_items+=1json_schema:JsonSchemaValue={'type':'array','prefixItems':prefix_items}ifmin_items:json_schema['minItems']=min_itemsifvar_args_schema:items_schema=self.generate_inner(var_args_schema)ifitems_schema:json_schema['items']=items_schemaelse:json_schema['maxItems']=len(prefix_items)returnjson_schema
defget_argument_name(self,argument:core_schema.ArgumentsParameter)->str:"""Retrieves the name of an argument. Args: argument: The core schema. Returns: The name of the argument. """name=argument['name']ifself.by_alias:alias=argument.get('alias')ifisinstance(alias,str):name=aliaselse:pass# might want to do something else?returnname
defcall_schema(self,schema:core_schema.CallSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a function call. Args: schema: The core schema. Returns: The generated JSON schema. """returnself.generate_inner(schema['arguments_schema'])
defcustom_error_schema(self,schema:core_schema.CustomErrorSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a custom error. Args: schema: The core schema. Returns: The generated JSON schema. """returnself.generate_inner(schema['schema'])
defjson_schema(self,schema:core_schema.JsonSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a JSON object. Args: schema: The core schema. Returns: The generated JSON schema. """content_core_schema=schema.get('schema')orcore_schema.any_schema()content_json_schema=self.generate_inner(content_core_schema)ifself.mode=='validation':return{'type':'string','contentMediaType':'application/json','contentSchema':content_json_schema}else:# self.mode == 'serialization'returncontent_json_schema
defurl_schema(self,schema:core_schema.UrlSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a URL. Args: schema: The core schema. Returns: The generated JSON schema. """json_schema={'type':'string','format':'uri','minLength':1}self.update_with_validations(json_schema,schema,self.ValidationsMapping.string)returnjson_schema
defmulti_host_url_schema(self,schema:core_schema.MultiHostUrlSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a URL that can be used with multiple hosts. Args: schema: The core schema. Returns: The generated JSON schema. """# Note: 'multi-host-uri' is a custom/pydantic-specific format, not part of the JSON Schema specjson_schema={'type':'string','format':'multi-host-uri','minLength':1}self.update_with_validations(json_schema,schema,self.ValidationsMapping.string)returnjson_schema
defuuid_schema(self,schema:core_schema.UuidSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a UUID. Args: schema: The core schema. Returns: The generated JSON schema. """return{'type':'string','format':'uuid'}
defdefinitions_schema(self,schema:core_schema.DefinitionsSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that defines a JSON object with definitions. Args: schema: The core schema. Returns: The generated JSON schema. """fordefinitioninschema['definitions']:try:self.generate_inner(definition)exceptPydanticInvalidForJsonSchemaase:core_ref:CoreRef=CoreRef(definition['ref'])# type: ignoreself._core_defs_invalid_for_json_schema[self.get_defs_ref((core_ref,self.mode))]=econtinuereturnself.generate_inner(schema['schema'])
defdefinition_ref_schema(self,schema:core_schema.DefinitionReferenceSchema)->JsonSchemaValue:"""Generates a JSON schema that matches a schema that references a definition. Args: schema: The core schema. Returns: The generated JSON schema. """core_ref=CoreRef(schema['schema_ref'])_,ref_json_schema=self.get_cache_defs_ref_schema(core_ref)returnref_json_schema
defser_schema(self,schema:core_schema.SerSchema|core_schema.IncExSeqSerSchema|core_schema.IncExDictSerSchema)->JsonSchemaValue|None:"""Generates a JSON schema that matches a schema that defines a serialized object. Args: schema: The core schema. Returns: The generated JSON schema. """schema_type=schema['type']ifschema_type=='function-plain'orschema_type=='function-wrap':# PlainSerializerFunctionSerSchema or WrapSerializerFunctionSerSchemareturn_schema=schema.get('return_schema')ifreturn_schemaisnotNone:returnself.generate_inner(return_schema)elifschema_type=='format'orschema_type=='to-string':# FormatSerSchema or ToStringSerSchemareturnself.str_schema(core_schema.str_schema())elifschema['type']=='model':# ModelSerSchemareturnself.generate_inner(schema['schema'])returnNone
defget_title_from_name(self,name:str)->str:"""Retrieves a title from a name. Args: name: The name to retrieve a title from. Returns: The title. """returnname.title().replace('_',' ')
Returns true if a field with the given schema should have a title set based on the field name.
Intuitively, we want this to return true for schemas that wouldn't otherwise provide their own title
(e.g., int, float, str), and false for those that would (e.g., BaseModel subclasses).
deffield_title_should_be_set(self,schema:CoreSchemaOrField)->bool:"""Returns true if a field with the given schema should have a title set based on the field name. Intuitively, we want this to return true for schemas that wouldn't otherwise provide their own title (e.g., int, float, str), and false for those that would (e.g., BaseModel subclasses). Args: schema: The schema to check. Returns: `True` if the field should have a title set, `False` otherwise. """if_core_utils.is_core_schema_field(schema):ifschema['type']=='computed-field':field_schema=schema['return_schema']else:field_schema=schema['schema']returnself.field_title_should_be_set(field_schema)elif_core_utils.is_core_schema(schema):ifschema.get('ref'):# things with refs, such as models and enums, should not have titles setreturnFalseifschema['type']in{'default','nullable','definitions'}:returnself.field_title_should_be_set(schema['schema'])# type: ignore[typeddict-item]if_core_utils.is_function_with_inner_schema(schema):returnself.field_title_should_be_set(schema['schema'])ifschema['type']=='definition-ref':# Referenced schemas should not have titles set for the same reason# schemas with refs should notreturnFalsereturnTrue# anything else should have title setelse:raisePydanticInvalidForJsonSchema(f'Unexpected schema type: schema={schema}')# pragma: no cover
defnormalize_name(self,name:str)->str:"""Normalizes a name to be used as a key in a dictionary. Args: name: The name to normalize. Returns: The normalized name. """returnre.sub(r'[^a-zA-Z0-9.\-_]','_',name).replace('.','__')
defget_defs_ref(self,core_mode_ref:CoreModeRef)->DefsRef:"""Override this method to change the way that definitions keys are generated from a core reference. Args: core_mode_ref: The core reference. Returns: The definitions key. """# Split the core ref into "components"; generic origins and arguments are each separate componentscore_ref,mode=core_mode_refcomponents=re.split(r'([\][,])',core_ref)# Remove IDs from each componentcomponents=[x.rsplit(':',1)[0]forxincomponents]core_ref_no_id=''.join(components)# Remove everything before the last period from each "component"components=[re.sub(r'(?:[^.[\]]+\.)+((?:[^.[\]]+))',r'\1',x)forxincomponents]short_ref=''.join(components)mode_title=_MODE_TITLE_MAPPING[mode]# It is important that the generated defs_ref values be such that at least one choice will not# be generated for any other core_ref. Currently, this should be the case because we include# the id of the source type in the core_refname=DefsRef(self.normalize_name(short_ref))name_mode=DefsRef(self.normalize_name(short_ref)+f'-{mode_title}')module_qualname=DefsRef(self.normalize_name(core_ref_no_id))module_qualname_mode=DefsRef(f'{module_qualname}-{mode_title}')module_qualname_id=DefsRef(self.normalize_name(core_ref))occurrence_index=self._collision_index.get(module_qualname_id)ifoccurrence_indexisNone:self._collision_counter[module_qualname]+=1occurrence_index=self._collision_index[module_qualname_id]=self._collision_counter[module_qualname]module_qualname_occurrence=DefsRef(f'{module_qualname}__{occurrence_index}')module_qualname_occurrence_mode=DefsRef(f'{module_qualname_mode}__{occurrence_index}')self._prioritized_defsref_choices[module_qualname_occurrence_mode]=[name,name_mode,module_qualname,module_qualname_mode,module_qualname_occurrence,module_qualname_occurrence_mode,]returnmodule_qualname_occurrence_mode
This method wraps the get_defs_ref method with some cache-lookup/population logic,
and returns both the produced defs_ref and the JSON schema that will refer to the right definition.
Parameters:
Name
Type
Description
Default
core_ref
CoreRef
The core reference to get the definitions reference for.
defget_cache_defs_ref_schema(self,core_ref:CoreRef)->tuple[DefsRef,JsonSchemaValue]:"""This method wraps the get_defs_ref method with some cache-lookup/population logic, and returns both the produced defs_ref and the JSON schema that will refer to the right definition. Args: core_ref: The core reference to get the definitions reference for. Returns: A tuple of the definitions reference and the JSON schema that will refer to it. """core_mode_ref=(core_ref,self.mode)maybe_defs_ref=self.core_to_defs_refs.get(core_mode_ref)ifmaybe_defs_refisnotNone:json_ref=self.core_to_json_refs[core_mode_ref]returnmaybe_defs_ref,{'$ref':json_ref}defs_ref=self.get_defs_ref(core_mode_ref)# populate the ref translation mappingsself.core_to_defs_refs[core_mode_ref]=defs_refself.defs_to_core_refs[defs_ref]=core_mode_refjson_ref=JsonRef(self.ref_template.format(model=defs_ref))self.core_to_json_refs[core_mode_ref]=json_refself.json_to_defs_refs[json_ref]=defs_refref_json_schema={'$ref':json_ref}returndefs_ref,ref_json_schema
It is not valid for a schema with a top-level $ref to have sibling keys.
During our own schema generation, we treat sibling keys as overrides to the referenced schema,
but this is not how the official JSON schema spec works.
Because of this, we first remove any sibling keys that are redundant with the referenced schema, then if
any remain, we transform the schema from a top-level '$ref' to use allOf to move the $ref out of the top level.
(See bottom of https://swagger.io/docs/specification/using-ref/ for a reference about this behavior)
defhandle_ref_overrides(self,json_schema:JsonSchemaValue)->JsonSchemaValue:"""It is not valid for a schema with a top-level $ref to have sibling keys. During our own schema generation, we treat sibling keys as overrides to the referenced schema, but this is not how the official JSON schema spec works. Because of this, we first remove any sibling keys that are redundant with the referenced schema, then if any remain, we transform the schema from a top-level '$ref' to use allOf to move the $ref out of the top level. (See bottom of https://swagger.io/docs/specification/using-ref/ for a reference about this behavior) """if'$ref'injson_schema:# prevent modifications to the input; this copy may be safe to drop if there is significant overheadjson_schema=json_schema.copy()referenced_json_schema=self.get_schema_from_definitions(JsonRef(json_schema['$ref']))ifreferenced_json_schemaisNone:# This can happen when building schemas for models with not-yet-defined references.# It may be a good idea to do a recursive pass at the end of the generation to remove# any redundant override keys.iflen(json_schema)>1:# Make it an allOf to at least resolve the sibling keys issuejson_schema=json_schema.copy()json_schema.setdefault('allOf',[])json_schema['allOf'].append({'$ref':json_schema['$ref']})deljson_schema['$ref']returnjson_schemafork,vinlist(json_schema.items()):ifk=='$ref':continueifkinreferenced_json_schemaandreferenced_json_schema[k]==v:deljson_schema[k]# redundant keyiflen(json_schema)>1:# There is a remaining "override" key, so we need to move $ref out of the top leveljson_ref=JsonRef(json_schema['$ref'])deljson_schema['$ref']assert'allOf'notinjson_schema# this should never happen, but just in casejson_schema['allOf']=[{'$ref':json_ref}]returnjson_schema
defencode_default(self,dft:Any)->Any:"""Encode a default value to a JSON-serializable value. This is used to encode default values for fields in the generated JSON schema. Args: dft: The default value to encode. Returns: The encoded default value. """from.type_adapterimportTypeAdapter,_type_has_configconfig=self._configtry:default=(dftif_type_has_config(type(dft))elseTypeAdapter(type(dft),config=config.config_dict).dump_python(dft,mode='json'))exceptPydanticSchemaGenerationError:raisepydantic_core.PydanticSerializationError(f'Unable to encode default value {dft}')returnpydantic_core.to_jsonable_python(default,timedelta_mode=config.ser_json_timedelta,bytes_mode=config.ser_json_bytes,)
Update the json_schema with the corresponding validations specified in the core_schema,
using the provided mapping to translate keys in core_schema to the appropriate keys for a JSON schema.
defupdate_with_validations(self,json_schema:JsonSchemaValue,core_schema:CoreSchema,mapping:dict[str,str])->None:"""Update the json_schema with the corresponding validations specified in the core_schema, using the provided mapping to translate keys in core_schema to the appropriate keys for a JSON schema. Args: json_schema: The JSON schema to update. core_schema: The core schema to get the validations from. mapping: A mapping from core_schema attribute names to the corresponding JSON schema attribute names. """forcore_key,json_schema_keyinmapping.items():ifcore_keyincore_schema:json_schema[json_schema_key]=core_schema[core_key]
defget_json_ref_counts(self,json_schema:JsonSchemaValue)->dict[JsonRef,int]:"""Get all values corresponding to the key '$ref' anywhere in the json_schema."""json_refs:dict[JsonRef,int]=Counter()def_add_json_refs(schema:Any)->None:ifisinstance(schema,dict):if'$ref'inschema:json_ref=JsonRef(schema['$ref'])ifnotisinstance(json_ref,str):return# in this case, '$ref' might have been the name of a propertyalready_visited=json_refinjson_refsjson_refs[json_ref]+=1ifalready_visited:return# prevent recursion on a definition that was already visiteddefs_ref=self.json_to_defs_refs[json_ref]ifdefs_refinself._core_defs_invalid_for_json_schema:raiseself._core_defs_invalid_for_json_schema[defs_ref]_add_json_refs(self.definitions[defs_ref])forvinschema.values():_add_json_refs(v)elifisinstance(schema,list):forvinschema:_add_json_refs(v)_add_json_refs(json_schema)returnjson_refs
This method simply emits PydanticJsonSchemaWarnings based on handling in the warning_message method.
Source code in pydantic/json_schema.py
21752176217721782179
defemit_warning(self,kind:JsonSchemaWarningKind,detail:str)->None:"""This method simply emits PydanticJsonSchemaWarnings based on handling in the `warning_message` method."""message=self.render_warning_message(kind,detail)ifmessageisnotNone:warnings.warn(message,PydanticJsonSchemaWarning)
This method is responsible for ignoring warnings as desired, and for formatting the warning messages.
You can override the value of ignored_warning_kinds in a subclass of GenerateJsonSchema
to modify what warnings are generated. If you want more control, you can override this method;
just return None in situations where you don't want warnings to be emitted.
defrender_warning_message(self,kind:JsonSchemaWarningKind,detail:str)->str|None:"""This method is responsible for ignoring warnings as desired, and for formatting the warning messages. You can override the value of `ignored_warning_kinds` in a subclass of GenerateJsonSchema to modify what warnings are generated. If you want more control, you can override this method; just return None in situations where you don't want warnings to be emitted. Args: kind: The kind of warning to render. It can be one of the following: - 'skipped-choice': A choice field was skipped because it had no valid choices. - 'non-serializable-default': A default value was skipped because it was not JSON-serializable. detail: A string with additional details about the warning. Returns: The formatted warning message, or `None` if no warning should be emitted. """ifkindinself.ignored_warning_kinds:returnNonereturnf'{detail} [{kind}]'
Add this as an annotation on a field to override the (base) JSON schema that would be generated for that field.
This provides a way to set a JSON schema for types that would otherwise raise errors when producing a JSON schema,
such as Callable, or types that have an is-instance core schema, without needing to go so far as creating a
custom subclass of pydantic.json_schema.GenerateJsonSchema.
Note that any modifications to the schema that would normally be made (such as setting the title for model fields)
will still be performed.
If mode is set this will only apply to that schema generation mode, allowing you
to set different json schemas for validation and serialization.
@deprecated('`update_json_schema` is deprecated, use a simple `my_dict.update(update_dict)` call instead.',category=None,)defupdate_json_schema(schema:JsonSchemaValue,updates:dict[str,Any])->JsonSchemaValue:"""Update a JSON schema in-place by providing a dictionary of updates. This function sets the provided key-value pairs in the schema and returns the updated schema. Args: schema: The JSON schema to update. updates: A dictionary of key-value pairs to set in the schema. Returns: The updated JSON schema. """schema.update(updates)returnschema
defmodel_json_schema(cls:type[BaseModel]|type[PydanticDataclass],by_alias:bool=True,ref_template:str=DEFAULT_REF_TEMPLATE,schema_generator:type[GenerateJsonSchema]=GenerateJsonSchema,mode:JsonSchemaMode='validation',)->dict[str,Any]:"""Utility function to generate a JSON Schema for a model. Args: cls: The model class to generate a JSON Schema for. by_alias: If `True` (the default), fields will be serialized according to their alias. If `False`, fields will be serialized according to their attribute name. ref_template: The template to use for generating JSON Schema references. schema_generator: The class to use for generating the JSON Schema. mode: The mode to use for generating the JSON Schema. It can be one of the following: - 'validation': Generate a JSON Schema for validating data. - 'serialization': Generate a JSON Schema for serializing data. Returns: The generated JSON Schema. """from.mainimportBaseModelschema_generator_instance=schema_generator(by_alias=by_alias,ref_template=ref_template)ifisinstance(cls.__pydantic_core_schema__,_mock_val_ser.MockCoreSchema):cls.__pydantic_core_schema__.rebuild()ifclsisBaseModel:raiseAttributeError('model_json_schema() must be called on a subclass of BaseModel, not BaseModel itself.')assertnotisinstance(cls.__pydantic_core_schema__,_mock_val_ser.MockCoreSchema),'this is a bug! please report it'returnschema_generator_instance.generate(cls.__pydantic_core_schema__,mode=mode)
A tuple where:
- The first element is a dictionary whose keys are tuples of JSON schema key type and JSON mode, and
whose values are the JSON schema corresponding to that pair of inputs. (These schemas may have
JsonRef references to definitions that are defined in the second returned element.)
- The second element is a JSON schema containing all definitions referenced in the first returned
element, along with the optional title and description keys.
defmodels_json_schema(models:Sequence[tuple[type[BaseModel]|type[PydanticDataclass],JsonSchemaMode]],*,by_alias:bool=True,title:str|None=None,description:str|None=None,ref_template:str=DEFAULT_REF_TEMPLATE,schema_generator:type[GenerateJsonSchema]=GenerateJsonSchema,)->tuple[dict[tuple[type[BaseModel]|type[PydanticDataclass],JsonSchemaMode],JsonSchemaValue],JsonSchemaValue]:"""Utility function to generate a JSON Schema for multiple models. Args: models: A sequence of tuples of the form (model, mode). by_alias: Whether field aliases should be used as keys in the generated JSON Schema. title: The title of the generated JSON Schema. description: The description of the generated JSON Schema. ref_template: The reference template to use for generating JSON Schema references. schema_generator: The schema generator to use for generating the JSON Schema. Returns: A tuple where: - The first element is a dictionary whose keys are tuples of JSON schema key type and JSON mode, and whose values are the JSON schema corresponding to that pair of inputs. (These schemas may have JsonRef references to definitions that are defined in the second returned element.) - The second element is a JSON schema containing all definitions referenced in the first returned element, along with the optional title and description keys. """forcls,_inmodels:ifisinstance(cls.__pydantic_core_schema__,_mock_val_ser.MockCoreSchema):cls.__pydantic_core_schema__.rebuild()instance=schema_generator(by_alias=by_alias,ref_template=ref_template)inputs:list[tuple[type[BaseModel]|type[PydanticDataclass],JsonSchemaMode,CoreSchema]]=[(m,mode,m.__pydantic_core_schema__)form,modeinmodels]json_schemas_map,definitions=instance.generate_definitions(inputs)json_schema:dict[str,Any]={}ifdefinitions:json_schema['$defs']=definitionsiftitle:json_schema['title']=titleifdescription:json_schema['description']=descriptionreturnjson_schemas_map,json_schema